dmar.h 5.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
/*
 * Copyright (c) 2006, Intel Corporation.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along with
 * this program; if not, write to the Free Software Foundation, Inc., 59 Temple
 * Place - Suite 330, Boston, MA 02111-1307 USA.
 *
 * Copyright (C) Ashok Raj <ashok.raj@intel.com>
 * Copyright (C) Shaohua Li <shaohua.li@intel.com>
 */

#ifndef __DMAR_H__
#define __DMAR_H__

#include <linux/acpi.h>
#include <linux/types.h>
26
#include <linux/msi.h>
27
#include <linux/irqreturn.h>
28
#include <linux/rwsem.h>
29
#include <linux/rcupdate.h>
30

31 32
struct acpi_dmar_header;

33 34 35 36
/* DMAR Flags */
#define DMAR_INTR_REMAP		0x1
#define DMAR_X2APIC_OPT_OUT	0x2

37
struct intel_iommu;
38

39 40 41 42 43 44
struct dmar_dev_scope {
	struct device __rcu *dev;
	u8 bus;
	u8 devfn;
};

45
#ifdef CONFIG_DMAR_TABLE
46
extern struct acpi_table_header *dmar_tbl;
47 48 49 50
struct dmar_drhd_unit {
	struct list_head list;		/* list of drhd units	*/
	struct  acpi_dmar_header *hdr;	/* ACPI header		*/
	u64	reg_base_addr;		/* register base address*/
51
	struct	dmar_dev_scope *devices;/* target device array	*/
52
	int	devices_cnt;		/* target device count	*/
53
	u16	segment;		/* PCI domain		*/
54 55 56 57 58
	u8	ignored:1; 		/* ignore drhd		*/
	u8	include_all:1;
	struct intel_iommu *iommu;
};

59 60 61 62 63 64 65 66 67
struct dmar_pci_notify_info {
	struct pci_dev			*dev;
	unsigned long			event;
	int				bus;
	u16				seg;
	u16				level;
	struct acpi_dmar_pci_path	path[];
}  __attribute__((packed));

68
extern struct rw_semaphore dmar_global_lock;
69 70 71
extern struct list_head dmar_drhd_units;

#define for_each_drhd_unit(drhd) \
72
	list_for_each_entry_rcu(drhd, &dmar_drhd_units, list)
73

74
#define for_each_active_drhd_unit(drhd)					\
75
	list_for_each_entry_rcu(drhd, &dmar_drhd_units, list)		\
76 77
		if (drhd->ignored) {} else

78
#define for_each_active_iommu(i, drhd)					\
79
	list_for_each_entry_rcu(drhd, &dmar_drhd_units, list)		\
80 81 82
		if (i=drhd->iommu, drhd->ignored) {} else

#define for_each_iommu(i, drhd)						\
83
	list_for_each_entry_rcu(drhd, &dmar_drhd_units, list)		\
84 85
		if (i=drhd->iommu, 0) {} else 

86 87 88 89 90 91 92 93
static inline bool dmar_rcu_check(void)
{
	return rwsem_is_locked(&dmar_global_lock) ||
	       system_state == SYSTEM_BOOTING;
}

#define	dmar_rcu_dereference(p)	rcu_dereference_check((p), dmar_rcu_check())

94
#define	for_each_dev_scope(a, c, p, d)	\
95
	for ((p) = 0; ((d) = (p) < (c) ? dmar_rcu_dereference((a)[(p)].dev) : \
96
			NULL, (p) < (c)); (p)++)
97 98 99 100

#define	for_each_active_dev_scope(a, c, p, d)	\
	for_each_dev_scope((a), (c), (p), (d))	if (!(d)) { continue; } else

101 102
extern int dmar_table_init(void);
extern int dmar_dev_scope_init(void);
103
extern int dmar_parse_dev_scope(void *start, void *end, int *cnt,
104
				struct dmar_dev_scope **devices, u16 segment);
105
extern void *dmar_alloc_dev_scope(void *start, void *end, int *cnt);
106
extern void dmar_free_dev_scope(struct dmar_dev_scope **devices, int *cnt);
107 108
extern int dmar_insert_dev_scope(struct dmar_pci_notify_info *info,
				 void *start, void*end, u16 segment,
109
				 struct dmar_dev_scope *devices,
110 111
				 int devices_cnt);
extern int dmar_remove_dev_scope(struct dmar_pci_notify_info *info,
112
				 u16 segment, struct dmar_dev_scope *devices,
113
				 int count);
114
/* Intel IOMMU detection */
115
extern int detect_intel_iommu(void);
116
extern int enable_drhd_fault_handling(void);
117
#else
118
struct dmar_pci_notify_info;
119
static inline int detect_intel_iommu(void)
120
{
121
	return -ENODEV;
122 123 124 125 126 127
}

static inline int dmar_table_init(void)
{
	return -ENODEV;
}
128 129 130 131
static inline int enable_drhd_fault_handling(void)
{
	return -1;
}
132
#endif /* !CONFIG_DMAR_TABLE */
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161

struct irte {
	union {
		struct {
			__u64	present 	: 1,
				fpd		: 1,
				dst_mode	: 1,
				redir_hint	: 1,
				trigger_mode	: 1,
				dlvry_mode	: 3,
				avail		: 4,
				__reserved_1	: 4,
				vector		: 8,
				__reserved_2	: 8,
				dest_id		: 32;
		};
		__u64 low;
	};

	union {
		struct {
			__u64	sid		: 16,
				sq		: 2,
				svt		: 2,
				__reserved_3	: 44;
		};
		__u64 high;
	};
};
162

163 164 165 166 167
enum {
	IRQ_REMAP_XAPIC_MODE,
	IRQ_REMAP_X2APIC_MODE,
};

168 169 170
/* Can't use the common MSI interrupt functions
 * since DMAR is not a pci device
 */
171 172 173
struct irq_data;
extern void dmar_msi_unmask(struct irq_data *data);
extern void dmar_msi_mask(struct irq_data *data);
174 175 176
extern void dmar_msi_read(int irq, struct msi_msg *msg);
extern void dmar_msi_write(int irq, struct msi_msg *msg);
extern int dmar_set_interrupt(struct intel_iommu *iommu);
177
extern irqreturn_t dmar_fault(int irq, void *dev_id);
178 179
extern int arch_setup_dmar_msi(unsigned int irq);

180
#ifdef CONFIG_INTEL_IOMMU
181
extern int iommu_detected, no_iommu;
182 183
extern int dmar_parse_one_rmrr(struct acpi_dmar_header *header);
extern int dmar_parse_one_atsr(struct acpi_dmar_header *header);
184
extern int dmar_iommu_notify_scope_dev(struct dmar_pci_notify_info *info);
185
extern int intel_iommu_init(void);
186
#else /* !CONFIG_INTEL_IOMMU: */
187
static inline int intel_iommu_init(void) { return -ENODEV; }
188 189 190 191 192 193 194 195
static inline int dmar_parse_one_rmrr(struct acpi_dmar_header *header)
{
	return 0;
}
static inline int dmar_parse_one_atsr(struct acpi_dmar_header *header)
{
	return 0;
}
196 197 198 199
static inline int dmar_iommu_notify_scope_dev(struct dmar_pci_notify_info *info)
{
	return 0;
}
200
#endif /* CONFIG_INTEL_IOMMU */
201

202
#endif /* __DMAR_H__ */