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

30 31
struct acpi_dmar_header;

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

36
struct intel_iommu;
37

38
#ifdef CONFIG_DMAR_TABLE
39
extern struct acpi_table_header *dmar_tbl;
40 41 42 43 44 45
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	*/
46
	u16	segment;		/* PCI domain		*/
47 48 49 50 51
	u8	ignored:1; 		/* ignore drhd		*/
	u8	include_all:1;
	struct intel_iommu *iommu;
};

52
extern struct rw_semaphore dmar_global_lock;
53 54 55 56 57
extern struct list_head dmar_drhd_units;

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

58 59 60 61
#define for_each_active_drhd_unit(drhd)					\
	list_for_each_entry(drhd, &dmar_drhd_units, list)		\
		if (drhd->ignored) {} else

62 63 64 65 66 67 68 69
#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 

70 71 72 73 74 75
#define	for_each_dev_scope(a, c, p, d)	\
	for ((p) = 0; ((d) = (p) < (c) ? (a)[(p)] : NULL, (p) < (c)); (p)++)

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

76 77
extern int dmar_table_init(void);
extern int dmar_dev_scope_init(void);
78 79
extern int dmar_parse_dev_scope(void *start, void *end, int *cnt,
				struct pci_dev ***devices, u16 segment);
80
extern void *dmar_alloc_dev_scope(void *start, void *end, int *cnt);
81
extern void dmar_free_dev_scope(struct pci_dev ***devices, int *cnt);
82 83

/* Intel IOMMU detection */
84
extern int detect_intel_iommu(void);
85
extern int enable_drhd_fault_handling(void);
86
#else
87
static inline int detect_intel_iommu(void)
88
{
89
	return -ENODEV;
90 91 92 93 94 95
}

static inline int dmar_table_init(void)
{
	return -ENODEV;
}
96 97 98 99
static inline int enable_drhd_fault_handling(void)
{
	return -1;
}
100
#endif /* !CONFIG_DMAR_TABLE */
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129

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

131 132 133 134 135
enum {
	IRQ_REMAP_XAPIC_MODE,
	IRQ_REMAP_X2APIC_MODE,
};

136 137 138
/* Can't use the common MSI interrupt functions
 * since DMAR is not a pci device
 */
139 140 141
struct irq_data;
extern void dmar_msi_unmask(struct irq_data *data);
extern void dmar_msi_mask(struct irq_data *data);
142 143 144
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);
145
extern irqreturn_t dmar_fault(int irq, void *dev_id);
146 147
extern int arch_setup_dmar_msi(unsigned int irq);

148
#ifdef CONFIG_INTEL_IOMMU
149
extern int iommu_detected, no_iommu;
150
extern int dmar_parse_rmrr_atsr_dev(void);
151 152
extern int dmar_parse_one_rmrr(struct acpi_dmar_header *header);
extern int dmar_parse_one_atsr(struct acpi_dmar_header *header);
153
extern int intel_iommu_init(void);
154
#else /* !CONFIG_INTEL_IOMMU: */
155
static inline int intel_iommu_init(void) { return -ENODEV; }
156 157 158 159 160 161 162 163 164 165 166 167
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;
}
168
#endif /* CONFIG_INTEL_IOMMU */
169

170
#endif /* __DMAR_H__ */