blob: 30dbdca90db78234ef9d9777551b9eb8bee23b9c [file] [log] [blame]
Bin Meng7f5df8d2015-06-23 12:18:51 +08001/*
2 * Copyright (C) 2015, Bin Meng <bmeng.cn@gmail.com>
3 *
4 * Adapted from coreboot src/arch/x86/include/arch/smp/mpspec.h
5 *
6 * SPDX-License-Identifier: GPL-2.0+
7 */
8
9#ifndef __ASM_MPSPEC_H
10#define __ASM_MPSPEC_H
11
12/*
13 * Structure definitions for SMP machines following the
14 * Intel MultiProcessor Specification 1.4
15 */
16
17#define MPSPEC_V14 4
18
19#define MPF_SIGNATURE "_MP_"
20
21struct mp_floating_table {
22 char mpf_signature[4]; /* "_MP_" */
23 u32 mpf_physptr; /* Configuration table address */
24 u8 mpf_length; /* Our length (paragraphs) */
25 u8 mpf_spec; /* Specification version */
26 u8 mpf_checksum; /* Checksum (makes sum 0) */
27 u8 mpf_feature1; /* Predefined or Unique configuration? */
28 u8 mpf_feature2; /* Bit7 set for IMCR/PIC */
29 u8 mpf_feature3; /* Unused (0) */
30 u8 mpf_feature4; /* Unused (0) */
31 u8 mpf_feature5; /* Unused (0) */
32};
33
34#define MPC_SIGNATURE "PCMP"
35
36struct mp_config_table {
37 char mpc_signature[4]; /* "PCMP" */
38 u16 mpc_length; /* Size of table */
39 u8 mpc_spec; /* Specification version */
40 u8 mpc_checksum; /* Checksum (makes sum 0) */
41 char mpc_oem[8]; /* OEM ID */
42 char mpc_product[12]; /* Product ID */
43 u32 mpc_oemptr; /* OEM table address */
44 u16 mpc_oemsize; /* OEM table size */
45 u16 mpc_entry_count; /* Number of entries in the table */
46 u32 mpc_lapic; /* Local APIC address */
47 u16 mpe_length; /* Extended table size */
48 u8 mpe_checksum; /* Extended table checksum */
49 u8 reserved;
50};
51
52/* Base MP configuration table entry types */
53
54enum mp_base_config_entry_type {
55 MP_PROCESSOR,
56 MP_BUS,
57 MP_IOAPIC,
58 MP_INTSRC,
59 MP_LINTSRC
60};
61
62#define MPC_CPU_EN (1 << 0)
63#define MPC_CPU_BP (1 << 1)
64
65struct mpc_config_processor {
66 u8 mpc_type;
67 u8 mpc_apicid;
68 u8 mpc_apicver;
69 u8 mpc_cpuflag;
70 u32 mpc_cpusignature;
71 u32 mpc_cpufeature;
72 u32 mpc_reserved[2];
73};
74
75#define BUSTYPE_CBUS "CBUS "
76#define BUSTYPE_CBUSII "CBUSII"
77#define BUSTYPE_EISA "EISA "
78#define BUSTYPE_FUTURE "FUTURE"
79#define BUSTYPE_INTERN "INTERN"
80#define BUSTYPE_ISA "ISA "
81#define BUSTYPE_MBI "MBI "
82#define BUSTYPE_MBII "MBII "
83#define BUSTYPE_MCA "MCA "
84#define BUSTYPE_MPI "MPI "
85#define BUSTYPE_MPSA "MPSA "
86#define BUSTYPE_NUBUS "NUBUS "
87#define BUSTYPE_PCI "PCI "
88#define BUSTYPE_PCMCIA "PCMCIA"
89#define BUSTYPE_TC "TC "
90#define BUSTYPE_VL "VL "
91#define BUSTYPE_VME "VME "
92#define BUSTYPE_XPRESS "XPRESS"
93
94struct mpc_config_bus {
95 u8 mpc_type;
96 u8 mpc_busid;
97 u8 mpc_bustype[6];
98};
99
100#define MPC_APIC_USABLE (1 << 0)
101
102struct mpc_config_ioapic {
103 u8 mpc_type;
104 u8 mpc_apicid;
105 u8 mpc_apicver;
106 u8 mpc_flags;
107 u32 mpc_apicaddr;
108};
109
110enum mp_irq_source_types {
111 MP_INT,
112 MP_NMI,
113 MP_SMI,
114 MP_EXTINT
115};
116
117#define MP_IRQ_POLARITY_DEFAULT 0x0
118#define MP_IRQ_POLARITY_HIGH 0x1
119#define MP_IRQ_POLARITY_LOW 0x3
120#define MP_IRQ_POLARITY_MASK 0x3
121#define MP_IRQ_TRIGGER_DEFAULT 0x0
122#define MP_IRQ_TRIGGER_EDGE 0x4
123#define MP_IRQ_TRIGGER_LEVEL 0xc
124#define MP_IRQ_TRIGGER_MASK 0xc
125
126#define MP_APIC_ALL 0xff
127
128struct mpc_config_intsrc {
129 u8 mpc_type;
130 u8 mpc_irqtype;
131 u16 mpc_irqflag;
132 u8 mpc_srcbus;
133 u8 mpc_srcbusirq;
134 u8 mpc_dstapic;
135 u8 mpc_dstirq;
136};
137
138struct mpc_config_lintsrc {
139 u8 mpc_type;
140 u8 mpc_irqtype;
141 u16 mpc_irqflag;
142 u8 mpc_srcbusid;
143 u8 mpc_srcbusirq;
144 u8 mpc_destapic;
145 u8 mpc_destlint;
146};
147
148/* Extended MP configuration table entry types */
149
150enum mp_ext_config_entry_type {
151 MPE_SYSTEM_ADDRESS_SPACE = 128,
152 MPE_BUS_HIERARCHY,
153 MPE_COMPAT_ADDRESS_SPACE
154};
155
156struct mp_ext_config {
157 u8 mpe_type;
158 u8 mpe_length;
159};
160
161#define ADDRESS_TYPE_IO 0
162#define ADDRESS_TYPE_MEM 1
163#define ADDRESS_TYPE_PREFETCH 2
164
165struct mp_ext_system_address_space {
166 u8 mpe_type;
167 u8 mpe_length;
168 u8 mpe_busid;
169 u8 mpe_addr_type;
170 u32 mpe_addr_base_low;
171 u32 mpe_addr_base_high;
172 u32 mpe_addr_length_low;
173 u32 mpe_addr_length_high;
174};
175
176#define BUS_SUBTRACTIVE_DECODE (1 << 0)
177
178struct mp_ext_bus_hierarchy {
179 u8 mpe_type;
180 u8 mpe_length;
181 u8 mpe_busid;
182 u8 mpe_bus_info;
183 u8 mpe_parent_busid;
184 u8 reserved[3];
185};
186
187#define ADDRESS_RANGE_ADD 0
188#define ADDRESS_RANGE_SUBTRACT 1
189
190/*
191 * X100 - X3FF
192 * X500 - X7FF
193 * X900 - XBFF
194 * XD00 - XFFF
195 */
196#define RANGE_LIST_IO_ISA 0
197/*
198 * X3B0 - X3BB
199 * X3C0 - X3DF
200 * X7B0 - X7BB
201 * X7C0 - X7DF
202 * XBB0 - XBBB
203 * XBC0 - XBDF
204 * XFB0 - XFBB
205 * XFC0 - XCDF
206 */
207#define RANGE_LIST_IO_VGA 1
208
209struct mp_ext_compat_address_space {
210 u8 mpe_type;
211 u8 mpe_length;
212 u8 mpe_busid;
213 u8 mpe_addr_modifier;
214 u32 mpe_range_list;
215};
216
217/**
218 * mp_next_mpc_entry() - Compute MP configuration table end to be used as
219 * next base table entry start address
220 *
221 * This computes the end address of current MP configuration table, without
222 * counting any extended configuration table entry.
223 *
224 * @mc: configuration table header address
225 * @return: configuration table end address
226 */
Simon Glasse71ffd02017-01-16 07:03:36 -0700227static inline ulong mp_next_mpc_entry(struct mp_config_table *mc)
Bin Meng7f5df8d2015-06-23 12:18:51 +0800228{
Simon Glasse71ffd02017-01-16 07:03:36 -0700229 return (ulong)mc + mc->mpc_length;
Bin Meng7f5df8d2015-06-23 12:18:51 +0800230}
231
232/**
233 * mp_add_mpc_entry() - Add a base MP configuration table entry
234 *
235 * This adds the base MP configuration table entry size with
236 * added base table entry length and increases entry count by 1.
237 *
238 * @mc: configuration table header address
239 * @length: length of the added table entry
240 */
241static inline void mp_add_mpc_entry(struct mp_config_table *mc, uint length)
242{
243 mc->mpc_length += length;
244 mc->mpc_entry_count++;
245}
246
247/**
248 * mp_next_mpe_entry() - Compute MP configuration table end to be used as
249 * next extended table entry start address
250 *
251 * This computes the end address of current MP configuration table,
252 * including any extended configuration table entry.
253 *
254 * @mc: configuration table header address
255 * @return: configuration table end address
256 */
Simon Glasse71ffd02017-01-16 07:03:36 -0700257static inline ulong mp_next_mpe_entry(struct mp_config_table *mc)
Bin Meng7f5df8d2015-06-23 12:18:51 +0800258{
Simon Glasse71ffd02017-01-16 07:03:36 -0700259 return (ulong)mc + mc->mpc_length + mc->mpe_length;
Bin Meng7f5df8d2015-06-23 12:18:51 +0800260}
261
262/**
263 * mp_add_mpe_entry() - Add an extended MP configuration table entry
264 *
265 * This adds the extended MP configuration table entry size with
266 * added extended table entry length.
267 *
268 * @mc: configuration table header address
269 * @mpe: extended table entry base address
270 */
271static inline void mp_add_mpe_entry(struct mp_config_table *mc,
272 struct mp_ext_config *mpe)
273{
274 mc->mpe_length += mpe->mpe_length;
275}
276
277/**
278 * mp_write_floating_table() - Write the MP floating table
279 *
280 * This writes the MP floating table, and points MP configuration table
281 * to its end address so that MP configuration table follows immediately
282 * after the floating table.
283 *
284 * @mf: MP floating table base address
285 * @return: MP configuration table header address
286 */
287struct mp_config_table *mp_write_floating_table(struct mp_floating_table *mf);
288
289/**
290 * mp_config_table_init() - Initialize the MP configuration table header
291 *
292 * This populates the MP configuration table header with valid bits.
293 *
294 * @mc: MP configuration table header address
295 */
296void mp_config_table_init(struct mp_config_table *mc);
297
298/**
299 * mp_write_processor() - Write a processor entry
300 *
301 * This writes a processor entry to the configuration table.
302 *
303 * @mc: MP configuration table header address
304 */
305void mp_write_processor(struct mp_config_table *mc);
306
307/**
308 * mp_write_bus() - Write a bus entry
309 *
310 * This writes a bus entry to the configuration table.
311 *
312 * @mc: MP configuration table header address
313 * @id: bus id
314 * @bustype: bus type name
315 */
316void mp_write_bus(struct mp_config_table *mc, int id, const char *bustype);
317
318/**
319 * mp_write_ioapic() - Write an I/O APIC entry
320 *
321 * This writes an I/O APIC entry to the configuration table.
322 *
323 * @mc: MP configuration table header address
324 * @id: I/O APIC id
325 * @ver: I/O APIC version
326 * @apicaddr: I/O APIC address
327 */
328void mp_write_ioapic(struct mp_config_table *mc, int id, int ver, u32 apicaddr);
329
330/**
331 * mp_write_intsrc() - Write an I/O interrupt assignment entry
332 *
333 * This writes an I/O interrupt assignment entry to the configuration table.
334 *
335 * @mc: MP configuration table header address
336 * @irqtype: IRQ type (INT/NMI/SMI/ExtINT)
337 * @irqflag: IRQ flag (level/trigger)
338 * @srcbus: source bus id where the interrupt comes from
339 * @srcbusirq: IRQ number mapped on the source bus
340 * @dstapic: destination I/O APIC id where the interrupt goes to
341 * @dstirq: destination I/O APIC pin where the interrupt goes to
342 */
343void mp_write_intsrc(struct mp_config_table *mc, int irqtype, int irqflag,
344 int srcbus, int srcbusirq, int dstapic, int dstirq);
345
346/**
347 * mp_write_pci_intsrc() - Write a PCI interrupt assignment entry
348 *
349 * This writes a PCI interrupt assignment entry to the configuration table.
350 *
351 * @mc: MP configuration table header address
352 * @irqtype: IRQ type (INT/NMI/SMI/ExtINT)
353 * @srcbus: PCI bus number where the interrupt comes from
354 * @dev: device number on the PCI bus
355 * @pin: PCI interrupt pin (INT A/B/C/D)
356 * @dstapic: destination I/O APIC id where the interrupt goes to
357 * @dstirq: destination I/O APIC pin where the interrupt goes to
358 */
359void mp_write_pci_intsrc(struct mp_config_table *mc, int irqtype,
360 int srcbus, int dev, int pin, int dstapic, int dstirq);
361
362/**
363 * mp_write_lintsrc() - Write a local interrupt assignment entry
364 *
365 * This writes a local interrupt assignment entry to the configuration table.
366 *
367 * @mc: MP configuration table header address
368 * @irqtype: IRQ type (INT/NMI/SMI/ExtINT)
369 * @irqflag: IRQ flag (level/trigger)
370 * @srcbus: PCI bus number where the interrupt comes from
371 * @srcbusirq: IRQ number mapped on the source bus
372 * @dstapic: destination local APIC id where the interrupt goes to
373 * @destlint: destination local APIC pin where the interrupt goes to
374 */
375void mp_write_lintsrc(struct mp_config_table *mc, int irqtype, int irqflag,
376 int srcbus, int srcbusirq, int destapic, int destlint);
377
378
379/**
380 * mp_write_address_space() - Write a system address space entry
381 *
382 * This writes a system address space entry to the configuration table.
383 *
384 * @mc: MP configuration table header address
385 * @busid: bus id for the bus where system address space is mapped
386 * @addr_type: system address type
387 * @addr_base_low: starting address low
388 * @addr_base_high: starting address high
389 * @addr_length_low: address length low
390 * @addr_length_high: address length high
391 */
392void mp_write_address_space(struct mp_config_table *mc,
393 int busid, int addr_type,
394 u32 addr_base_low, u32 addr_base_high,
395 u32 addr_length_low, u32 addr_length_high);
396
397/**
398 * mp_write_bus_hierarchy() - Write a bus hierarchy descriptor entry
399 *
400 * This writes a bus hierarchy descriptor entry to the configuration table.
401 *
402 * @mc: MP configuration table header address
403 * @busid: bus id
404 * @bus_info: bit0 indicates if the bus is a subtractive decode bus
405 * @parent_busid: parent bus id
406 */
407void mp_write_bus_hierarchy(struct mp_config_table *mc,
408 int busid, int bus_info, int parent_busid);
409
410/**
411 * mp_write_compat_address_space() - Write a compat bus address space entry
412 *
413 * This writes a compatibility bus address space modifier entry to the
414 * configuration table.
415 *
416 * @mc: MP configuration table header address
417 * @busid: bus id
418 * @addr_modifier: add or subtract to predefined address range list
419 * @range_list: list of predefined address space ranges
420 */
421void mp_write_compat_address_space(struct mp_config_table *mc, int busid,
422 int addr_modifier, u32 range_list);
423
424/**
425 * mptable_finalize() - Finalize the MP table
426 *
427 * This finalizes the MP table by calculating required checksums.
428 *
429 * @mc: MP configuration table header address
430 * @return: MP table end address
431 */
432u32 mptable_finalize(struct mp_config_table *mc);
433
Bin Meng07545d82015-06-23 12:18:52 +0800434/**
Bin Mengabab9122015-07-22 01:21:09 -0700435 * mp_determine_pci_dstirq() - Determine PCI device's int pin on the I/O APIC
436 *
437 * This determines a PCI device's interrupt pin number on the I/O APIC.
438 *
439 * This can be implemented by platform codes to handle specifal cases, which
440 * do not conform to the normal chipset/board design where PIRQ[A-H] are mapped
441 * directly to I/O APIC INTPIN#16-23.
442 *
443 * @bus: bus number of the pci device
444 * @dev: device number of the pci device
445 * @func: function number of the pci device
446 * @pirq: PIRQ number the PCI device's interrupt pin is routed to
447 * @return: interrupt pin number on the I/O APIC
448 */
449int mp_determine_pci_dstirq(int bus, int dev, int func, int pirq);
450
451/**
Bin Meng07545d82015-06-23 12:18:52 +0800452 * write_mp_table() - Write MP table
453 *
454 * This writes MP table at a given address.
455 *
456 * @addr: start address to write MP table
457 * @return: end address of MP table
458 */
Simon Glass42fd8c12017-01-16 07:03:35 -0700459ulong write_mp_table(ulong addr);
Bin Meng07545d82015-06-23 12:18:52 +0800460
Bin Meng7f5df8d2015-06-23 12:18:51 +0800461#endif /* __ASM_MPSPEC_H */