Compare commits
No commits in common. "1fb7e8fcea2639852c03d53eb32fce8803e9f68a" and "62768bf81e74ef35ca4eb0ef622033861a898d5e" have entirely different histories.
1fb7e8fcea
...
62768bf81e
@ -319,8 +319,7 @@ struct riscv_hart_common {
|
|||||||
uint64_t tohost = tohost_dflt;
|
uint64_t tohost = tohost_dflt;
|
||||||
uint64_t fromhost = fromhost_dflt;
|
uint64_t fromhost = fromhost_dflt;
|
||||||
|
|
||||||
bool read_elf_file(std::string name, uint8_t expected_elf_class,
|
bool read_elf_file(std::string name, uint8_t expected_elf_class, std::function<iss::status(uint64_t, uint64_t, const uint8_t* const)> cb) {
|
||||||
std::function<iss::status(uint64_t, uint64_t, const uint8_t* const)> cb) {
|
|
||||||
// Create elfio reader
|
// Create elfio reader
|
||||||
ELFIO::elfio reader;
|
ELFIO::elfio reader;
|
||||||
// Load ELF data
|
// Load ELF data
|
||||||
@ -328,12 +327,12 @@ struct riscv_hart_common {
|
|||||||
// check elf properties
|
// check elf properties
|
||||||
if(reader.get_class() != expected_elf_class)
|
if(reader.get_class() != expected_elf_class)
|
||||||
return false;
|
return false;
|
||||||
if(reader.get_type() != ELFIO::ET_EXEC)
|
if(reader.get_type() != ET_EXEC)
|
||||||
return false;
|
return false;
|
||||||
if(reader.get_machine() != ELFIO::EM_RISCV)
|
if(reader.get_machine() != EM_RISCV)
|
||||||
return false;
|
return false;
|
||||||
entry_address = reader.get_entry();
|
entry_address = reader.get_entry();
|
||||||
for(const auto& pseg : reader.segments) {
|
for(const auto pseg : reader.segments) {
|
||||||
const auto fsize = pseg->get_file_size(); // 0x42c/0x0
|
const auto fsize = pseg->get_file_size(); // 0x42c/0x0
|
||||||
const auto seg_data = pseg->get_data();
|
const auto seg_data = pseg->get_data();
|
||||||
const auto type = pseg->get_type();
|
const auto type = pseg->get_type();
|
||||||
@ -344,7 +343,7 @@ struct riscv_hart_common {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
const auto sym_sec = reader.sections[".symtab"];
|
const auto sym_sec = reader.sections[".symtab"];
|
||||||
if(ELFIO::SHT_SYMTAB == sym_sec->get_type() || ELFIO::SHT_DYNSYM == sym_sec->get_type()) {
|
if(SHT_SYMTAB == sym_sec->get_type() || SHT_DYNSYM == sym_sec->get_type()) {
|
||||||
ELFIO::symbol_section_accessor symbols(reader, sym_sec);
|
ELFIO::symbol_section_accessor symbols(reader, sym_sec);
|
||||||
auto sym_no = symbols.get_symbols_num();
|
auto sym_no = symbols.get_symbols_num();
|
||||||
std::string name;
|
std::string name;
|
||||||
|
@ -572,7 +572,7 @@ riscv_hart_m_p<BASE, FEAT, LOGCAT>::riscv_hart_m_p(feature_config cfg)
|
|||||||
|
|
||||||
template <typename BASE, features_e FEAT, typename LOGCAT>
|
template <typename BASE, features_e FEAT, typename LOGCAT>
|
||||||
std::pair<uint64_t, bool> riscv_hart_m_p<BASE, FEAT, LOGCAT>::load_file(std::string name, int type) {
|
std::pair<uint64_t, bool> riscv_hart_m_p<BASE, FEAT, LOGCAT>::load_file(std::string name, int type) {
|
||||||
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFIO::ELFCLASS32 : ELFIO::ELFCLASS64,
|
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFCLASS32 : ELFCLASS64,
|
||||||
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
||||||
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
||||||
data);
|
data);
|
||||||
@ -706,7 +706,7 @@ iss::status riscv_hart_m_p<BASE, FEAT, LOGCAT>::write(const address_type type, c
|
|||||||
<< std::hex << addr;
|
<< std::hex << addr;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
CPPLOG(TRACE) << prefix << "write of " << length << " bytes @addr 0x" << std::hex << addr;
|
CPPLOG(TRACE) << prefix << "write of " << length << " bytes @addr " << addr;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
try {
|
try {
|
||||||
@ -1326,8 +1326,7 @@ uint64_t riscv_hart_m_p<BASE, FEAT, LOGCAT>::enter_trap(uint64_t flags, uint64_t
|
|||||||
#endif
|
#endif
|
||||||
if((flags & 0xffffffff) != 0xffffffff)
|
if((flags & 0xffffffff) != 0xffffffff)
|
||||||
NSCLOG(INFO, LOGCAT) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
NSCLOG(INFO, LOGCAT) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
||||||
<< cause << ")"
|
<< cause << ")" << " at address " << buffer.data() << " occurred";
|
||||||
<< " at address " << buffer.data() << " occurred";
|
|
||||||
return this->reg.NEXT_PC;
|
return this->reg.NEXT_PC;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -555,7 +555,7 @@ riscv_hart_msu_vp<BASE>::riscv_hart_msu_vp()
|
|||||||
}
|
}
|
||||||
|
|
||||||
template <typename BASE> std::pair<uint64_t, bool> riscv_hart_msu_vp<BASE>::load_file(std::string name, int type) {
|
template <typename BASE> std::pair<uint64_t, bool> riscv_hart_msu_vp<BASE>::load_file(std::string name, int type) {
|
||||||
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFIO::ELFCLASS32 : ELFIO::ELFCLASS64,
|
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFCLASS32 : ELFCLASS64,
|
||||||
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
||||||
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
||||||
data);
|
data);
|
||||||
@ -1337,9 +1337,8 @@ template <typename BASE> uint64_t riscv_hart_msu_vp<BASE>::enter_trap(uint64_t f
|
|||||||
sprintf(buffer.data(), "0x%016lx", addr);
|
sprintf(buffer.data(), "0x%016lx", addr);
|
||||||
if((flags & 0xffffffff) != 0xffffffff)
|
if((flags & 0xffffffff) != 0xffffffff)
|
||||||
CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
||||||
<< cause << ")"
|
<< cause << ")" << " at address " << buffer.data() << " occurred, changing privilege level from "
|
||||||
<< " at address " << buffer.data() << " occurred, changing privilege level from " << lvl[cur_priv] << " to "
|
<< lvl[cur_priv] << " to " << lvl[new_priv];
|
||||||
<< lvl[new_priv];
|
|
||||||
// reset trap state
|
// reset trap state
|
||||||
this->reg.PRIV = new_priv;
|
this->reg.PRIV = new_priv;
|
||||||
this->reg.trap_state = 0;
|
this->reg.trap_state = 0;
|
||||||
|
@ -649,7 +649,7 @@ riscv_hart_mu_p<BASE, FEAT, LOGCAT>::riscv_hart_mu_p(feature_config cfg)
|
|||||||
|
|
||||||
template <typename BASE, features_e FEAT, typename LOGCAT>
|
template <typename BASE, features_e FEAT, typename LOGCAT>
|
||||||
std::pair<uint64_t, bool> riscv_hart_mu_p<BASE, FEAT, LOGCAT>::load_file(std::string name, int type) {
|
std::pair<uint64_t, bool> riscv_hart_mu_p<BASE, FEAT, LOGCAT>::load_file(std::string name, int type) {
|
||||||
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFIO::ELFCLASS32 : ELFIO::ELFCLASS64,
|
if(read_elf_file(name, sizeof(reg_t) == 4 ? ELFCLASS32 : ELFCLASS64,
|
||||||
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
[this](uint64_t addr, uint64_t size, const uint8_t* const data) -> iss::status {
|
||||||
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
return this->write(iss::address_type::PHYSICAL, iss::access_type::DEBUG_WRITE, traits<BASE>::MEM, addr, size,
|
||||||
data);
|
data);
|
||||||
@ -1565,9 +1565,8 @@ uint64_t riscv_hart_mu_p<BASE, FEAT, LOGCAT>::enter_trap(uint64_t flags, uint64_
|
|||||||
#endif
|
#endif
|
||||||
if((flags & 0xffffffff) != 0xffffffff)
|
if((flags & 0xffffffff) != 0xffffffff)
|
||||||
CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" << (trap_id ? irq_str[cause] : trap_str[cause]) << "' ("
|
||||||
<< cause << ")"
|
<< cause << ")" << " at address " << buffer.data() << " occurred, changing privilege level from "
|
||||||
<< " at address " << buffer.data() << " occurred, changing privilege level from " << lvl[this->reg.PRIV]
|
<< lvl[this->reg.PRIV] << " to " << lvl[new_priv];
|
||||||
<< " to " << lvl[new_priv];
|
|
||||||
// reset trap state
|
// reset trap state
|
||||||
this->reg.PRIV = new_priv;
|
this->reg.PRIV = new_priv;
|
||||||
this->reg.trap_state = 0;
|
this->reg.trap_state = 0;
|
||||||
|
@ -203,8 +203,8 @@ uint32_t fclass_s(uint32_t v1) {
|
|||||||
uA.f = a;
|
uA.f = a;
|
||||||
uiA = uA.ui;
|
uiA = uA.ui;
|
||||||
|
|
||||||
bool infOrNaN = expF32UI(uiA) == 0xFF;
|
uint_fast16_t infOrNaN = expF32UI(uiA) == 0xFF;
|
||||||
bool subnormalOrZero = expF32UI(uiA) == 0;
|
uint_fast16_t subnormalOrZero = expF32UI(uiA) == 0;
|
||||||
bool sign = signF32UI(uiA);
|
bool sign = signF32UI(uiA);
|
||||||
bool fracZero = fracF32UI(uiA) == 0;
|
bool fracZero = fracF32UI(uiA) == 0;
|
||||||
bool isNaN = isNaNF32UI(uiA);
|
bool isNaN = isNaNF32UI(uiA);
|
||||||
@ -217,13 +217,9 @@ uint32_t fclass_s(uint32_t v1) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint32_t fconv_d2f(uint64_t v1, uint8_t mode) {
|
uint32_t fconv_d2f(uint64_t v1, uint8_t mode) {
|
||||||
bool isNan = isNaNF64UI(v1);
|
|
||||||
bool isSNaN = softfloat_isSigNaNF64UI(v1);
|
|
||||||
softfloat_roundingMode = rmm_map.at(mode);
|
softfloat_roundingMode = rmm_map.at(mode);
|
||||||
softfloat_exceptionFlags = 0;
|
bool nan = (v1 & defaultNaNF64UI) == defaultNaNF64UI;
|
||||||
if(isNan) {
|
if(nan) {
|
||||||
if(isSNaN)
|
|
||||||
softfloat_raiseFlags(softfloat_flag_invalid);
|
|
||||||
return defaultNaNF32UI;
|
return defaultNaNF32UI;
|
||||||
} else {
|
} else {
|
||||||
float32_t res = f64_to_f32(float64_t{v1});
|
float32_t res = f64_to_f32(float64_t{v1});
|
||||||
@ -232,11 +228,11 @@ uint32_t fconv_d2f(uint64_t v1, uint8_t mode) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint64_t fconv_f2d(uint32_t v1, uint8_t mode) {
|
uint64_t fconv_f2d(uint32_t v1, uint8_t mode) {
|
||||||
bool infOrNaN = expF32UI(v1) == 0xFF;
|
bool nan = (v1 & defaultNaNF32UI) == defaultNaNF32UI;
|
||||||
bool subnormalOrZero = expF32UI(v1) == 0;
|
if(nan) {
|
||||||
if(infOrNaN || subnormalOrZero) {
|
|
||||||
return defaultNaNF64UI;
|
return defaultNaNF64UI;
|
||||||
} else {
|
} else {
|
||||||
|
softfloat_roundingMode = rmm_map.at(mode);
|
||||||
float64_t res = f32_to_f64(float32_t{v1});
|
float64_t res = f32_to_f64(float32_t{v1});
|
||||||
return res.v;
|
return res.v;
|
||||||
}
|
}
|
||||||
@ -316,23 +312,22 @@ uint64_t fcmp_d(uint64_t v1, uint64_t v2, uint32_t op) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint64_t fcvt_d(uint64_t v1, uint32_t op, uint8_t mode) {
|
uint64_t fcvt_d(uint64_t v1, uint32_t op, uint8_t mode) {
|
||||||
|
|
||||||
float64_t v1f{v1};
|
float64_t v1f{v1};
|
||||||
softfloat_exceptionFlags = 0;
|
softfloat_exceptionFlags = 0;
|
||||||
float64_t r;
|
float64_t r;
|
||||||
switch(op) {
|
switch(op) {
|
||||||
case 0: { // l from d
|
case 0: { // l->d, fp to int32
|
||||||
int64_t res = f64_to_i64(v1f, rmm_map.at(mode), true);
|
int64_t res = f64_to_i64(v1f, rmm_map.at(mode), true);
|
||||||
return (uint64_t)res;
|
return (uint64_t)res;
|
||||||
}
|
}
|
||||||
case 1: { // lu from d
|
case 1: { // lu->s
|
||||||
uint64_t res = f64_to_ui64(v1f, rmm_map.at(mode), true);
|
uint64_t res = f64_to_ui64(v1f, rmm_map.at(mode), true);
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
case 2: // d from l
|
case 2: // s->l
|
||||||
r = i64_to_f64(v1);
|
r = i64_to_f64(v1);
|
||||||
return r.v;
|
return r.v;
|
||||||
case 3: // d from lu
|
case 3: // s->lu
|
||||||
r = ui64_to_f64(v1);
|
r = ui64_to_f64(v1);
|
||||||
return r.v;
|
return r.v;
|
||||||
}
|
}
|
||||||
@ -340,24 +335,12 @@ uint64_t fcvt_d(uint64_t v1, uint32_t op, uint8_t mode) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint64_t fmadd_d(uint64_t v1, uint64_t v2, uint64_t v3, uint32_t op, uint8_t mode) {
|
uint64_t fmadd_d(uint64_t v1, uint64_t v2, uint64_t v3, uint32_t op, uint8_t mode) {
|
||||||
uint64_t F64_SIGN = 1ULL << 63;
|
// op should be {softfloat_mulAdd_subProd(2), softfloat_mulAdd_subC(1)}
|
||||||
switch(op) {
|
|
||||||
case 0: // FMADD_D
|
|
||||||
break;
|
|
||||||
case 1: // FMSUB_D
|
|
||||||
v3 ^= F64_SIGN;
|
|
||||||
break;
|
|
||||||
case 2: // FNMADD_D
|
|
||||||
v1 ^= F64_SIGN;
|
|
||||||
v3 ^= F64_SIGN;
|
|
||||||
break;
|
|
||||||
case 3: // FNMSUB_D
|
|
||||||
v1 ^= F64_SIGN;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
softfloat_roundingMode = rmm_map.at(mode);
|
softfloat_roundingMode = rmm_map.at(mode);
|
||||||
softfloat_exceptionFlags = 0;
|
softfloat_exceptionFlags = 0;
|
||||||
float64_t res = softfloat_mulAddF64(v1, v2, v3, 0);
|
float64_t res = softfloat_mulAddF64(v1, v2, v3, op & 0x1);
|
||||||
|
if(op > 1)
|
||||||
|
res.v ^= 1ULL << 63;
|
||||||
return res.v;
|
return res.v;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -393,8 +376,8 @@ uint64_t fclass_d(uint64_t v1) {
|
|||||||
uA.f = a;
|
uA.f = a;
|
||||||
uiA = uA.ui;
|
uiA = uA.ui;
|
||||||
|
|
||||||
bool infOrNaN = expF64UI(uiA) == 0x7FF;
|
uint_fast16_t infOrNaN = expF64UI(uiA) == 0x7FF;
|
||||||
bool subnormalOrZero = expF64UI(uiA) == 0;
|
uint_fast16_t subnormalOrZero = expF64UI(uiA) == 0;
|
||||||
bool sign = signF64UI(uiA);
|
bool sign = signF64UI(uiA);
|
||||||
bool fracZero = fracF64UI(uiA) == 0;
|
bool fracZero = fracF64UI(uiA) == 0;
|
||||||
bool isNaN = isNaNF64UI(uiA);
|
bool isNaN = isNaNF64UI(uiA);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user