Redo the FloatRegFile using unsigned integers

Edit the convert_and_round function which access FloatRegFile

arch/isa_parser.py:
    recognize when we are writing a 'uint64_t' FloatReg and set the width appropriately
arch/mips/isa/decoder.isa:
    Send a 'float' to the convert function instead of a unsigned word. Do this so we dont have to worry about the
    bit manipulation ourselves. We can just concern ourselves with values.

    Use unsigned double to get movd...
arch/mips/isa/formats/fp.isa:
    float debug statement
arch/mips/isa_traits.cc:
    add different versions of convert_and_round functions
arch/mips/isa_traits.hh:
    Use an array of uint32_t unsigned integers to represent the Floating Point Regfile
configs/test/hello_mips:
    basic FP program
cpu/simple/cpu.hh:
    spacing

--HG--
extra : convert_revision : a6fca91ad6365c83025f1131d71fa1b8ee76d7bc
This commit is contained in:
Korey Sewell 2006-05-02 20:05:16 -04:00
parent 2d077df1a0
commit 97429d8eee
7 changed files with 124 additions and 55 deletions

View file

@ -1251,6 +1251,9 @@ class FloatRegOperand(Operand):
elif (self.ctype == 'double'):
width = 64
func = 'setFloatReg'
elif (self.ctype == 'uint64_t'):
func = 'setFloatRegBits'
width = 64
else:
func = 'setFloatRegBits'
final_ctype = 'uint%d_t' % self.dflt_size

View file

@ -397,7 +397,7 @@ decode OPCODE_HI default Unknown::unknown() {
format FloatOp {
0x0: mfc1 ({{ Rt.uw = Fs.uw<31:0>; }});
0x3: mfhc1({{ Rt.uw = Fs.ud<63:32>;}});
0x4: mtc1 ({{ Fs.uw = Rt.uw; }});
0x4: mtc1 ({{ Fs.uw = Rt.uw; }});
0x7: mthc1({{
uint64_t fs_hi = Rt.ud << 32;
uint64_t fs_lo = Fs.ud & 0x0000FFFF;
@ -572,7 +572,7 @@ decode OPCODE_HI default Unknown::unknown() {
format FloatOp {
0x1: cvt_d_s({{
int rnd_mode = xc->readMiscReg(FCSR) & 0x03;
Fd.ud = convert_and_round(Fs.uw, SINGLE_TO_DOUBLE, rnd_mode);
Fd.ud = convert_and_round(Fs.sf, SINGLE_TO_DOUBLE, rnd_mode);
}});
0x4: cvt_w_s({{
@ -605,7 +605,7 @@ decode OPCODE_HI default Unknown::unknown() {
0x3: divd({{ Fd.df = Fs.df / Ft.df;}});
0x4: sqrtd({{ Fd.df = sqrt(Fs.df);}});
0x5: absd({{ Fd.df = fabs(Fs.df);}});
0x6: movd({{ Fd.df = Fs.df;}});
0x6: movd({{ Fd.ud = Fs.ud;}});
0x7: negd({{ Fd.df = -1 * Fs.df;}});
}
}

View file

@ -32,6 +32,7 @@ output decoder {{
// Primary format for float operate instructions:
def format FloatOp(code, *flags) {{
code = 'std::cout << "Floating Point Op" << std::endl;\n' + code
iop = InstObjParams(name, Name, 'MipsStaticInst', CodeBlock(code), flags)
header_output = BasicDeclare.subst(iop)
decoder_output = BasicConstructor.subst(iop)
@ -41,6 +42,7 @@ def format FloatOp(code, *flags) {{
// Primary format for float64 operate instructions:
def format Float64Op(code, *flags) {{
code = 'std::cout << "Floating Point 64" << std::endl;\n' + code
iop = InstObjParams(name, Name, 'MipsStaticInst', CodeBlock(code), flags)
header_output = BasicDeclare.subst(iop)
decoder_output = BasicConstructor.subst(iop)

View file

@ -30,25 +30,95 @@
#include "config/full_system.hh"
#include "cpu/static_inst.hh"
#include "sim/serialize.hh"
#include "base/bitfield.hh"
using namespace MipsISA;
using namespace std;
uint64_t
MipsISA::convert_and_round(uint64_t fp_val, ConvertType cvt_type, int rnd_mode)
MipsISA::convert_and_round(uint32_t fp_val, ConvertType cvt_type, int rnd_mode)
{
uint64_t ret_val = 0;
switch (cvt_type)
{
case SINGLE_TO_DOUBLE:
uint64_t single_sign = fp_val & 0x80000000;
break;
uint64_t single_exp = (fp_val & 0x7F800000) >> 22;
single_exp -= 127;
uint64_t single_mantissa = fp_val & 0x007FFFFF;
uint64_t double_exp = single_exp + 1023;
double_exp = double_exp << 51;
uint64_t double_val = single_sign << 63 | double_exp | single_mantissa;
return double_val;
default:
panic("Invalid Floating Point Conversion Type (%d) being used.\n",cvt_type);
return ret_val;
}
}
return ret_val;
uint64_t
MipsISA::convert_and_round(uint64_t fp_val, ConvertType cvt_type, int rnd_mode)
{
uint64_t ret_val = 0;
switch (cvt_type)
{
case SINGLE_TO_DOUBLE:
uint64_t single_sign = fp_val & 0x80000000;
uint64_t single_exp = (fp_val & 0x7F800000) >> 22;
single_exp -= 127;
uint64_t single_mantissa = fp_val & 0x007FFFFF;
uint64_t double_exp = single_exp + 1023;
double_exp = double_exp << 51;
uint64_t double_val = single_sign << 63 | double_exp | single_mantissa;
return double_val;
default:
panic("Invalid Floating Point Conversion Type (%d) being used.\n",cvt_type);
return ret_val;
}
}
uint64_t
MipsISA::convert_and_round(float fp_val, ConvertType cvt_type, int rnd_mode)
{
void * ptr = &fp_val;
uint32_t fp_bits = * (uint32_t *) ptr;
cout << "Converting " << fp_val << " (" << hex << fp_bits << ") " << endl;
uint64_t ret_val = 0;
switch (cvt_type)
{
case SINGLE_TO_DOUBLE:
double double_val = fp_val;
void *double_ptr = &double_val;
uint64_t dp_bits = *(uint64_t *) double_ptr ;
cout << "To " << double_val << " (" << hex << dp_bits << ") " << endl;
double_ptr = &dp_bits;
cout << "Testing: " << *(double *) double_ptr << endl;
return dp_bits;
default:
panic("Invalid Floating Point Conversion Type (%d) being used.\n",cvt_type);
return ret_val;
}
}
void

View file

@ -189,21 +189,24 @@ namespace MipsISA
};
typedef double FloatReg;
typedef float FloatReg;
typedef uint32_t FloatReg32;
typedef uint64_t FloatReg64;
typedef uint64_t FloatRegBits;
// const uint64_t hi_mask64 = 0xFFFFFFFF00000000;
//const uint64_t lo_mask64 = 0x00000000FFFFFFFF;
const int SingleWidth = 32;
const int SingleBytes = SingleWidth / 4;
const int SingleBytes = 4;
const int DoubleWidth = 64;
const int DoubleBytes = DoubleWidth / 4;
const int DoubleBytes = 8;
const int QuadWidth = 128;
const int QuadBytes = QuadWidth / 4;
const int FloatRegSize = SingleWidth / SingleBytes;
const int DoubleRegSize = FloatRegSize * 2;
class FloatRegFile
{
protected:
@ -211,31 +214,27 @@ namespace MipsISA
//Since the floating point registers overlap each other,
//A generic storage space is used. The float to be returned is
//pulled from the appropriate section of this region.
char regSpace[FloatRegSize * NumFloatRegs];
//char regSpace[SingleBytes * NumFloatRegs];
FloatReg32 regs[NumFloatRegs];
public:
void clear()
{
bzero(regSpace, sizeof(regSpace));
bzero(regs, sizeof(regs));
}
FloatReg readReg(int floatReg, int width)
double readReg(int floatReg, int width)
{
//In each of these cases, we have to copy the value into a temporary
//variable. This is because we may otherwise try to access an
//unaligned portion of memory.
switch(width)
{
case SingleWidth:
float result32;
memcpy(&result32, regSpace + 4 * floatReg, FloatRegSize);
return htog(result32);
void *float_ptr = &regs[floatReg];
return *(float *) float_ptr;
case DoubleWidth:
double result64;
memcpy(&result64, regSpace + 4 * floatReg, DoubleRegSize);
return htog(result64);
void *double_ptr = &regs[floatReg];
return *(double *) double_ptr;
default:
panic("Attempted to read a %d bit floating point register!", width);
@ -244,20 +243,17 @@ namespace MipsISA
FloatRegBits readRegBits(int floatReg, int width)
{
//In each of these cases, we have to copy the value into a temporary
//variable. This is because we may otherwise try to access an
//unaligned portion of memory.
using namespace std;
switch(width)
{
case SingleWidth:
uint32_t result32;
memcpy(&result32, regSpace + 4 * floatReg, FloatRegSize);
return htog(result32);
return regs[floatReg];
case DoubleWidth:
uint64_t result64;
memcpy(&result64, regSpace + 4 * floatReg, DoubleRegSize);
return htog(result64);
cout << hex << "Combining " << regs[floatReg + 1] << " & " << regs[floatReg + 1] << endl;
cout << hex << "Returning " << ((FloatReg64)regs[floatReg] << 32 | regs[floatReg + 1]) << endl;
return (FloatReg64)regs[floatReg] << 32 | regs[floatReg + 1];
default:
panic("Attempted to read a %d bit floating point register!", width);
@ -266,47 +262,45 @@ namespace MipsISA
Fault setReg(int floatReg, const FloatReg &val, int width)
{
//In each of these cases, we have to copy the value into a temporary
//variable. This is because we may otherwise try to access an
//unaligned portion of memory.
switch(width)
{
case SingleWidth:
uint32_t result32;
result32 = gtoh((uint32_t)val);
memcpy(regSpace + 4 * floatReg, &result32, FloatRegSize);
float temp = val;
void *float_ptr = &temp;
regs[floatReg] = *(FloatReg32 *) float_ptr;
break;
case DoubleWidth:
uint64_t result64;
result64 = gtoh((uint64_t)val);
memcpy(regSpace + 4 * floatReg, &result64, DoubleRegSize);
const void *double_ptr = &val;
FloatReg64 temp_double = *(FloatReg64 *) double_ptr;
regs[floatReg] = temp_double >> 32;
regs[floatReg + 1] = temp_double;
break;
default:
panic("Attempted to read a %d bit floating point register!", width);
}
return NoFault;
}
Fault setRegBits(int floatReg, const FloatRegBits &val, int width)
{
//In each of these cases, we have to copy the value into a temporary
//variable. This is because we may otherwise try to access an
//unaligned portion of memory.
using namespace std;
switch(width)
{
case SingleWidth:
uint32_t result32;
result32 = gtoh((uint32_t)val);
memcpy(regSpace + 4 * floatReg, &result32, FloatRegSize);
regs[floatReg] = val;
break;
case DoubleWidth:
uint64_t result64;
result64 = gtoh((uint64_t)val);
memcpy(regSpace + 4 * floatReg, &result64, DoubleRegSize);
cout << hex << "Setting val: " << val << endl;
regs[floatReg] = val >> 32;
regs[floatReg + 1] = val;
cout << dec << "f" << floatReg << ": " << hex<< readRegBits(floatReg,32) << endl;
cout << dec << "f" << floatReg + 1 << ": " << hex << readRegBits(floatReg+1,32) << endl;
break;
default:
@ -348,8 +342,9 @@ namespace MipsISA
RND_NEAREST
};
uint64_t convert_and_round(uint32_t fp_val,ConvertType cvt_type, int rnd_mode = 0);
uint64_t convert_and_round(uint64_t fp_val,ConvertType cvt_type, int rnd_mode = 0);
uint64_t convert_and_round(float fp_val,ConvertType cvt_type, int rnd_mode = 0);
void copyRegs(ExecContext *src, ExecContext *dest);
@ -637,7 +632,6 @@ extern const Addr PageOffset;
return miscRegFile.setRegWithEffect(miscReg, val, xc);
}
FloatReg readFloatReg(int floatReg)
{
return floatRegFile.readReg(floatReg,SingleWidth);

Binary file not shown.

View file

@ -369,7 +369,7 @@ class SimpleCPU : public BaseCPU
}
void setFloatRegBits(const StaticInst *si, int idx,
FloatRegBits val, int width)
FloatRegBits val, int width)
{
int reg_idx = si->destRegIdx(idx) - TheISA::FP_Base_DepTag;
cpuXC->setFloatRegBits(reg_idx, val, width);