Go to the documentation of this file.
36 #include "debug/I82094AA.hh"
47 lowestPriorityOffset(0),
48 intRequestPort(
name() +
".int_request", this, this,
p.int_latency)
54 assert(
p.apic_id < 0xff);
58 RedirTableEntry entry = 0;
65 for (
int i = 0;
i <
p.port_inputs_connection_count;
i++)
78 panic_if(!intRequestPort.isConnected(),
79 "Int port not connected to anything!");
85 if (if_name ==
"int_requestor")
86 return intRequestPort;
87 if (if_name ==
"inputs")
88 return *inputs.at(idx);
100 pkt->
setLE<uint32_t>(regSel);
103 pkt->
setLE<uint32_t>(readReg(regSel));
106 panic(
"Illegal read from I/O APIC.\n");
119 regSel = pkt->
getLE<uint32_t>();
122 writeReg(regSel, pkt->
getLE<uint32_t>());
125 panic(
"Illegal write to I/O APIC.\n");
135 id =
bits(value, 31, 24);
136 }
else if (
offset == 0x1) {
138 }
else if (
offset == 0x2) {
139 arbId =
bits(value, 31, 24);
140 }
else if (
offset >= 0x10 &&
offset <= (0x10 + TableSize * 2)) {
143 redirTable[
index].topDW = value;
144 redirTable[
index].topReserved = 0;
146 redirTable[
index].bottomDW = value;
147 redirTable[
index].bottomReserved = 0;
150 warn(
"Access to undefined I/O APIC register %#x.\n",
offset);
153 "Wrote %#x to I/O APIC register %#x .\n", value,
offset);
162 }
else if (
offset == 0x1) {
163 result = ((TableSize - 1) << 16) | APICVersion;
164 }
else if (
offset == 0x2) {
165 result = arbId << 24;
166 }
else if (
offset >= 0x10 &&
offset <= (0x10 + TableSize * 2)) {
169 result = redirTable[
index].topDW;
171 result = redirTable[
index].bottomDW;
174 warn(
"Access to undefined I/O APIC register %#x.\n",
offset);
177 "Read %#x from I/O APIC register %#x.\n", result,
offset);
185 assert(line < TableSize);
186 RedirTableEntry entry = redirTable[line];
191 TriggerIntMessage message = 0;
192 message.destination = entry.dest;
195 message.vector = extIntPic->getVector();
197 message.vector = entry.vector;
199 message.deliveryMode = entry.deliveryMode;
200 message.destMode = entry.destMode;
201 message.level = entry.polarity;
202 message.trigger = entry.trigger;
204 int numContexts = sys->threads.size();
205 if (message.destMode == 0) {
207 panic(
"Lowest priority delivery mode from the "
208 "IO APIC aren't supported in physical "
209 "destination mode.\n");
211 if (message.destination == 0xFF) {
212 for (
int i = 0;
i < numContexts;
i++) {
216 apics.push_back(message.destination);
219 for (
int i = 0;
i < numContexts;
i++) {
221 getCpuPtr()->getInterruptController(0);
222 auto *localApic =
dynamic_cast<Interrupts *
>(base_int);
224 message.destination) {
225 apics.push_back(localApic->getInitialApicId());
234 uint64_t modOffset = lowestPriorityOffset % apics.size();
235 lowestPriorityOffset++;
236 auto apicIt = apics.begin();
237 while (modOffset--) {
239 assert(apicIt != apics.end());
241 int selected = *apicIt;
243 apics.push_back(selected);
246 for (
auto id: apics) {
248 intRequestPort.sendMessage(pkt, sys->isTimingMode());
256 assert(number < TableSize);
257 if (!pinStates[number])
258 signalInterrupt(number);
259 pinStates[number] =
true;
265 assert(number < TableSize);
266 pinStates[number] =
false;
272 uint64_t* redirTableArray = (uint64_t*)redirTable;
285 uint64_t redirTableArray[TableSize];
293 for (
int i = 0;
i < TableSize;
i++) {
294 redirTable[
i] = (RedirTableEntry)redirTableArray[
i];
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Tick read(PacketPtr pkt) override
Pure virtual function that the device must implement.
@ APIC_LOGICAL_DESTINATION
#define UNSERIALIZE_SCALAR(scalar)
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
std::vector< IntSinkPin< I82094AA > * > inputs
I82094AA(const Params &p)
std::string csprintf(const char *format, const Args &...args)
void makeAtomicResponse()
void init() override
init() is called after all C++ SimObjects have been created and all ports are connected.
void lowerInterruptPin(int number)
EndBitUnion(RedirTableEntry) protected uint8_t regSel
void unserialize(CheckpointIn &cp) override
Unserialize an object.
bool pinStates[TableSize]
virtual std::string name() const
void writeReg(uint8_t offset, uint32_t value)
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
uint64_t Tick
Tick count type.
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
void raiseInterruptPin(int number)
#define SERIALIZE_ARRAY(member, size)
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
const std::string & name()
#define SERIALIZE_SCALAR(scalar)
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
static PacketPtr buildIntTriggerPacket(int id, TriggerIntMessage message)
void serialize(CheckpointOut &cp) const override
Serialize an object.
Ports are used to interface objects to each other.
#define UNSERIALIZE_ARRAY(member, size)
Tick write(PacketPtr pkt) override
Pure virtual function that the device must implement.
RedirTableEntry redirTable[TableSize]
uint32_t readReg(uint8_t offset)
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
std::ostream CheckpointOut
void setLE(T v)
Set the value in the data pointer to v as little endian.
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
static const uint8_t TableSize
void signalInterrupt(int line)
#define panic(...)
This implements a cprintf based panic() function.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Generated on Wed Jul 28 2021 12:10:27 for gem5 by doxygen 1.8.17