gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
clint.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2021 Huawei International
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #include "dev/riscv/clint.hh"
39 
40 #include "debug/Clint.hh"
41 #include "mem/packet.hh"
42 #include "mem/packet_access.hh"
43 #include "params/Clint.hh"
44 #include "sim/system.hh"
45 
46 using namespace RiscvISA;
47 
48 Clint::Clint(const Params &params) :
49  BasicPioDevice(params, params.pio_size),
50  system(params.system),
51  intrctrl(params.intrctrl),
52  signal(params.name + ".signal", 0, this),
53  registers(params.name + ".registers", params.pio_addr, this)
54 {
55 }
56 
57 void
59 {
60  // Increment mtime
61  uint64_t& mtime = registers.mtime.get();
62  mtime++;
63 
64  for (int context_id = 0; context_id < nThread; context_id++) {
65 
66  // Update misc reg file
67  ISA* isa = dynamic_cast<ISA*>(
68  system->threads[context_id]->getIsaPtr());
69  isa->setMiscRegNoEffect(MISCREG_TIME, mtime);
70 
71  // Post timer interrupt
72  uint64_t mtimecmp = registers.mtimecmp[context_id].get();
73  if (mtime >= mtimecmp) {
74  if (mtime == mtimecmp) {
75  DPRINTF(Clint,
76  "MTIP posted - thread: %d, mtime: %d, mtimecmp: %d\n",
77  context_id, mtime, mtimecmp);
78  }
80  } else {
82  }
83  }
84 }
85 
86 void
88 {
89  using namespace std::placeholders;
90 
91  // Calculate reserved space size
92  const size_t reserved0_size = mtimecmpStart - clint->nThread * 4;
93  reserved.emplace_back("reserved0", reserved0_size);
94  const size_t reserved1_size = mtimeStart
95  - mtimecmpStart - clint->nThread * 8;
96  reserved.emplace_back("reserved1", reserved1_size);
97 
98  // Sanity check
99  assert((int) clint->pioSize <= maxBankSize);
100 
101  // Initialize registers
102  for (int i = 0; i < clint->nThread; i++) {
103  msip.emplace_back(std::string("msip") + std::to_string(i), 0);
104  mtimecmp.emplace_back(std::string("mtimecmp") + std::to_string(i), 0);
105  }
106 
107  // Add registers to bank
108  for (int i = 0; i < clint->nThread; i++) {
109  auto read_cb = std::bind(&Clint::readMSIP, clint, _1, i);
110  msip[i].reader(read_cb);
111  auto write_cb = std::bind(&Clint::writeMSIP, clint, _1, _2, i);
112  msip[i].writer(write_cb);
113  addRegister(msip[i]);
114  }
115  addRegister(reserved[0]);
116  for (int i = 0; i < clint->nThread; i++) {
118  }
119  addRegister(reserved[1]);
120  mtime.readonly();
122 }
123 
124 uint32_t
125 Clint::readMSIP(Register32& reg, const int thread_id)
126 {
127  // To avoid discrepancies if mip is externally set using remote_gdb etc.
128  auto tc = system->threads[thread_id];
129  RegVal mip = tc->readMiscReg(MISCREG_IP);
130  uint32_t msip = bits<uint32_t>(mip, ExceptionCode::INT_SOFTWARE_MACHINE);
131  reg.update(msip);
132  return reg.get();
133 };
134 
135 void
136 Clint::writeMSIP(Register32& reg, const uint32_t& data, const int thread_id)
137 {
138  reg.update(data);
139  assert(data <= 1);
140  if (data > 0) {
141  DPRINTF(Clint,
142  "MSIP posted - thread: %d\n", thread_id);
143  intrctrl->post(thread_id,
145  } else {
146  DPRINTF(Clint,
147  "MSIP cleared - thread: %d\n", thread_id);
148  intrctrl->clear(thread_id,
150  }
151 };
152 
153 Tick
155 {
156  // Check for atomic operation
157  bool is_atomic = pkt->isAtomicOp() && pkt->cmd == MemCmd::SwapReq;
158  DPRINTF(Clint,
159  "Read request - addr: %#x, size: %#x, atomic:%d\n",
160  pkt->getAddr(), pkt->getSize(), is_atomic);
161 
162  // Perform register read
163  registers.read(pkt->getAddr(), pkt->getPtr<void>(), pkt->getSize());
164 
165  if (is_atomic) {
166  // Perform atomic operation
167  (*(pkt->getAtomicOp()))(pkt->getPtr<uint8_t>());
168  return write(pkt);
169  } else {
170  pkt->makeResponse();
171  return pioDelay;
172  }
173 }
174 
175 Tick
177 {
178  DPRINTF(Clint,
179  "Write request - addr: %#x, size: %#x\n",
180  pkt->getAddr(), pkt->getSize());
181 
182  // Perform register write
183  registers.write(pkt->getAddr(), pkt->getPtr<void>(), pkt->getSize());
184 
185  pkt->makeResponse();
186  return pioDelay;
187 }
188 
189 void
191 {
192  nThread = system->threads.size();
193  registers.init();
195 }
196 
197 Port &
198 Clint::getPort(const std::string &if_name, PortID idx)
199 {
200  if (if_name == "int_pin")
201  return signal;
202  else
203  return BasicPioDevice::getPort(if_name, idx);
204 }
205 
206 void
208 {
209  for (auto const &reg: registers.msip) {
210  paramOut(cp, reg.name(), reg);
211  }
212  for (auto const &reg: registers.mtimecmp) {
213  paramOut(cp, reg.name(), reg);
214  }
215  paramOut(cp, "mtime", registers.mtime);
216 }
217 
218 void
220 {
221  for (auto &reg: registers.msip) {
222  paramIn(cp, reg.name(), reg);
223  }
224  for (auto &reg: registers.mtimecmp) {
225  paramIn(cp, reg.name(), reg);
226  }
227  paramIn(cp, "mtime", registers.mtime);
228 }
Clint::signal
IntSinkPin< Clint > signal
Definition: clint.hh:76
IntrControl::clear
void clear(int cpu_id, int int_num, int index)
Definition: intr_control.cc:53
system.hh
data
const char data[]
Definition: circlebuf.test.cc:47
IntrControl::post
void post(int cpu_id, int int_num, int index)
Definition: intr_control.cc:45
Clint::ClintRegisters::msip
std::vector< Register32 > msip
Definition: clint.hh:110
Packet::getAddr
Addr getAddr() const
Definition: packet.hh:755
PioDevice::init
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
Definition: io_device.cc:56
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
Clint::ClintRegisters::clint
Clint * clint
Definition: clint.hh:119
Clint::registers
Clint::ClintRegisters registers
Clint::write
Tick write(PacketPtr pkt) override
Pure virtual function that the device must implement.
Definition: clint.cc:176
Clint::ClintRegisters::mtimecmpStart
const Addr mtimecmpStart
Definition: clint.hh:106
RegisterBank< ByteOrder::little >::addRegister
void addRegister(RegisterBase &reg)
Definition: reg_bank.hh:798
sc_dt::to_string
const std::string to_string(sc_enc enc)
Definition: sc_fxdefs.cc:91
Clint::ClintRegisters::mtimecmp
std::vector< Register64 > mtimecmp
Definition: clint.hh:111
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
Clint::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: clint.cc:198
PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:243
Packet::isAtomicOp
bool isAtomicOp() const
Definition: packet.hh:794
RiscvISA::INT_TIMER_MACHINE
@ INT_TIMER_MACHINE
Definition: faults.hh:85
Packet::getSize
unsigned getSize() const
Definition: packet.hh:765
RegisterBank< ByteOrder::little >::write
virtual void write(Addr addr, const void *buf, Addr bytes)
Definition: reg_bank.hh:862
X86ISA::reg
Bitfield< 5, 3 > reg
Definition: types.hh:88
Clint::ClintRegisters::mtime
Register64 mtime
Definition: clint.hh:112
clint.hh
Clint::nThread
int nThread
Definition: clint.hh:75
Clint::ClintRegisters::init
void init()
Definition: clint.cc:87
RiscvISA
Definition: fs_workload.cc:37
Clint::raiseInterruptPin
void raiseInterruptPin(int id)
Timer tick callback.
Definition: clint.cc:58
packet.hh
RegisterBank::read
virtual void read(Addr addr, void *buf, Addr bytes)
Definition: reg_bank.hh:805
X86ISA::system
Bitfield< 15 > system
Definition: misc.hh:997
Clint::Register32
ClintRegisters::Register32 Register32
Definition: clint.hh:125
cp
Definition: cprintf.cc:37
MemCmd::SwapReq
@ SwapReq
Definition: packet.hh:112
Clint::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: clint.cc:219
RiscvISA::MISCREG_IP
@ MISCREG_IP
Definition: registers.hh:161
Clint::ClintRegisters::mtimeStart
const Addr mtimeStart
Definition: clint.hh:107
Clint::Clint
Clint(const Params &params)
Definition: clint.cc:48
RiscvISA::INT_SOFTWARE_MACHINE
@ INT_SOFTWARE_MACHINE
Definition: faults.hh:82
PioDevice::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: io_device.cc:64
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
Port
Ports are used to interface objects to each other.
Definition: port.hh:56
Packet::getAtomicOp
AtomicOpFunctor * getAtomicOp() const
Accessor function to atomic op.
Definition: packet.hh:793
System::Threads::size
int size() const
Definition: system.hh:204
BasicPioDevice::pioSize
Addr pioSize
Size that the device's address range.
Definition: io_device.hh:151
Clint::intrctrl
IntrControl * intrctrl
Definition: clint.hh:74
Clint
NOTE: This implementation of CLINT is based on the SiFive U54MC datasheet: https://sifive....
Definition: clint.hh:69
RiscvISA::ISA::setMiscRegNoEffect
void setMiscRegNoEffect(int misc_reg, RegVal val)
Definition: isa.cc:315
RiscvISA::MISCREG_TIME
@ MISCREG_TIME
Definition: registers.hh:164
Clint::readMSIP
uint32_t readMSIP(Register32 &reg, const int thread_id)
Definition: clint.cc:125
Packet::makeResponse
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
Definition: packet.hh:1005
name
const std::string & name()
Definition: trace.cc:48
paramOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
Definition: types.cc:37
packet_access.hh
Packet::cmd
MemCmd cmd
The command field of the packet.
Definition: packet.hh:336
System::threads
Threads threads
Definition: system.hh:304
Clint::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: clint.cc:207
RiscvISA::ISA
Definition: isa.hh:71
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
BasicPioDevice
Definition: io_device.hh:144
BasicPioDevice::pioDelay
Tick pioDelay
Delay that the device experinces on an access.
Definition: io_device.hh:154
Packet::getPtr
T * getPtr()
get a pointer to the data ptr.
Definition: packet.hh:1158
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
paramIn
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
Definition: types.cc:69
CheckpointIn
Definition: serialize.hh:68
Clint::ClintRegisters::reserved
std::vector< RegisterRaz > reserved
Definition: clint.hh:113
Clint::init
void init() override
SimObject functions.
Definition: clint.cc:190
PioDevice::Params
PioDeviceParams Params
Definition: io_device.hh:131
Clint::ClintRegisters::maxBankSize
const Addr maxBankSize
Definition: clint.hh:108
RegVal
uint64_t RegVal
Definition: types.hh:174
Clint::read
Tick read(PacketPtr pkt) override
PioDevice interface functions.
Definition: clint.cc:154
Clint::system
System * system
Definition: clint.hh:73
Clint::writeMSIP
void writeMSIP(Register32 &reg, const uint32_t &data, const int thread_id)
Definition: clint.cc:136

Generated on Tue Jun 22 2021 15:28:28 for gem5 by doxygen 1.8.17