1 /*
2  * Copyright (c) 2006, Intel Corporation.
3  *
4  * This program is free software; you can redistribute it and/or modify it
5  * under the terms and conditions of the GNU General Public License,
6  * version 2, as published by the Free Software Foundation.
7  *
8  * This program is distributed in the hope it will be useful, but WITHOUT
9  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
11  * more details.
12  *
13  * You should have received a copy of the GNU General Public License along with
14  * this program; If not, see <http://www.gnu.org/licenses/>.
15  *
16  * Copyright (C) Ashok Raj <ashok.raj@intel.com>
17  * Copyright (C) Shaohua Li <shaohua.li@intel.com>
18  */
19 
20 #ifndef _DMAR_H_
21 #define _DMAR_H_
22 
23 #include <xen/list.h>
24 #include <xen/iommu.h>
25 #include <xen/kexec.h>
26 
27 /* This one is for interrupt remapping */
28 struct acpi_ioapic_unit {
29     struct list_head list;
30     int apic_id;
31     union {
32         u16 info;
33         struct {
34             u16 func: 3,
35                 dev:  5,
36                 bus:  8;
37         }bdf;
38     }ioapic;
39 };
40 
41 struct acpi_hpet_unit {
42     struct list_head list;
43     unsigned int id;
44     union {
45         u16 bdf;
46         struct {
47             u16 func: 3,
48                 dev:  5,
49                 bus:  8;
50         };
51     };
52 };
53 
54 struct dmar_scope {
55     DECLARE_BITMAP(buses, 256);         /* buses owned by this unit */
56     u16    *devices;                    /* devices owned by this unit */
57     int    devices_cnt;
58 };
59 
60 struct acpi_drhd_unit {
61     struct dmar_scope scope;
62     struct list_head list;
63     u64    address;                     /* register base address of the unit */
64     u16    segment;
65     u8     include_all:1;
66     struct vtd_iommu *iommu;
67     struct list_head ioapic_list;
68     struct list_head hpet_list;
69 };
70 
71 struct acpi_rmrr_unit {
72     struct dmar_scope scope;
73     struct list_head list;
74     u64    base_address;
75     u64    end_address;
76     u16    segment;
77     u8     allow_all:1;
78 };
79 
80 struct acpi_atsr_unit {
81     struct dmar_scope scope;
82     struct list_head list;
83     u16    segment;
84     u8     all_ports:1;
85 };
86 
87 struct acpi_rhsa_unit {
88     struct list_head list;
89     u64    address;
90     u32    proximity_domain;
91 };
92 
93 #define for_each_drhd_unit(drhd) \
94     list_for_each_entry(drhd, &acpi_drhd_units, list)
95 
96 #define for_each_rmrr_device(rmrr, bdf, idx)            \
97     list_for_each_entry(rmrr, &acpi_rmrr_units, list)   \
98         /* assume there never is a bdf == 0 */          \
99         for (idx = 0; (bdf = rmrr->scope.devices[idx]) && \
100                  idx < rmrr->scope.devices_cnt; idx++)
101 
102 struct acpi_drhd_unit *acpi_find_matched_drhd_unit(const struct pci_dev *);
103 struct acpi_atsr_unit *acpi_find_matched_atsr_unit(const struct pci_dev *);
104 
105 #define DMAR_TYPE 1
106 #define RMRR_TYPE 2
107 #define ATSR_TYPE 3
108 
109 #define DMAR_OPERATION_TIMEOUT MILLISECS(1000)
110 
111 #define IOMMU_WAIT_OP(iommu, offset, op, cond, sts) \
112 do {                                                \
113     s_time_t start_time = NOW();                    \
114     while (1) {                                     \
115         sts = op(iommu->reg, offset);               \
116         if ( cond )                                 \
117             break;                                  \
118         if ( NOW() > start_time + DMAR_OPERATION_TIMEOUT ) {    \
119             if ( !kexecing )                                    \
120             {                                                   \
121                 dump_execution_state();                         \
122                 panic("DMAR hardware malfunction\n");           \
123             }                                                   \
124             break;                                              \
125         }                                                       \
126         cpu_relax();                                            \
127     }                                                           \
128 } while (0)
129 
130 int vtd_hw_check(void);
131 void disable_pmr(struct vtd_iommu *iommu);
132 int is_igd_drhd(struct acpi_drhd_unit *drhd);
133 
134 #endif /* _DMAR_H_ */
135