Go to the documentation of this file.
28 #ifndef __DEV_REG_BANK_HH__
29 #define __DEV_REG_BANK_HH__
35 #include <initializer_list>
294 template <ByteOrder BankByteOrder>
299 template <
typename Data>
300 static constexpr Data
303 return value & bitmask;
306 template <
typename Data>
307 static constexpr Data
327 virtual const std::string &
name()
const {
return _name; }
333 virtual void read(
void *buf) = 0;
334 virtual void read(
void *buf, off_t
offset,
size_t bytes) = 0;
337 virtual void write(
const void *buf) = 0;
338 virtual void write(
const void *buf, off_t
offset,
size_t bytes) = 0;
341 virtual void serialize(std::ostream &
os)
const = 0;
350 const std::string &new_name,
size_t new_size) :
354 virtual void fill(
void *buf, off_t
offset,
size_t bytes) = 0;
358 void write(
const void *buf)
override {}
359 void write(
const void *buf, off_t
offset,
size_t bytes)
override {}
396 memset(buf, 0xff, bytes);
412 RegisterBuf(
const std::string &new_name,
void *ptr,
size_t bytes) :
438 template <
int BufBytes>
453 for (
int i = 1;
i < BufBytes;
i++) {
463 std::istringstream
is(
s);
467 tokens.push_back(
token);
469 if (tokens.size() != BufBytes) {
470 warn(
"Size mismatch unserialing %s, expected %d, got %d",
471 this->
name(), BufBytes, tokens.size());
475 for (
int i = 0;
i < BufBytes;
i++) {
484 template <
typename Data, ByteOrder RegByteOrder=BankByteOrder>
496 void (
This &
reg,
const Data &value,
int first,
int last)>;
525 reg._writer(
reg, writeWithMask<Data>(
reg._reader(
reg), value,
532 switch (RegByteOrder) {
535 case ByteOrder::little:
538 panic(
"Unrecognized byte order %d.", (
unsigned)RegByteOrder);
545 switch (RegByteOrder) {
548 case ByteOrder::little:
551 panic(
"Unrecognized byte order %d.", (
unsigned)RegByteOrder);
567 constexpr
Register(
const std::string &new_name,
const Data &new_data) :
571 const Data &&new_data) :
595 template <
class Parent,
class... Args>
597 reader(Parent *parent, Data (Parent::*nr)(Args... args))
599 auto wrapper = [parent, nr](Args&&... args) -> Data {
600 return (parent->*nr)(std::forward<Args>(args)...);
610 template <
class Parent,
class... Args>
612 writer(Parent *parent,
void (Parent::*
nw)(Args... args))
614 auto wrapper = [parent,
nw](Args&&... args) {
615 (parent->*
nw)(std::forward<Args>(args)...);
636 template <
class Parent,
class... Args>
640 auto wrapper = [parent, nr](Args&&... args) -> Data {
641 return (parent->*nr)(std::forward<Args>(args)...);
651 template <
class Parent,
class... Args>
655 auto wrapper = [parent,
nw](Args&&... args) {
656 return (parent->*
nw)(std::forward<Args>(args)...);
675 update(
const Data &new_data,
const Data &bitmask)
697 memcpy(buf, (uint8_t *)&
data,
sizeof(
data));
705 const off_t host_off = (RegByteOrder != ByteOrder::little) ?
708 const int first = (host_off + bytes) * 8 - 1;
709 const int last = host_off * 8;
712 memcpy(buf, (uint8_t *)&
data +
offset, bytes);
720 memcpy((uint8_t *)&
data, buf,
sizeof(
data));
729 memcpy((uint8_t *)&
data +
offset, buf, bytes);
735 const off_t host_off = (RegByteOrder != ByteOrder::little) ?
738 const int first = (host_off + bytes) * 8 - 1;
739 const int last = host_off * 8;
758 std::map<Addr, std::reference_wrapper<RegisterBase>>
_offsetMap;
788 std::initializer_list<std::reference_wrapper<RegisterBase>> regs)
790 panic_if(regs.size() == 0,
"Adding an empty list of registers to %s?",
792 for (
auto &
reg: regs) {
807 uint8_t *ptr = (uint8_t *)buf;
812 "Out of bounds read in register bank %s, address %#x, size %d.",
819 if (it->first <
addr) {
820 RegisterBase &
reg = it->second.get();
824 const off_t reg_off =
addr - it->first;
825 const size_t reg_bytes = std::min(
reg.size() - reg_off,
829 reg.read(ptr, reg_off, reg_bytes);
839 RegisterBase &
reg = it->second.get();
841 const size_t reg_size =
reg.size();
842 const size_t remaining = bytes - done;
844 if (remaining == reg_size) {
846 reg.read(ptr + done);
848 }
else if (remaining > reg_size) {
850 reg.read(ptr + done);
855 reg.read(ptr + done, 0, remaining);
864 const uint8_t *ptr = (
const uint8_t *)buf;
869 "Out of bounds write in register bank %s, address %#x, size %d.",
876 if (it->first <
addr) {
877 RegisterBase &
reg = it->second.get();
881 const off_t reg_off =
addr - it->first;
882 const size_t reg_bytes = std::min(
reg.size() - reg_off,
886 reg.write(ptr, reg_off, reg_bytes);
896 RegisterBase &
reg = it->second.get();
898 const size_t reg_size =
reg.size();
899 const size_t remaining = bytes - done;
901 if (remaining == reg_size) {
903 reg.write(ptr + done);
905 }
else if (remaining > reg_size) {
907 reg.write(ptr + done);
912 reg.write(ptr + done, 0, remaining);
925 typename RegisterBankBase::RegisterBaseBase, T>::value>>
930 return value.unserialize(
s);
936 typename RegisterBankBase::RegisterBaseBase, T>::value>>
945 #endif // __DEV_REG_BANK_HH__
constexpr This & partialWriter(Parent *parent, void(Parent::*nw)(Args... args))
const Data & writeable() const
RegisterLBuf(const std::string &new_name)
void addRegisters(std::initializer_list< std::reference_wrapper< RegisterBase >> regs)
void read(void *buf, off_t offset, size_t bytes) override
void write(const void *buf) override
Register< uint64_t, ByteOrder::little > Register64LE
void update(const Data &new_data)
RegisterRaz(const std::string &new_name, size_t new_size)
std::function< void(This ®, const BackingType &value)> WriteFunc
RegisterBuf(const std::string &new_name, void *ptr, size_t bytes)
void write(const void *buf, off_t offset, size_t bytes) override
void serialize(std::ostream &os) const override
Register< Data, RegByteOrder > This
void read(void *buf) override
Register< uint32_t, ByteOrder::big > Register32BE
constexpr Data htoreg(Data data)
void write(const void *buf) override
void update(const Data &new_data, const Data &bitmask)
void addRegister(RegisterBase ®)
void serialize(std::ostream &os) const override
constexpr This & partialReader(const PartialReadFunc &new_reader)
static Data defaultPartialReader(This ®, int first, int last)
Register< uint16_t, ByteOrder::little > Register16LE
PartialReadFunc _partialReader
constexpr This & reader(Parent *parent, Data(Parent::*nr)(Args... args))
constexpr Data regtoh(Data data)
static void defaultWriter(This ®, const Data &value)
void fill(void *buf, off_t offset, size_t bytes) override
void read(void *buf) override
static constexpr Data readWithMask(const Data &value, const Data &bitmask)
virtual void write(Addr addr, const void *buf, Addr bytes)
Register< uint64_t > Register64
void fill(void *buf, off_t offset, size_t bytes) override
Register< uint8_t, ByteOrder::little > Register8LE
constexpr RegisterBank(const std::string &new_name, Addr new_base)
std::function< BackingType(This ®)> ReadFunc
static constexpr Data writeWithMask(const Data &old, const Data &value, const Data &bitmask)
virtual void read(Addr addr, void *buf, Addr bytes)
Register< uint32_t, ByteOrder::little > Register32LE
constexpr This & writeable(const Data &new_mask)
PartialWriteFunc _partialWriter
constexpr RegisterRoFill(const std::string &new_name, size_t new_size)
constexpr This & reader(const ReadFunc &new_reader)
Register< uint8_t, ByteOrder::big > Register8BE
Register< uint8_t > Register8
virtual void read(void *buf)=0
constexpr This & writer(Parent *parent, void(Parent::*nw)(Args... args))
Register< uint64_t, ByteOrder::big > Register64BE
std::array< uint8_t, BufBytes > buffer
Register< uint32_t > Register32
bool unserialize(const std::string &s) override
void serialize(std::ostream &os) const override
void serialize(std::ostream &os) const override
virtual void write(const void *buf)=0
virtual const std::string & name() const
constexpr T mbits(T val, unsigned first, unsigned last)
Mask off the given bits in place like bits() but without shifting.
std::function< void(This ®, const BackingType &value, int first, int last)> PartialWriteFunc
void write(const void *buf, off_t offset, size_t bytes) override
static void defaultPartialWriter(This ®, const Data &value, int first, int last)
virtual void fill(void *buf, off_t offset, size_t bytes)=0
Register< uint16_t, ByteOrder::big > Register16BE
void read(void *buf, off_t offset, size_t bytes) override
bool unserialize(const std::string &s) override
static Data defaultReader(This ®)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
virtual bool unserialize(const std::string &s)=0
static bool parse(const std::string &s, T &value)
std::function< BackingType(This ®, int first, int last)> PartialReadFunc
constexpr Register(const std::string &new_name, const Data &new_data)
constexpr This & partialWriter(const PartialWriteFunc &new_writer)
constexpr Register(const std::string &new_name, const Data &&new_data)
constexpr RegisterBase(const std::string &new_name, size_t new_size)
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Overload hash function for BasicBlockRange type.
constexpr This & partialReader(Parent *parent, Data(Parent::*nr)(Args... args))
const std::string & name() const
constexpr This & readonly()
RegisterRao(const std::string &new_name, size_t new_size)
Register< uint16_t > Register16
bool unserialize(const std::string &s) override
void write(const void *buf) override
virtual void serialize(std::ostream &os) const =0
static void show(std::ostream &os, const T &value)
constexpr Register(const std::string &new_name)
static void show(std::ostream &os, const T &value)
bool unserialize(const std::string &s) override
void read(void *buf) override
constexpr This & writer(const WriteFunc &new_writer)
void read(void *buf, off_t offset, size_t bytes) override
std::map< Addr, std::reference_wrapper< RegisterBase > > _offsetMap
#define panic(...)
This implements a cprintf based panic() function.
void write(const void *buf, off_t offset, size_t bytes) override
Generated on Tue Mar 23 2021 19:41:26 for gem5 by doxygen 1.8.17