dmar.h 6.7 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

29 30
struct acpi_dmar_header;

31 32 33 34
/* DMAR Flags */
#define DMAR_INTR_REMAP		0x1
#define DMAR_X2APIC_OPT_OUT	0x2

35
struct intel_iommu;
36
#ifdef CONFIG_DMAR_TABLE
37
extern struct acpi_table_header *dmar_tbl;
38 39 40 41 42 43
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*/
	struct	pci_dev **devices; 	/* target device array	*/
	int	devices_cnt;		/* target device count	*/
44
	u16	segment;		/* PCI domain		*/
45 46 47 48 49 50 51 52 53 54
	u8	ignored:1; 		/* ignore drhd		*/
	u8	include_all:1;
	struct intel_iommu *iommu;
};

extern struct list_head dmar_drhd_units;

#define for_each_drhd_unit(drhd) \
	list_for_each_entry(drhd, &dmar_drhd_units, list)

55 56 57 58 59 60 61 62
#define for_each_active_iommu(i, drhd)					\
	list_for_each_entry(drhd, &dmar_drhd_units, list)		\
		if (i=drhd->iommu, drhd->ignored) {} else

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

63 64 65 66
extern int dmar_table_init(void);
extern int dmar_dev_scope_init(void);

/* Intel IOMMU detection */
67
extern int detect_intel_iommu(void);
68
extern int enable_drhd_fault_handling(void);
69 70 71 72

extern int parse_ioapics_under_ir(void);
extern int alloc_iommu(struct dmar_drhd_unit *);
#else
73
static inline int detect_intel_iommu(void)
74
{
75
	return -ENODEV;
76 77 78 79 80 81
}

static inline int dmar_table_init(void)
{
	return -ENODEV;
}
82 83 84 85
static inline int enable_drhd_fault_handling(void)
{
	return -1;
}
86
#endif /* !CONFIG_DMAR_TABLE */
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115

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;
	};
};
116

117
#ifdef CONFIG_IRQ_REMAP
118 119
extern void disable_intr_remapping(void);
extern int reenable_intr_remapping(int);
120

121 122 123 124 125 126 127 128
extern int get_irte(int irq, struct irte *entry);
extern int modify_irte(int irq, struct irte *irte_modified);
extern int alloc_irte(struct intel_iommu *iommu, int irq, u16 count);
extern int set_irte_irq(int irq, struct intel_iommu *iommu, u16 index,
   			u16 sub_handle);
extern int map_irq_to_irte_handle(int irq, u16 *sub_handle);
extern int free_irte(int irq);

129
extern struct intel_iommu *map_dev_to_ir(struct pci_dev *dev);
130
extern struct intel_iommu *map_ioapic_to_ir(int apic);
131
extern struct intel_iommu *map_hpet_to_ir(u8 id);
132
extern int set_ioapic_sid(struct irte *irte, int apic);
133
extern int set_hpet_sid(struct irte *irte, u8 id);
134
extern int set_msi_sid(struct irte *irte, struct pci_dev *dev);
135
#else
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 162 163 164
static inline int alloc_irte(struct intel_iommu *iommu, int irq, u16 count)
{
	return -1;
}
static inline int modify_irte(int irq, struct irte *irte_modified)
{
	return -1;
}
static inline int free_irte(int irq)
{
	return -1;
}
static inline int map_irq_to_irte_handle(int irq, u16 *sub_handle)
{
	return -1;
}
static inline int set_irte_irq(int irq, struct intel_iommu *iommu, u16 index,
			       u16 sub_handle)
{
	return -1;
}
static inline struct intel_iommu *map_dev_to_ir(struct pci_dev *dev)
{
	return NULL;
}
static inline struct intel_iommu *map_ioapic_to_ir(int apic)
{
	return NULL;
}
165 166 167 168
static inline struct intel_iommu *map_hpet_to_ir(unsigned int hpet_id)
{
	return NULL;
}
169 170 171 172
static inline int set_ioapic_sid(struct irte *irte, int apic)
{
	return 0;
}
173 174 175 176
static inline int set_hpet_sid(struct irte *irte, u8 id)
{
	return -1;
}
177 178 179 180 181
static inline int set_msi_sid(struct irte *irte, struct pci_dev *dev)
{
	return 0;
}

182
#define intr_remapping_enabled		(0)
183

184
static inline int enable_intr_remapping(void)
185 186 187 188 189 190 191 192 193 194 195 196
{
	return -1;
}

static inline void disable_intr_remapping(void)
{
}

static inline int reenable_intr_remapping(int eim)
{
	return 0;
}
197 198
#endif

199 200 201 202 203
enum {
	IRQ_REMAP_XAPIC_MODE,
	IRQ_REMAP_X2APIC_MODE,
};

204 205 206
/* Can't use the common MSI interrupt functions
 * since DMAR is not a pci device
 */
207 208 209
struct irq_data;
extern void dmar_msi_unmask(struct irq_data *data);
extern void dmar_msi_mask(struct irq_data *data);
210 211 212
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);
213
extern irqreturn_t dmar_fault(int irq, void *dev_id);
214 215
extern int arch_setup_dmar_msi(unsigned int irq);

216
#ifdef CONFIG_INTEL_IOMMU
217
extern int iommu_detected, no_iommu;
218 219 220
extern struct list_head dmar_rmrr_units;
struct dmar_rmrr_unit {
	struct list_head list;		/* list of rmrr units	*/
221
	struct acpi_dmar_header *hdr;	/* ACPI header		*/
222 223 224 225 226 227
	u64	base_address;		/* reserved base address*/
	u64	end_address;		/* reserved end address */
	struct pci_dev **devices;	/* target devices */
	int	devices_cnt;		/* target device count */
};

228 229
#define for_each_rmrr_units(rmrr) \
	list_for_each_entry(rmrr, &dmar_rmrr_units, list)
230 231 232 233 234 235 236 237 238

struct dmar_atsr_unit {
	struct list_head list;		/* list of ATSR units */
	struct acpi_dmar_header *hdr;	/* ACPI header */
	struct pci_dev **devices;	/* target devices */
	int devices_cnt;		/* target device count */
	u8 include_all:1;		/* include all ports */
};

239 240 241 242 243
int dmar_parse_rmrr_atsr_dev(void);
extern int dmar_parse_one_rmrr(struct acpi_dmar_header *header);
extern int dmar_parse_one_atsr(struct acpi_dmar_header *header);
extern int dmar_parse_dev_scope(void *start, void *end, int *cnt,
				struct pci_dev ***devices, u16 segment);
244
extern int intel_iommu_init(void);
245
#else /* !CONFIG_INTEL_IOMMU: */
246
static inline int intel_iommu_init(void) { return -ENODEV; }
247 248 249 250 251 252 253 254 255 256 257 258
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;
}
static inline int dmar_parse_rmrr_atsr_dev(void)
{
	return 0;
}
259
#endif /* CONFIG_INTEL_IOMMU */
260

261
#endif /* __DMAR_H__ */