gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
uart8250.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 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 
33 #include "dev/serial/uart8250.hh"
34 
35 #include <string>
36 #include <vector>
37 
38 #include "base/inifile.hh"
39 #include "base/trace.hh"
40 #include "debug/Uart.hh"
41 #include "dev/platform.hh"
42 #include "mem/packet.hh"
43 #include "mem/packet_access.hh"
44 #include "sim/serialize.hh"
45 
46 void
48 {
49  if (intrBit & registers.ier.get()) {
50  DPRINTF(Uart, "UART InterEvent, interrupting\n");
52  status |= intrBit;
53  lastTxInt = curTick();
54  } else {
55  DPRINTF(Uart, "UART InterEvent, not interrupting\n");
56  }
57 
58 }
59 
60 /* The linux serial driver (8250.c about line 1182) loops reading from
61  * the device until the device reports it has no more data to
62  * read. After a maximum of 255 iterations the code prints "serial8250
63  * too much work for irq X," and breaks out of the loop. Since the
64  * simulated system is so much slower than the actual system, if a
65  * user is typing on the keyboard it is very easy for them to provide
66  * input at a fast enough rate to not allow the loop to exit and thus
67  * the error to be printed. This magic number provides a delay between
68  * the time the UART receives a character to send to the simulated
69  * system and the time it actually notifies the system it has a
70  * character to send to alleviate this problem. --Ali
71  */
72 void
74 {
75  static const Tick interval = 225 * SimClock::Int::ns;
76  DPRINTF(Uart, "Scheduling IER interrupt for %s, at cycle %lld\n",
77  event->name(), curTick() + interval);
78  if (!event->scheduled())
79  schedule(event, curTick() + interval);
80  else
81  reschedule(event, curTick() + interval);
82 }
83 
84 
86  : Uart(p, p.pio_size), registers(this, name() + ".registers"),
87  lastTxInt(0),
88  txIntrEvent([this]{ processIntrEvent(TX_INT); }, "TX"),
89  rxIntrEvent([this]{ processIntrEvent(RX_INT); }, "RX")
90 {
91 }
92 
93 Uart8250::Registers::Registers(Uart8250 *uart, const std::string &new_name) :
94  RegisterBankLE(new_name, 0), rbrThr(rbr, thr), rbrThrDll(rbrThr, dll),
95  ierDlh(ier, dlh), iirFcr(iir, fcr)
96 {
99  ier.writer(uart, &Uart8250::writeIer);
100  iir.reader(uart, &Uart8250::readIir);
101 
102  lcr.writer([this](auto &reg, const auto &value) {
103  reg.update(value);
104  rbrThrDll.select(value.dlab);
105  ierDlh.select(value.dlab);
106  });
107 
108  mcr.writer([](auto &reg, const auto &value) {
109  if (value == (UART_MCR_LOOP | 0x0A))
110  reg.update(0x9A);
111  });
112 
113  lsr.readonly().
114  reader([device = uart->device](auto &reg) {
115  Lsr lsr = 0;
116  if (device->dataAvailable())
117  lsr.rdr = 1;
118  lsr.tbe = 1;
119  lsr.txEmpty = 1;
120  return lsr;
121  });
122 
123  msr.readonly();
124 
126 }
127 
128 uint8_t
130 {
131  uint8_t data = 0;
132  if (device->dataAvailable())
133  data = device->readData();
134  else
135  DPRINTF(Uart, "empty read of RX register\n");
136 
137  status &= ~RX_INT;
139 
140  if (device->dataAvailable() && registers.ier.get().rdi)
142 
143  return data;
144 }
145 
146 void
148 {
151  status &= ~TX_INT;
152  if (registers.ier.get().thri)
154 }
155 
156 Uart8250::Iir
158 {
159  DPRINTF(Uart, "IIR Read, status = %#x\n", (uint32_t)status);
160 
161  Iir iir = 0;
162  if (status & RX_INT) {
163  // Rx data interrupt has a higher priority.
164  iir.id = (uint8_t)InterruptIds::Rx;
165  } else if (status & TX_INT) {
166  iir.id = (uint8_t)InterruptIds::Tx;
167  // Tx interrupts are cleared on IIR reads.
168  status &= ~TX_INT;
169  } else {
170  iir.pending = 1;
171  }
172  return iir;
173 }
174 
175 void
177 {
178  reg.update(ier);
179 
180  if (ier.thri) {
181  DPRINTF(Uart, "IER: IER_THRI set, scheduling TX intrrupt\n");
182  if (curTick() - lastTxInt > 225 * SimClock::Int::ns) {
183  DPRINTF(Uart, "-- Interrupting Immediately... %d,%d\n",
184  curTick(), lastTxInt);
186  } else {
187  DPRINTF(Uart, "-- Delaying interrupt... %d,%d\n",
188  curTick(), lastTxInt);
190  }
191  } else {
192  DPRINTF(Uart, "IER: IER_THRI cleared, descheduling TX intrrupt\n");
193  if (txIntrEvent.scheduled())
195  if (status & TX_INT)
197  status &= ~TX_INT;
198  }
199 
200  if (ier.rdi && device->dataAvailable()) {
201  DPRINTF(Uart, "IER: IER_RDI set, scheduling RX intrrupt\n");
203  } else {
204  DPRINTF(Uart, "IER: IER_RDI cleared, descheduling RX intrrupt\n");
205  if (rxIntrEvent.scheduled())
207  if (status & RX_INT)
209  status &= ~RX_INT;
210  }
211 }
212 
213 Tick
215 {
216  Addr daddr = pkt->getAddr() - pioAddr;
217 
218  DPRINTF(Uart, "Read register %#x\n", daddr);
219 
220  registers.read(daddr, pkt->getPtr<void>(), pkt->getSize());
221 
222  pkt->makeAtomicResponse();
223  return pioDelay;
224 }
225 
226 Tick
228 {
229  Addr daddr = pkt->getAddr() - pioAddr;
230 
231  DPRINTF(Uart, "Write register %#x value %#x\n", daddr,
232  pkt->getRaw<uint8_t>());
233 
234  registers.write(daddr, pkt->getPtr<void>(), pkt->getSize());
235 
236  pkt->makeAtomicResponse();
237  return pioDelay;
238 }
239 
240 void
242 {
243  // If the kernel wants an interrupt when we have data.
244  if (registers.ier.get().rdi) {
246  status |= RX_INT;
247  }
248 
249 }
250 
253 {
254  AddrRangeList ranges;
255  ranges.push_back(RangeSize(pioAddr, pioSize));
256  return ranges;
257 }
258 
259 void
261 {
263  paramOut(cp, "IER", registers.ier);
264  paramOut(cp, "LCR", registers.lcr);
265  paramOut(cp, "MCR", registers.mcr);
266  Tick rxintrwhen;
267  if (rxIntrEvent.scheduled())
268  rxintrwhen = rxIntrEvent.when();
269  else
270  rxintrwhen = 0;
271  Tick txintrwhen;
272  if (txIntrEvent.scheduled())
273  txintrwhen = txIntrEvent.when();
274  else
275  txintrwhen = 0;
276  SERIALIZE_SCALAR(rxintrwhen);
277  SERIALIZE_SCALAR(txintrwhen);
278 }
279 
280 void
282 {
284  paramIn(cp, "IER", registers.ier);
285  paramIn(cp, "LCR", registers.lcr);
286  paramIn(cp, "MCR", registers.mcr);
287  Tick rxintrwhen;
288  Tick txintrwhen;
289  UNSERIALIZE_SCALAR(rxintrwhen);
290  UNSERIALIZE_SCALAR(txintrwhen);
291  if (rxintrwhen != 0)
292  schedule(rxIntrEvent, rxintrwhen);
293  if (txintrwhen != 0)
294  schedule(txIntrEvent, txintrwhen);
295 }
RegisterBank::addRegisters
void addRegisters(std::initializer_list< std::reference_wrapper< RegisterBase >> regs)
Definition: reg_bank.hh:787
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
RegisterBank::Register
Definition: reg_bank.hh:485
Packet::makeAtomicResponse
void makeAtomicResponse()
Definition: packet.hh:1017
Uart8250::Registers::lcr
Register< Lcr > lcr
Definition: uart8250.hh:184
BasicPioDevice::pioAddr
Addr pioAddr
Address that the device listens to.
Definition: io_device.hh:148
SimClock::Int::ns
Tick ns
nanosecond
Definition: core.cc:62
Uart8250::Registers::ier
Register< Ier > ier
Definition: uart8250.hh:174
data
const char data[]
Definition: circlebuf.test.cc:47
serialize.hh
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
Packet::getAddr
Addr getAddr() const
Definition: packet.hh:755
Uart8250::Registers::thr
Register8 thr
Definition: uart8250.hh:167
EventManager::reschedule
void reschedule(Event &event, Tick when, bool always=false)
Definition: eventq.hh:1034
Uart8250::readIir
Iir readIir(Register< Iir > &reg)
Definition: uart8250.cc:157
Uart8250::Registers::iir
Register< Iir > iir
Definition: uart8250.hh:179
uart8250.hh
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
Uart8250::Registers::iirFcr
RWSwitchedRegister iirFcr
Definition: uart8250.hh:181
Uart8250::processIntrEvent
void processIntrEvent(int intrBit)
Definition: uart8250.cc:47
EventManager::deschedule
void deschedule(Event &event)
Definition: eventq.hh:1025
Uart8250::Registers::mcr
Register8 mcr
Definition: uart8250.hh:185
Packet::getSize
unsigned getSize() const
Definition: packet.hh:765
Uart8250::writeIer
void writeIer(Register< Ier > &reg, const Ier &ier)
Definition: uart8250.cc:176
RegisterBank::write
virtual void write(Addr addr, const void *buf, Addr bytes)
Definition: reg_bank.hh:862
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
Platform::clearConsoleInt
virtual void clearConsoleInt()=0
Clear a posted CPU interrupt.
Uart8250::writeThr
void writeThr(Register8 &reg, const uint8_t &data)
Definition: uart8250.cc:147
packet.hh
Uart8250::txIntrEvent
EventFunctionWrapper txIntrEvent
Definition: uart8250.hh:208
Uart8250::getAddrRanges
AddrRangeList getAddrRanges() const override
Determine the address ranges that this device responds to.
Definition: uart8250.cc:252
RegisterBank::read
virtual void read(Addr addr, void *buf, Addr bytes)
Definition: reg_bank.hh:805
Uart8250::write
Tick write(PacketPtr pkt) override
Pure virtual function that the device must implement.
Definition: uart8250.cc:227
cp
Definition: cprintf.cc:37
Uart8250::Registers::BankedRegister::select
void select(bool second)
Definition: uart8250.hh:124
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
Uart
Definition: uart.hh:46
Uart8250::registers
Registers registers
Definition: uart8250.hh:196
Event
Definition: eventq.hh:248
SerialDevice::dataAvailable
virtual bool dataAvailable() const =0
Check if there is pending data from the serial device.
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
RegisterBank::Register::reader
constexpr This & reader(const ReadFunc &new_reader)
Definition: reg_bank.hh:590
Uart8250
Definition: uart8250.hh:48
Uart::device
SerialDevice * device
Definition: uart.hh:51
MipsISA::event
Bitfield< 10, 5 > event
Definition: pra_constants.hh:297
RangeSize
AddrRange RangeSize(Addr start, Addr size)
Definition: addr_range.hh:651
BasicPioDevice::pioSize
Addr pioSize
Size that the device's address range.
Definition: io_device.hh:151
RegisterBank< ByteOrder::little >
Platform::postConsoleInt
virtual void postConsoleInt()=0
Cause the cpu to post a serial interrupt to the CPU.
inifile.hh
Uart::status
int status
Definition: uart.hh:49
TX_INT
const int TX_INT
Definition: uart.hh:44
SerialDevice::readData
virtual uint8_t readData()=0
Read a character from the device.
Uart8250::read
Tick read(PacketPtr pkt) override
Pure virtual function that the device must implement.
Definition: uart8250.cc:214
platform.hh
Uart8250::scheduleIntr
void scheduleIntr(Event *event)
Definition: uart8250.cc:73
Uart8250::Registers::sr
RegisterRaz sr
Definition: uart8250.hh:190
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
Uart8250::lastTxInt
Tick lastTxInt
Definition: uart8250.hh:203
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
Uart8250::rxIntrEvent
EventFunctionWrapper rxIntrEvent
Definition: uart8250.hh:209
paramOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
Definition: types.cc:37
packet_access.hh
Uart8250::Registers::msr
Register8 msr
Definition: uart8250.hh:187
EventFunctionWrapper::process
void process()
Definition: eventq.hh:1138
Packet::getRaw
T getRaw() const
Get the data in the packet without byte swapping.
Definition: packet_access.hh:49
Uart8250::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: uart8250.cc:260
RX_INT
const int RX_INT
Definition: uart.hh:43
Uart8250::readRbr
uint8_t readRbr(Register8 &reg)
Definition: uart8250.cc:129
UART_MCR_LOOP
const uint8_t UART_MCR_LOOP
Definition: uart8250.hh:43
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
Uart8250::Registers::ierDlh
BankedRegister ierDlh
Definition: uart8250.hh:176
RegisterBank::Register::readonly
constexpr This & readonly()
Definition: reg_bank.hh:584
Uart8250::dataAvailable
void dataAvailable() override
Inform the uart that there is data available.
Definition: uart8250.cc:241
BasicPioDevice::pioDelay
Tick pioDelay
Delay that the device experinces on an access.
Definition: io_device.hh:154
Uart8250::Registers::Registers
Registers(Uart8250 *uart, const std::string &new_name)
Definition: uart8250.cc:93
Packet::getPtr
T * getPtr()
get a pointer to the data ptr.
Definition: packet.hh:1158
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
SerialDevice::writeData
virtual void writeData(uint8_t c)=0
Transmit a character from the host interface to the device.
Uart8250::Registers::lsr
Register< Lsr > lsr
Definition: uart8250.hh:186
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
trace.hh
Uart8250::Uart8250
Uart8250(const Params &p)
Definition: uart8250.cc:85
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
Uart8250::Registers::rbrThrDll
BankedRegister rbrThrDll
Definition: uart8250.hh:171
std::list< AddrRange >
Uart8250::Registers::rbr
Register8 rbr
Definition: uart8250.hh:166
paramIn
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
Definition: types.cc:69
CheckpointIn
Definition: serialize.hh:68
PioDevice::Params
PioDeviceParams Params
Definition: io_device.hh:131
Uart8250::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: uart8250.cc:281
RegisterBank::Register::writer
constexpr This & writer(const WriteFunc &new_writer)
Definition: reg_bank.hh:605
Uart::platform
Platform * platform
Definition: uart.hh:50

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