gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
mc146818.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2004-2005 The Regents of The University of Michigan
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #include "dev/mc146818.hh"
30 
31 #include <sys/time.h>
32 
33 #include <ctime>
34 #include <string>
35 
36 #include "base/bitfield.hh"
37 #include "base/time.hh"
38 #include "base/trace.hh"
39 #include "debug/MC146818.hh"
40 #include "dev/rtcreg.h"
41 
42 static uint8_t
43 bcdize(uint8_t val)
44 {
45  uint8_t result;
46  result = val % 10;
47  result += (val / 10) << 4;
48  return result;
49 }
50 
51 static uint8_t
52 unbcdize(uint8_t val)
53 {
54  uint8_t result;
55  result = val & 0xf;
56  result += (val >> 4) * 10;
57  return result;
58 }
59 
60 void
61 MC146818::setTime(const struct tm time)
62 {
63  curTime = time;
64  year = time.tm_year;
65  // Unix is 0-11 for month, data seet says start at 1
66  mon = time.tm_mon + 1;
67  mday = time.tm_mday;
68  hour = time.tm_hour;
69  min = time.tm_min;
70  sec = time.tm_sec;
71 
72  // Datasheet says 1 is sunday
73  wday = time.tm_wday + 1;
74 
75  if (!stat_regB.dm) {
76  // The datasheet says that the year field can be either BCD or
77  // years since 1900. Linux seems to be happy with years since
78  // 1900.
79  year = bcdize(year % 100);
80  mon = bcdize(mon);
81  mday = bcdize(mday);
82  hour = bcdize(hour);
83  min = bcdize(min);
84  sec = bcdize(sec);
85  }
86 }
87 
88 MC146818::MC146818(EventManager *em, const std::string &n,
89  const struct tm time, bool bcd, Tick frequency)
90  : EventManager(em), _name(n), event(this, frequency), tickEvent(this)
91 {
92  memset(clock_data, 0, sizeof(clock_data));
93 
94  stat_regA = 0;
95  stat_regA.dv = RTCA_DV_32768HZ;
96  stat_regA.rs = RTCA_RS_1024HZ;
97 
98  stat_regB = 0;
99  stat_regB.pie = 1;
100  stat_regB.format24h = 1;
101  stat_regB.dm = bcd ? 0 : 1;
102 
103  setTime(time);
104  DPRINTFN("Real-time clock set to %s", asctime(&time));
105 }
106 
108 {
110  deschedule(event);
111 }
112 
113 bool
114 MC146818::rega_dv_disabled(const RtcRegA &reg)
115 {
116  return reg.dv == RTCA_DV_DISABLED0 ||
117  reg.dv == RTCA_DV_DISABLED1;
118 }
119 
120 void
122 {
123  assert(!event.scheduled());
124  assert(!tickEvent.scheduled());
125 
126  if (stat_regB.pie)
128  if (!rega_dv_disabled(stat_regA))
130 }
131 
132 void
133 MC146818::writeData(const uint8_t addr, const uint8_t data)
134 {
135  bool panic_unsupported(false);
136 
137  if (addr < RTC_STAT_REGA) {
138  clock_data[addr] = data;
139  curTime.tm_sec = unbcdize(sec);
140  curTime.tm_min = unbcdize(min);
141  curTime.tm_hour = unbcdize(hour);
142  curTime.tm_mday = unbcdize(mday);
143  curTime.tm_mon = unbcdize(mon) - 1;
144  curTime.tm_year = ((unbcdize(year) + 50) % 100) + 1950;
145  curTime.tm_wday = unbcdize(wday) - 1;
146  } else {
147  switch (addr) {
148  case RTC_STAT_REGA: {
149  RtcRegA old_rega(stat_regA);
150  stat_regA = data;
151  // The "update in progress" bit is read only.
152  stat_regA.uip = old_rega;
153 
154  if (!rega_dv_disabled(stat_regA) &&
155  stat_regA.dv != RTCA_DV_32768HZ) {
156  inform("RTC: Unimplemented divider configuration: %i\n",
157  stat_regA.dv);
158  panic_unsupported = true;
159  }
160 
161  if (stat_regA.rs != RTCA_RS_1024HZ) {
162  inform("RTC: Unimplemented interrupt rate: %i\n",
163  stat_regA.rs);
164  panic_unsupported = true;
165  }
166 
167  if (rega_dv_disabled(stat_regA)) {
168  // The divider is disabled, make sure that we don't
169  // schedule any ticks.
170  if (tickEvent.scheduled())
172  } else if (rega_dv_disabled(old_rega)) {
173  // According to the specification, the next tick
174  // happens after 0.5s when the divider chain goes
175  // from reset to active. So, we simply schedule the
176  // tick after 0.5s.
177  assert(!tickEvent.scheduled());
179  }
180  } break;
181  case RTC_STAT_REGB:
182  stat_regB = data;
183  if (stat_regB.aie || stat_regB.uie) {
184  inform("RTC: Unimplemented interrupt configuration: %s %s\n",
185  stat_regB.aie ? "alarm" : "",
186  stat_regB.uie ? "update" : "");
187  panic_unsupported = true;
188  }
189 
190  if (stat_regB.dm) {
191  inform("RTC: The binary interface is not fully implemented.\n");
192  panic_unsupported = true;
193  }
194 
195  if (!stat_regB.format24h) {
196  inform("RTC: The 12h time format not supported.\n");
197  panic_unsupported = true;
198  }
199 
200  if (stat_regB.dse) {
201  inform("RTC: Automatic daylight saving time not supported.\n");
202  panic_unsupported = true;
203  }
204 
205  if (stat_regB.pie) {
206  if (!event.scheduled())
207  event.scheduleIntr();
208  } else {
209  if (event.scheduled())
210  deschedule(event);
211  }
212  break;
213  case RTC_STAT_REGC:
214  case RTC_STAT_REGD:
215  panic("RTC status registers C and D are not implemented.\n");
216  break;
217  }
218  }
219 
220  if (panic_unsupported)
221  panic("Unimplemented RTC configuration!\n");
222 
223 }
224 
225 uint8_t
227 {
228  if (addr < RTC_STAT_REGA)
229  return clock_data[addr];
230  else {
231  switch (addr) {
232  case RTC_STAT_REGA:
233  // toggle UIP bit for linux
234  stat_regA.uip = !stat_regA.uip;
235  return stat_regA;
236  break;
237  case RTC_STAT_REGB:
238  return stat_regB;
239  break;
240  case RTC_STAT_REGC:
241  case RTC_STAT_REGD:
242  return 0x00;
243  break;
244  default:
245  panic("Shouldn't be here");
246  }
247  }
248 }
249 
250 void
252 {
253  assert(!rega_dv_disabled(stat_regA));
254 
255  if (stat_regB.set)
256  return;
257  time_t calTime = mkutctime(&curTime);
258  calTime++;
259  setTime(*gmtime(&calTime));
260 }
261 
262 void
263 MC146818::serialize(const std::string &base, CheckpointOut &cp) const
264 {
265  uint8_t regA_serial(stat_regA);
266  uint8_t regB_serial(stat_regB);
267 
268  arrayParamOut(cp, base + ".clock_data", clock_data, sizeof(clock_data));
269  paramOut(cp, base + ".stat_regA", (uint8_t)regA_serial);
270  paramOut(cp, base + ".stat_regB", (uint8_t)regB_serial);
271 
272  //
273  // save the timer tick and rtc clock tick values to correctly reschedule
274  // them during unserialize
275  //
276  Tick rtcTimerInterruptTickOffset = event.when() - curTick();
277  SERIALIZE_SCALAR(rtcTimerInterruptTickOffset);
278  Tick rtcClockTickOffset = tickEvent.when() - curTick();
279  SERIALIZE_SCALAR(rtcClockTickOffset);
280 }
281 
282 void
284 {
285  uint8_t tmp8;
286 
287  arrayParamIn(cp, base + ".clock_data", clock_data,
288  sizeof(clock_data));
289 
290  paramIn(cp, base + ".stat_regA", tmp8);
291  stat_regA = tmp8;
292  paramIn(cp, base + ".stat_regB", tmp8);
293  stat_regB = tmp8;
294 
295  //
296  // properly schedule the timer and rtc clock events
297  //
298  Tick rtcTimerInterruptTickOffset;
299  UNSERIALIZE_SCALAR(rtcTimerInterruptTickOffset);
300  event.offset = rtcTimerInterruptTickOffset;
301  Tick rtcClockTickOffset;
302  UNSERIALIZE_SCALAR(rtcClockTickOffset);
303  tickEvent.offset = rtcClockTickOffset;
304 }
305 
307  : parent(_parent), interval(i), offset(i)
308 {
309  DPRINTF(MC146818, "RTC Event Initilizing\n");
310 }
311 
312 void
314 {
315  parent->schedule(this, curTick() + interval);
316 }
317 
318 void
320 {
321  DPRINTF(MC146818, "RTC Timer Interrupt\n");
322  parent->schedule(this, curTick() + interval);
323  parent->handleEvent();
324 }
325 
326 const char *
328 {
329  return "RTC interrupt";
330 }
331 
332 void
334 {
335  DPRINTF(MC146818, "RTC clock tick\n");
336  parent->schedule(this, curTick() + SimClock::Int::s);
337  parent->tickClock();
338 }
339 
340 const char *
342 {
343  return "RTC clock tick";
344 }
unbcdize
static uint8_t unbcdize(uint8_t val)
Definition: mc146818.cc:52
RTC_STAT_REGA
static const int RTC_STAT_REGA
Definition: rtcreg.h:40
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
MC146818::year
uint8_t year
Definition: mc146818.hh:107
MC146818::unserialize
void unserialize(const std::string &base, CheckpointIn &cp)
Reconstruct the state of this object from a checkpoint.
Definition: mc146818.cc:283
RTC_STAT_REGD
static const int RTC_STAT_REGD
Definition: rtcreg.h:54
data
const char data[]
Definition: circlebuf.test.cc:47
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
MC146818::RTCEvent::scheduleIntr
void scheduleIntr()
Schedule the RTC periodic interrupt.
Definition: mc146818.cc:313
mkutctime
time_t mkutctime(struct tm *time)
Definition: time.cc:151
time.hh
MC146818::RTCTickEvent::description
const char * description() const
Event description.
Definition: mc146818.cc:341
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
X86ISA::base
Bitfield< 51, 12 > base
Definition: pagetable.hh:138
EventManager::deschedule
void deschedule(Event &event)
Definition: eventq.hh:1025
MC146818::stat_regB
RtcRegB stat_regB
RTC status register B.
Definition: mc146818.hh:148
MC146818::tickEvent
RTCTickEvent tickEvent
RTC tick event.
Definition: mc146818.hh:91
MC146818::hour
uint8_t hour
Definition: mc146818.hh:102
RTC_STAT_REGB
static const int RTC_STAT_REGB
Definition: rtcreg.h:51
Event::when
Tick when() const
Get the time that the event is scheduled.
Definition: eventq.hh:505
X86ISA::reg
Bitfield< 5, 3 > reg
Definition: types.hh:88
MC146818::wday
uint8_t wday
Definition: mc146818.hh:104
RTCA_DV_DISABLED1
static const int RTCA_DV_DISABLED1
Definition: rtcreg.h:46
MC146818::serialize
void serialize(const std::string &base, CheckpointOut &cp) const
Serialize this object to the given output stream.
Definition: mc146818.cc:263
ArmISA::n
Bitfield< 31 > n
Definition: miscregs_types.hh:450
MC146818::mon
uint8_t mon
Definition: mc146818.hh:106
MC146818::mday
uint8_t mday
Definition: mc146818.hh:105
MC146818::tickClock
void tickClock()
Definition: mc146818.cc:251
cp
Definition: cprintf.cc:37
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
arrayParamOut
decltype(std::begin(std::declval< const T & >()), std::end(std::declval< const T & >()), void()) arrayParamOut(CheckpointOut &os, const std::string &name, const T &param)
Definition: serialize.hh:415
bitfield.hh
MC146818::readData
uint8_t readData(const uint8_t addr)
RTC read data.
Definition: mc146818.cc:226
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
MipsISA::event
Bitfield< 10, 5 > event
Definition: pra_constants.hh:297
RTCA_DV_DISABLED0
static const int RTCA_DV_DISABLED0
Definition: rtcreg.h:45
SimClock::Int::s
Tick s
second
Definition: core.cc:59
X86ISA::em
Bitfield< 2 > em
Definition: misc.hh:602
MC146818::RTCEvent::RTCEvent
RTCEvent(MC146818 *_parent, Tick i)
Definition: mc146818.cc:306
MC146818::RTCEvent::process
virtual void process()
Event process to occur at interrupt.
Definition: mc146818.cc:319
MC146818
Real-Time Clock (MC146818)
Definition: mc146818.hh:38
MC146818::RTCEvent::description
virtual const char * description() const
Event description.
Definition: mc146818.cc:327
X86ISA::val
Bitfield< 63 > val
Definition: misc.hh:769
MC146818::RTCTickEvent::process
void process()
Event process to occur at interrupt.
Definition: mc146818.cc:333
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
paramOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
Definition: types.cc:37
bcdize
static uint8_t bcdize(uint8_t val)
Definition: mc146818.cc:43
MC146818::curTime
struct tm curTime
Definition: mc146818.hh:111
RTCA_DV_32768HZ
static const int RTCA_DV_32768HZ
Definition: rtcreg.h:44
MC146818::RTCTickEvent::offset
Tick offset
Definition: mc146818.hh:70
MC146818::RTCEvent::offset
Tick offset
Definition: mc146818.hh:52
X86ISA::addr
Bitfield< 3 > addr
Definition: types.hh:80
RTCA_RS_1024HZ
static const int RTCA_RS_1024HZ
Definition: rtcreg.h:49
inform
#define inform(...)
Definition: logging.hh:240
MC146818::writeData
void writeData(const uint8_t addr, const uint8_t data)
RTC write data.
Definition: mc146818.cc:133
RTC_STAT_REGC
static const int RTC_STAT_REGC
Definition: rtcreg.h:53
MC146818::MC146818
MC146818(EventManager *em, const std::string &name, const struct tm time, bool bcd, Tick frequency)
Definition: mc146818.cc:88
MC146818::~MC146818
virtual ~MC146818()
Definition: mc146818.cc:107
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
arrayParamIn
void arrayParamIn(CheckpointIn &cp, const std::string &name, CircleBuf< T > &param)
Definition: circlebuf.hh:254
MC146818::clock_data
uint8_t clock_data[10]
Definition: mc146818.hh:95
rtcreg.h
EventManager
Definition: eventq.hh:984
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
MC146818::min
uint8_t min
Definition: mc146818.hh:100
trace.hh
DPRINTFN
#define DPRINTFN(...)
Definition: trace.hh:241
MC146818::setTime
void setTime(const struct tm time)
Definition: mc146818.cc:61
paramIn
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
Definition: types.cc:69
CheckpointIn
Definition: serialize.hh:68
MC146818::event
RTCEvent event
RTC periodic interrupt event.
Definition: mc146818.hh:88
MC146818::sec
uint8_t sec
Definition: mc146818.hh:98
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
mc146818.hh
ArmISA::offset
Bitfield< 23, 0 > offset
Definition: types.hh:153
MC146818::startup
virtual void startup()
Start ticking.
Definition: mc146818.cc:121

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