gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
rtc_pl031.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2010-2012 ARM Limited
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/arm/rtc_pl031.hh"
39 
40 #include "base/intmath.hh"
41 #include "base/time.hh"
42 #include "base/trace.hh"
43 #include "debug/Checkpoint.hh"
44 #include "debug/Timer.hh"
45 #include "dev/arm/amba_device.hh"
46 #include "mem/packet.hh"
47 #include "mem/packet_access.hh"
48 
50  : AmbaIntDevice(p, 0x1000), lastWrittenTick(0), loadVal(0), matchVal(0),
51  rawInt(false), maskInt(false), pendingInt(false),
52  matchEvent([this]{ counterMatch(); }, name())
53 {
54  // Make a temporary copy so mkutctime can modify it.
55  struct tm local_time = p.time;
56  timeVal = mkutctime(&local_time);
57 }
58 
59 
60 Tick
62 {
63  assert(pkt->getAddr() >= pioAddr && pkt->getAddr() < pioAddr + pioSize);
64  assert(pkt->getSize() <= 4);
65  Addr daddr = pkt->getAddr() - pioAddr;
66  uint32_t data;
67 
68  DPRINTF(Timer, "Reading from RTC at offset: %#x\n", daddr);
69 
70  switch (daddr) {
71  case DataReg:
73  break;
74  case MatchReg:
75  data = matchVal;
76  break;
77  case LoadReg:
78  data = loadVal;
79  break;
80  case ControlReg:
81  data = 1; // Always enabled otherwise there is no point
82  break;
83  case IntMask:
84  data = maskInt;
85  break;
86  case RawISR:
87  data = rawInt;
88  break;
89  case MaskedISR:
90  data = pendingInt;
91  break;
92  default:
93  if (readId(pkt, ambaId, pioAddr)) {
94  // Hack for variable sized access
95  data = pkt->getUintX(ByteOrder::little);
96  break;
97  }
98  panic("Tried to read PL031 at offset %#x that doesn't exist\n", daddr);
99  break;
100  }
101 
102  pkt->setUintX(data, ByteOrder::little);
103  pkt->makeAtomicResponse();
104  return pioDelay;
105 }
106 
107 Tick
109 {
110  assert(pkt->getAddr() >= pioAddr && pkt->getAddr() < pioAddr + pioSize);
111  assert(pkt->getSize() <= 4);
112  Addr daddr = pkt->getAddr() - pioAddr;
113  DPRINTF(Timer, "Writing to RTC at offset: %#x\n", daddr);
114 
115  switch (daddr) {
116  case DataReg:
117  break;
118  case MatchReg:
119  matchVal = pkt->getLE<uint32_t>();
120  resyncMatch();
121  break;
122  case LoadReg:
124  timeVal = pkt->getLE<uint32_t>();
125  loadVal = timeVal;
126  resyncMatch();
127  break;
128  case ControlReg:
129  break; // Can't stop when started
130  case IntMask:
131  maskInt = pkt->getLE<uint32_t>();
132  break;
133  case IntClear:
134  if (pkt->getLE<uint32_t>()) {
135  rawInt = false;
136  pendingInt = false;
137  }
138  break;
139  default:
140  if (readId(pkt, ambaId, pioAddr))
141  break;
142  panic("Tried to read PL031 at offset %#x that doesn't exist\n", daddr);
143  break;
144  }
145 
146  pkt->makeAtomicResponse();
147  return pioDelay;
148 }
149 
150 void
152 {
153  DPRINTF(Timer, "Setting up new match event match=%d time=%d\n", matchVal,
154  timeVal);
155 
156  uint32_t seconds_until = matchVal - timeVal;
157  Tick ticks_until = SimClock::Int::s * seconds_until;
158 
159  if (matchEvent.scheduled()) {
160  DPRINTF(Timer, "-- Event was already schedule, de-scheduling\n");
162  }
163  schedule(matchEvent, curTick() + ticks_until);
164  DPRINTF(Timer, "-- Scheduling new event for: %d\n", curTick() + ticks_until);
165 }
166 
167 void
169 {
170  DPRINTF(Timer, "Counter reached zero\n");
171 
172  rawInt = true;
173  bool old_pending = pendingInt;
175  if (pendingInt && !old_pending) {
176  DPRINTF(Timer, "-- Causing interrupt\n");
177  interrupt->raise();
178  }
179 }
180 
181 void
183 {
184  DPRINTF(Checkpoint, "Serializing Arm PL031\n");
192 
193  bool is_in_event = matchEvent.scheduled();
194  SERIALIZE_SCALAR(is_in_event);
195 
196  Tick event_time;
197  if (is_in_event){
198  event_time = matchEvent.when();
199  SERIALIZE_SCALAR(event_time);
200  }
201 }
202 
203 void
205 {
206  DPRINTF(Checkpoint, "Unserializing Arm PL031\n");
207 
215 
216  bool is_in_event;
217  UNSERIALIZE_SCALAR(is_in_event);
218 
219  Tick event_time;
220  if (is_in_event){
221  UNSERIALIZE_SCALAR(event_time);
222  schedule(matchEvent, event_time);
223  }
224 }
PL031::RawISR
@ RawISR
Definition: rtc_pl031.hh:57
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
Packet::makeAtomicResponse
void makeAtomicResponse()
Definition: packet.hh:1017
PL031::read
Tick read(PacketPtr pkt) override
Handle a read to the device.
Definition: rtc_pl031.cc:61
BasicPioDevice::pioAddr
Addr pioAddr
Address that the device listens to.
Definition: io_device.hh:148
PL031::MatchReg
@ MatchReg
Definition: rtc_pl031.hh:53
data
const char data[]
Definition: circlebuf.test.cc:47
PL031::write
Tick write(PacketPtr pkt) override
Handle writes to the device.
Definition: rtc_pl031.cc:108
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
Packet::getAddr
Addr getAddr() const
Definition: packet.hh:755
PL031::DataReg
@ DataReg
Definition: rtc_pl031.hh:52
AmbaPioDevice::ambaId
uint64_t ambaId
Definition: amba_device.hh:79
amba_device.hh
mkutctime
time_t mkutctime(struct tm *time)
Definition: time.cc:151
time.hh
PL031::matchVal
uint32_t matchVal
Definition: rtc_pl031.hh:75
ArmInterruptPin::raise
virtual void raise()=0
Signal an interrupt.
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
EventManager::deschedule
void deschedule(Event &event)
Definition: eventq.hh:1025
Packet::getSize
unsigned getSize() const
Definition: packet.hh:765
Event::when
Tick when() const
Get the time that the event is scheduled.
Definition: eventq.hh:505
PL031::matchEvent
EventFunctionWrapper matchEvent
Definition: rtc_pl031.hh:92
packet.hh
PL031::IntMask
@ IntMask
Definition: rtc_pl031.hh:56
cp
Definition: cprintf.cc:37
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
PL031::resyncMatch
void resyncMatch()
Called to update the matchEvent when the load Value or match value are written.
Definition: rtc_pl031.cc:151
AmbaIntDevice::interrupt
ArmInterruptPin *const interrupt
Definition: amba_device.hh:89
Packet::getUintX
uint64_t getUintX(ByteOrder endian) const
Get the data in the packet byte swapped from the specified endianness and zero-extended to 64 bits.
Definition: packet.cc:350
PL031::maskInt
bool maskInt
If the timer interrupt mask that is anded with the raw interrupt to generate a pending interrupt.
Definition: rtc_pl031.hh:84
PL031::IntClear
@ IntClear
Definition: rtc_pl031.hh:59
PL031::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: rtc_pl031.cc:204
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
PL031::rawInt
bool rawInt
If timer has caused an interrupt.
Definition: rtc_pl031.hh:79
PL031::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: rtc_pl031.cc:182
SimClock::Int::s
Tick s
second
Definition: core.cc:59
rtc_pl031.hh
Packet::setUintX
void setUintX(uint64_t w, ByteOrder endian)
Set the value in the word w after truncating it to the length of the packet and then byteswapping it ...
Definition: packet.cc:367
AmbaDevice::readId
bool readId(PacketPtr pkt, uint64_t amba_id, Addr pio_addr)
Definition: amba_device.cc:72
BasicPioDevice::pioSize
Addr pioSize
Size that the device's address range.
Definition: io_device.hh:151
AmbaIntDevice
Definition: amba_device.hh:86
PL031::ControlReg
@ ControlReg
Definition: rtc_pl031.hh:55
PL031::loadVal
uint32_t loadVal
Definition: rtc_pl031.hh:70
PL031::LoadReg
@ LoadReg
Definition: rtc_pl031.hh:54
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
packet_access.hh
Packet::getLE
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
Definition: packet_access.hh:75
AmbaPioDevice::Params
AmbaPioDeviceParams Params
Definition: amba_device.hh:82
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
PL031::timeVal
uint32_t timeVal
Definition: rtc_pl031.hh:64
PL031::counterMatch
void counterMatch()
Called when the counter reaches matches.
Definition: rtc_pl031.cc:168
BasicPioDevice::pioDelay
Tick pioDelay
Delay that the device experinces on an access.
Definition: io_device.hh:154
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
trace.hh
PL031::lastWrittenTick
Tick lastWrittenTick
Definition: rtc_pl031.hh:67
PL031::MaskedISR
@ MaskedISR
Definition: rtc_pl031.hh:58
PL031::PL031
PL031(const Params &p)
The constructor for RealView just registers itself with the MMU.
Definition: rtc_pl031.cc:49
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
intmath.hh
CheckpointIn
Definition: serialize.hh:68
PL031::pendingInt
bool pendingInt
If an interrupt is currently pending.
Definition: rtc_pl031.hh:88
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171

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