blob: f229b13a5f3020175d9b940c40c5d3602747ae5b [file] [log] [blame]
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -04001#ifndef _ASM_X86_IOMMU_TABLE_H
2#define _ASM_X86_IOMMU_TABLE_H
3
4#include <asm/swiotlb.h>
5
6/*
7 * History lesson:
8 * The execution chain of IOMMUs in 2.6.36 looks as so:
9 *
10 * [xen-swiotlb]
11 * |
12 * +----[swiotlb *]--+
13 * / | \
14 * / | \
15 * [GART] [Calgary] [Intel VT-d]
16 * /
17 * /
18 * [AMD-Vi]
19 *
20 * *: if SWIOTLB detected 'iommu=soft'/'swiotlb=force' it would skip
21 * over the rest of IOMMUs and unconditionally initialize the SWIOTLB.
22 * Also it would surreptitiously initialize set the swiotlb=1 if there were
23 * more than 4GB and if the user did not pass in 'iommu=off'. The swiotlb
24 * flag would be turned off by all IOMMUs except the Calgary one.
25 *
26 * The IOMMU_INIT* macros allow a similar tree (or more complex if desired)
27 * to be built by defining who we depend on.
28 *
29 * And all that needs to be done is to use one of the macros in the IOMMU
30 * and the pci-dma.c will take care of the rest.
31 */
32
33struct iommu_table_entry {
34 initcall_t detect;
35 initcall_t depend;
36 void (*early_init)(void); /* No memory allocate available. */
37 void (*late_init)(void); /* Yes, can allocate memory. */
38#define IOMMU_FINISH_IF_DETECTED (1<<0)
39#define IOMMU_DETECTED (1<<1)
40 int flags;
41};
42/*
43 * Macro fills out an entry in the .iommu_table that is equivalent
44 * to the fields that 'struct iommu_table_entry' has. The entries
45 * that are put in the .iommu_table section are not put in any order
46 * hence during boot-time we will have to resort them based on
47 * dependency. */
48
49
50#define __IOMMU_INIT(_detect, _depend, _early_init, _late_init, _finish)\
51 static const struct iommu_table_entry const \
52 __iommu_entry_##_detect __used \
53 __attribute__ ((unused, __section__(".iommu_table"), \
54 aligned((sizeof(void *))))) \
55 = {_detect, _depend, _early_init, _late_init, \
56 _finish ? IOMMU_FINISH_IF_DETECTED : 0}
57/*
58 * The simplest IOMMU definition. Provide the detection routine
59 * and it will be run after the SWIOTLB and the other IOMMUs
60 * that utilize this macro. If the IOMMU is detected (ie, the
61 * detect routine returns a positive value), the other IOMMUs
Konrad Rzeszutek Wilk6e963662010-10-08 14:53:48 -040062 * are also checked. You can use IOMMU_INIT_POST_FINISH if you prefer
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -040063 * to stop detecting the other IOMMUs after yours has been detected.
64 */
65#define IOMMU_INIT_POST(_detect) \
Konrad Rzeszutek Wilkefa631c2010-08-26 13:57:59 -040066 __IOMMU_INIT(_detect, pci_swiotlb_detect_4gb, 0, 0, 0)
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -040067
68#define IOMMU_INIT_POST_FINISH(detect) \
Konrad Rzeszutek Wilkefa631c2010-08-26 13:57:59 -040069 __IOMMU_INIT(_detect, pci_swiotlb_detect_4gb, 0, 0, 1)
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -040070
71/*
72 * A more sophisticated version of IOMMU_INIT. This variant requires:
73 * a). A detection routine function.
74 * b). The name of the detection routine we depend on to get called
75 * before us.
76 * c). The init routine which gets called if the detection routine
77 * returns a positive value from the pci_iommu_alloc. This means
78 * no presence of a memory allocator.
79 * d). Similar to the 'init', except that this gets called from pci_iommu_init
80 * where we do have a memory allocator.
81 *
Konrad Rzeszutek Wilk6e963662010-10-08 14:53:48 -040082 * The standard vs the _FINISH differs in that the _FINISH variant will
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -040083 * continue detecting other IOMMUs in the call list after the
Konrad Rzeszutek Wilk6e963662010-10-08 14:53:48 -040084 * the detection routine returns a positive number. The _FINISH will
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -040085 * stop the execution chain. Both will still call the 'init' and
86 * 'late_init' functions if they are set.
87 */
88#define IOMMU_INIT_FINISH(_detect, _depend, _init, _late_init) \
89 __IOMMU_INIT(_detect, _depend, _init, _late_init, 1)
90
91#define IOMMU_INIT(_detect, _depend, _init, _late_init) \
92 __IOMMU_INIT(_detect, _depend, _init, _late_init, 0)
93
Konrad Rzeszutek Wilk5bef80a2010-08-26 13:57:58 -040094void sort_iommu_table(struct iommu_table_entry *start,
95 struct iommu_table_entry *finish);
96
97void check_iommu_entries(struct iommu_table_entry *start,
98 struct iommu_table_entry *finish);
99
Konrad Rzeszutek Wilk0444ad92010-08-26 13:57:56 -0400100#endif /* _ASM_X86_IOMMU_TABLE_H */