gem5  v21.0.0.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  system->threads[context_id]->setMiscRegNoEffect(MISCREG_TIME, mtime);
68 
69  // Post timer interrupt
70  uint64_t mtimecmp = registers.mtimecmp[context_id].get();
71  if (mtime >= mtimecmp) {
72  if (mtime == mtimecmp) {
73  DPRINTF(Clint,
74  "MTIP posted - thread: %d, mtime: %d, mtimecmp: %d\n",
75  context_id, mtime, mtimecmp);
76  }
78  } else {
80  }
81  }
82 }
83 
84 void
86 {
87  using namespace std::placeholders;
88 
89  // Calculate reserved space size
90  const size_t reserved0_size = mtimecmpStart - clint->nThread * 4;
91  reserved.emplace_back("reserved0", reserved0_size);
92  const size_t reserved1_size = mtimeStart
93  - mtimecmpStart - clint->nThread * 8;
94  reserved.emplace_back("reserved1", reserved1_size);
95 
96  // Sanity check
97  assert((int) clint->pioSize <= maxBankSize);
98 
99  // Initialize registers
100  for (int i = 0; i < clint->nThread; i++) {
101  msip.emplace_back(std::string("msip") + std::to_string(i), 0);
102  mtimecmp.emplace_back(std::string("mtimecmp") + std::to_string(i), 0);
103  }
104 
105  // Add registers to bank
106  for (int i = 0; i < clint->nThread; i++) {
107  auto read_cb = std::bind(&Clint::readMSIP, clint, _1, i);
108  msip[i].reader(read_cb);
109  auto write_cb = std::bind(&Clint::writeMSIP, clint, _1, _2, i);
110  msip[i].writer(write_cb);
111  addRegister(msip[i]);
112  }
113  addRegister(reserved[0]);
114  for (int i = 0; i < clint->nThread; i++) {
116  }
117  addRegister(reserved[1]);
118  mtime.readonly();
120 }
121 
122 uint32_t
123 Clint::readMSIP(Register32& reg, const int thread_id)
124 {
125  // To avoid discrepancies if mip is externally set using remote_gdb etc.
126  auto tc = system->threads[thread_id];
127  RegVal mip = tc->readMiscReg(MISCREG_IP);
128  uint32_t msip = bits<uint32_t>(mip, ExceptionCode::INT_SOFTWARE_MACHINE);
129  reg.update(msip);
130  return reg.get();
131 };
132 
133 void
134 Clint::writeMSIP(Register32& reg, const uint32_t& data, const int thread_id)
135 {
136  reg.update(data);
137  assert(data <= 1);
138  if (data > 0) {
139  DPRINTF(Clint,
140  "MSIP posted - thread: %d\n", thread_id);
141  intrctrl->post(thread_id,
143  } else {
144  DPRINTF(Clint,
145  "MSIP cleared - thread: %d\n", thread_id);
146  intrctrl->clear(thread_id,
148  }
149 };
150 
151 Tick
153 {
154  // Check for atomic operation
155  bool is_atomic = pkt->isAtomicOp() && pkt->cmd == MemCmd::SwapReq;
156  DPRINTF(Clint,
157  "Read request - addr: %#x, size: %#x, atomic:%d\n",
158  pkt->getAddr(), pkt->getSize(), is_atomic);
159 
160  // Perform register read
161  registers.read(pkt->getAddr(), pkt->getPtr<void>(), pkt->getSize());
162 
163  if (is_atomic) {
164  // Perform atomic operation
165  (*(pkt->getAtomicOp()))(pkt->getPtr<uint8_t>());
166  return write(pkt);
167  } else {
168  pkt->makeResponse();
169  return pioDelay;
170  }
171 }
172 
173 Tick
175 {
176  DPRINTF(Clint,
177  "Write request - addr: %#x, size: %#x\n",
178  pkt->getAddr(), pkt->getSize());
179 
180  // Perform register write
181  registers.write(pkt->getAddr(), pkt->getPtr<void>(), pkt->getSize());
182 
183  pkt->makeResponse();
184  return pioDelay;
185 }
186 
187 void
189 {
190  nThread = system->threads.size();
191  registers.init();
193 }
194 
195 Port &
196 Clint::getPort(const std::string &if_name, PortID idx)
197 {
198  if (if_name == "int_pin")
199  return signal;
200  else
201  return BasicPioDevice::getPort(if_name, idx);
202 }
203 
204 void
206 {
207  for (auto const &reg: registers.msip) {
208  paramOut(cp, reg.name(), reg);
209  }
210  for (auto const &reg: registers.mtimecmp) {
211  paramOut(cp, reg.name(), reg);
212  }
213  paramOut(cp, "mtime", registers.mtime);
214 }
215 
216 void
218 {
219  for (auto &reg: registers.msip) {
220  paramIn(cp, reg.name(), reg);
221  }
222  for (auto &reg: registers.mtimecmp) {
223  paramIn(cp, reg.name(), reg);
224  }
225  paramIn(cp, "mtime", registers.mtime);
226 }
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:174
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:196
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:85
RiscvISA
Definition: fs_workload.cc:36
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:217
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::MISCREG_TIME
@ MISCREG_TIME
Definition: registers.hh:164
Clint::readMSIP
uint32_t readMSIP(Register32 &reg, const int thread_id)
Definition: clint.cc:123
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:205
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:188
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:152
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:134

Generated on Tue Mar 23 2021 19:41:26 for gem5 by doxygen 1.8.17