ReactOS 0.4.15-dev-8632-gbc8c7d1
madt.c
Go to the documentation of this file.
1/*
2 * PROJECT: ReactOS Kernel
3 * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later)
4 * PURPOSE: Source File for MADT Table parsing
5 * COPYRIGHT: Copyright 2021 Justin Miller <justinmiller100@gmail.com>
6 * Copyright 2023 Serge Gautherie <reactos-git_serge_171003@gautherie.fr>
7 */
8
9/* INCLUDES *******************************************************************/
10
11#include <hal.h>
12#include <acpi.h>
13/* ACPI_BIOS_ERROR defined in acoutput.h and bugcodes.h */
14#undef ACPI_BIOS_ERROR
15#include <smp.h>
16
17#define NDEBUG
18#include <debug.h>
19
20// See HalpParseApicTables(). Only enable this to local-debug it.
21// That needs, for example, to test-call the function later or to use the "FrLdrDbgPrint" hack.
22#if DBG && 0
23 #define DPRINT01 DPRINT1
24 #define DPRINT00 DPRINT
25#else
26#if defined(_MSC_VER)
27 #define DPRINT01 __noop
28 #define DPRINT00 __noop
29#else
30 #define DPRINT01(...) do { if(0) { DbgPrint(__VA_ARGS__); } } while(0)
31 #define DPRINT00(...) do { if(0) { DbgPrint(__VA_ARGS__); } } while(0)
32#endif // _MSC_VER
33#endif // DBG && 0
34
35/* GLOBALS ********************************************************************/
36
38
39// ACPI_MADT_LOCAL_APIC.LapicFlags masks
40#define LAPIC_FLAG_ENABLED 0x00000001
41#define LAPIC_FLAG_ONLINE_CAPABLE 0x00000002
42// Bits 2-31 are reserved.
43
46
47#if 0
49#endif
50
51/* FUNCTIONS ******************************************************************/
52
53// Note: HalpParseApicTables() is called early, so its DPRINT*() do nothing.
54VOID
56 _In_ PLOADER_PARAMETER_BLOCK LoaderBlock)
57{
58 ACPI_TABLE_MADT *MadtTable;
59 ACPI_SUBTABLE_HEADER *AcpiHeader;
60 ULONG_PTR TableEnd;
61
62 MadtTable = HalAcpiGetTable(LoaderBlock, APIC_SIGNATURE);
63 if (!MadtTable)
64 {
65 DPRINT01("MADT table not found\n");
66 return;
67 }
68
69 if (MadtTable->Header.Length < sizeof(*MadtTable))
70 {
71 DPRINT01("Length is too short: %p, %u\n", MadtTable, MadtTable->Header.Length);
72 return;
73 }
74
75 DPRINT00("MADT table: Address %08X, Flags %08X\n", MadtTable->Address, MadtTable->Flags);
76
77#if 1
78
79 // TODO: We support only legacy APIC for now
81 // TODO: What about 'MadtTable->Flags & ACPI_MADT_PCAT_COMPAT'?
82
83#else // TODO: Is that correct?
84
86 {
88 }
89 else // if ((MadtTable->Flags & ACPI_MADT_PCAT_COMPAT) == ACPI_MADT_MULTIPLE_APIC)
90 {
91#if 1
92 DPRINT01("ACPI_MADT_MULTIPLE_APIC support is UNIMPLEMENTED\n");
93 return;
94#else
95 HalpApicInfoTable.ApicMode = HALP_APIC_MODE_xyz;
96#endif
97 }
98
99#endif
100
102
103 AcpiHeader = (ACPI_SUBTABLE_HEADER *)((ULONG_PTR)MadtTable + sizeof(*MadtTable));
104 TableEnd = (ULONG_PTR)MadtTable + MadtTable->Header.Length;
105 DPRINT00(" MadtTable %p, subtables %p - %p\n", MadtTable, AcpiHeader, (PVOID)TableEnd);
106
107 while ((ULONG_PTR)(AcpiHeader + 1) <= TableEnd)
108 {
109 if (AcpiHeader->Length < sizeof(*AcpiHeader))
110 {
111 DPRINT01("Length is too short: %p, %u\n", AcpiHeader, AcpiHeader->Length);
112 return;
113 }
114
115 if ((ULONG_PTR)AcpiHeader + AcpiHeader->Length > TableEnd)
116 {
117 DPRINT01("Length mismatch: %p, %u, %p\n",
118 AcpiHeader, AcpiHeader->Length, (PVOID)TableEnd);
119 return;
120 }
121
122 switch (AcpiHeader->Type)
123 {
125 {
126 ACPI_MADT_LOCAL_APIC *LocalApic = (ACPI_MADT_LOCAL_APIC *)AcpiHeader;
127
128 if (AcpiHeader->Length != sizeof(*LocalApic))
129 {
130 DPRINT01("Type/Length mismatch: %p, %u\n", AcpiHeader, AcpiHeader->Length);
131 return;
132 }
133
134 DPRINT00(" Local Apic, Processor %lu: ProcessorId %u, Id %u, LapicFlags %08X\n",
136 LocalApic->ProcessorId, LocalApic->Id, LocalApic->LapicFlags);
137
139 {
140 DPRINT00(" Ignored: unusable\n");
141 break;
142 }
143
145 {
146 DPRINT00(" Skipped: array is full\n");
147 // We assume ignoring this processor is acceptable, until proven otherwise.
148 break;
149 }
150
151 // Note: ProcessorId and Id are not validated in any way (yet).
153 LocalApic->ProcessorId;
155
157
158 break;
159 }
161 {
162 ACPI_MADT_IO_APIC *IoApic = (ACPI_MADT_IO_APIC *)AcpiHeader;
163
164 if (AcpiHeader->Length != sizeof(*IoApic))
165 {
166 DPRINT01("Type/Length mismatch: %p, %u\n", AcpiHeader, AcpiHeader->Length);
167 return;
168 }
169
170 DPRINT00(" Io Apic: Id %u, Address %08X, GlobalIrqBase %08X\n",
171 IoApic->Id, IoApic->Address, IoApic->GlobalIrqBase);
172
173 // Ensure HalpApicInfoTable.IOAPICCount consistency.
174 if (HalpApicInfoTable.IoApicPA[IoApic->Id] != 0)
175 {
176 DPRINT01("Id duplication: %p, %u\n", IoApic, IoApic->Id);
177 return;
178 }
179
180 // Note: Address and GlobalIrqBase are not validated in any way (yet).
181 HalpApicInfoTable.IoApicPA[IoApic->Id] = IoApic->Address;
183
185
186 break;
187 }
189 {
190 ACPI_MADT_INTERRUPT_OVERRIDE *InterruptOverride =
191 (ACPI_MADT_INTERRUPT_OVERRIDE *)AcpiHeader;
192
193 if (AcpiHeader->Length != sizeof(*InterruptOverride))
194 {
195 DPRINT01("Type/Length mismatch: %p, %u\n", AcpiHeader, AcpiHeader->Length);
196 return;
197 }
198
199 DPRINT00(" Interrupt Override: Bus %u, SourceIrq %u, GlobalIrq %08X, IntiFlags %04X / UNIMPLEMENTED\n",
200 InterruptOverride->Bus, InterruptOverride->SourceIrq,
201 InterruptOverride->GlobalIrq, InterruptOverride->IntiFlags);
202
203 if (InterruptOverride->Bus != 0) // 0 = ISA
204 {
205 DPRINT01("Invalid Bus: %p, %u\n", InterruptOverride, InterruptOverride->Bus);
206 return;
207 }
208
209#if 1
210 // TODO: Implement it.
211#else // TODO: Is that correct?
212 if (InterruptOverride->SourceIrq > _countof(HalpPicVectorRedirect))
213 {
214 DPRINT01("Invalid SourceIrq: %p, %u\n",
215 InterruptOverride, InterruptOverride->SourceIrq);
216 return;
217 }
218
219 // Note: GlobalIrq is not validated in any way (yet).
220 HalpPicVectorRedirect[InterruptOverride->SourceIrq] = InterruptOverride->GlobalIrq;
221 // TODO: What about 'InterruptOverride->IntiFlags'?
222#endif
223
224 break;
225 }
226 default:
227 {
228 DPRINT01(" UNIMPLEMENTED: Type %u, Length %u\n",
229 AcpiHeader->Type, AcpiHeader->Length);
230 return;
231 }
232 }
233
234 AcpiHeader = (ACPI_SUBTABLE_HEADER *)((ULONG_PTR)AcpiHeader + AcpiHeader->Length);
235 }
236
237 if ((ULONG_PTR)AcpiHeader != TableEnd)
238 {
239 DPRINT01("Length mismatch: %p, %p, %p\n", MadtTable, AcpiHeader, (PVOID)TableEnd);
240 return;
241 }
242}
243
244VOID
246{
247#if DBG
248 ULONG i;
249
250 DPRINT1("Physical processor count: %lu\n", HalpApicInfoTable.ProcessorCount);
251 for (i = 0; i < HalpApicInfoTable.ProcessorCount; i++)
252 {
253 DPRINT1(" Processor %lu: ProcessorId %u, LapicId %u, ProcessorStarted %u, BSPCheck %u, ProcessorPrcb %p\n",
254 i,
255 HalpProcessorIdentity[i].ProcessorId,
256 HalpProcessorIdentity[i].LapicId,
257 HalpProcessorIdentity[i].ProcessorStarted,
258 HalpProcessorIdentity[i].BSPCheck,
259 HalpProcessorIdentity[i].ProcessorPrcb);
260 }
261#endif
262}
@ ACPI_MADT_TYPE_IO_APIC
Definition: actbl2.h:1006
@ ACPI_MADT_TYPE_INTERRUPT_OVERRIDE
Definition: actbl2.h:1007
@ ACPI_MADT_TYPE_LOCAL_APIC
Definition: actbl2.h:1005
#define ACPI_MADT_PCAT_COMPAT
Definition: actbl2.h:993
#define ACPI_MADT_DUAL_PIC
Definition: actbl2.h:997
#define DPRINT1
Definition: precomp.h:8
#define ULONG_PTR
Definition: config.h:101
#define MAXIMUM_PROCESSORS
Definition: rwlock.h:5
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
ULONG HalpPicVectorRedirect[]
Definition: halacpi.c:40
PVOID NTAPI HalAcpiGetTable(IN PLOADER_PARAMETER_BLOCK LoaderBlock, IN ULONG Signature)
Definition: halacpi.c:446
VOID HalpParseApicTables(_In_ PLOADER_PARAMETER_BLOCK LoaderBlock)
Definition: madt.c:55
VOID HalpPrintApicTables(VOID)
Definition: madt.c:245
HALP_APIC_INFO_TABLE HalpApicInfoTable
Definition: madt.c:37
#define DPRINT00(...)
Definition: madt.c:31
static PROCESSOR_IDENTITY HalpStaticProcessorIdentity[MAXIMUM_PROCESSORS]
Definition: madt.c:44
const PPROCESSOR_IDENTITY HalpProcessorIdentity
Definition: madt.c:45
#define DPRINT01(...)
Definition: madt.c:30
#define LAPIC_FLAG_ENABLED
Definition: madt.c:40
#define LAPIC_FLAG_ONLINE_CAPABLE
Definition: madt.c:41
#define _In_
Definition: ms_sal.h:308
#define APIC_SIGNATURE
Definition: acpi.h:33
#define HALP_APIC_MODE_LEGACY
Definition: smp.h:35
#define _countof(array)
Definition: sndvol32.h:70
ULONG ProcessorCount
Definition: smp.h:25
ULONG IoApicPA[HALP_APIC_INFO_TABLE_IOAPIC_NUMBER]
Definition: smp.h:29
ULONG IOAPICCount
Definition: smp.h:26
ULONG ApicMode
Definition: smp.h:24
ULONG IoApicIrqBase[HALP_APIC_INFO_TABLE_IOAPIC_NUMBER]
Definition: smp.h:30
ULONG LocalApicPA
Definition: smp.h:27
UCHAR ProcessorId
Definition: smp.h:13
UCHAR LapicId
Definition: smp.h:14
UINT32 Address
Definition: actbl2.h:1057
UINT32 GlobalIrqBase
Definition: actbl2.h:1058
UINT32 Length
Definition: actbl.h:109
ACPI_TABLE_HEADER Header
Definition: actbl2.h:985
UINT32 Address
Definition: actbl2.h:986
UINT32 Flags
Definition: actbl2.h:987
uint32_t ULONG_PTR
Definition: typedefs.h:65
uint32_t ULONG
Definition: typedefs.h:59