Commit 480125ba authored by Konrad Rzeszutek Wilk's avatar Konrad Rzeszutek Wilk Committed by H. Peter Anvin

x86, iommu: Make all IOMMU's detection routines return a value.

We return 1 if the IOMMU has been detected. Zero or an error number
if we failed to find it. This is in preperation of using the IOMMU_INIT
so that we can detect whether an IOMMU is present. I have not
tested this for regression on Calgary, nor on AMD Vi chipsets as
I don't have that hardware.

CC: Muli Ben-Yehuda <muli@il.ibm.com>
CC: "Jon D. Mason" <jdmason@kudzu.us>
CC: "Darrick J. Wong" <djwong@us.ibm.com>
CC: Jesse Barnes <jbarnes@virtuousgeek.org>
CC: David Woodhouse <David.Woodhouse@intel.com>
CC: Chris Wright <chrisw@sous-sol.org>
CC: Yinghai Lu <yinghai@kernel.org>
CC: Joerg Roedel <joerg.roedel@amd.com>
CC: H. Peter Anvin <hpa@zytor.com>
CC: Fujita Tomonori <fujita.tomonori@lab.ntt.co.jp>
Signed-off-by: default avatarKonrad Rzeszutek Wilk <konrad.wilk@oracle.com>
LKML-Reference: <1282845485-8991-3-git-send-email-konrad.wilk@oracle.com>
Signed-off-by: default avatarH. Peter Anvin <hpa@linux.intel.com>
parent 0444ad93
...@@ -24,11 +24,11 @@ ...@@ -24,11 +24,11 @@
#ifdef CONFIG_AMD_IOMMU #ifdef CONFIG_AMD_IOMMU
extern void amd_iommu_detect(void); extern int amd_iommu_detect(void);
#else #else
static inline void amd_iommu_detect(void) { } static inline int amd_iommu_detect(void) { return -ENODEV; }
#endif #endif
......
...@@ -62,9 +62,9 @@ struct cal_chipset_ops { ...@@ -62,9 +62,9 @@ struct cal_chipset_ops {
extern int use_calgary; extern int use_calgary;
#ifdef CONFIG_CALGARY_IOMMU #ifdef CONFIG_CALGARY_IOMMU
extern void detect_calgary(void); extern int detect_calgary(void);
#else #else
static inline void detect_calgary(void) { return; } static inline int detect_calgary(void) { return -ENODEV; }
#endif #endif
#endif /* _ASM_X86_CALGARY_H */ #endif /* _ASM_X86_CALGARY_H */
...@@ -37,7 +37,7 @@ extern int gart_iommu_aperture_disabled; ...@@ -37,7 +37,7 @@ extern int gart_iommu_aperture_disabled;
extern void early_gart_iommu_check(void); extern void early_gart_iommu_check(void);
extern int gart_iommu_init(void); extern int gart_iommu_init(void);
extern void __init gart_parse_options(char *); extern void __init gart_parse_options(char *);
extern void gart_iommu_hole_init(void); extern int gart_iommu_hole_init(void);
#else #else
#define gart_iommu_aperture 0 #define gart_iommu_aperture 0
...@@ -50,8 +50,9 @@ static inline void early_gart_iommu_check(void) ...@@ -50,8 +50,9 @@ static inline void early_gart_iommu_check(void)
static inline void gart_parse_options(char *options) static inline void gart_parse_options(char *options)
{ {
} }
static inline void gart_iommu_hole_init(void) static inline int gart_iommu_hole_init(void)
{ {
return -ENODEV;
} }
#endif #endif
......
...@@ -1382,13 +1382,13 @@ static int __init early_amd_iommu_detect(struct acpi_table_header *table) ...@@ -1382,13 +1382,13 @@ static int __init early_amd_iommu_detect(struct acpi_table_header *table)
return 0; return 0;
} }
void __init amd_iommu_detect(void) int __init amd_iommu_detect(void)
{ {
if (no_iommu || (iommu_detected && !gart_iommu_aperture)) if (no_iommu || (iommu_detected && !gart_iommu_aperture))
return; return -ENODEV;
if (amd_iommu_disabled) if (amd_iommu_disabled)
return; return -ENODEV;
if (acpi_table_parse("IVRS", early_amd_iommu_detect) == 0) { if (acpi_table_parse("IVRS", early_amd_iommu_detect) == 0) {
iommu_detected = 1; iommu_detected = 1;
...@@ -1397,7 +1397,9 @@ void __init amd_iommu_detect(void) ...@@ -1397,7 +1397,9 @@ void __init amd_iommu_detect(void)
/* Make sure ACS will be enabled */ /* Make sure ACS will be enabled */
pci_request_acs(); pci_request_acs();
return 1;
} }
return -ENODEV;
} }
/**************************************************************************** /****************************************************************************
......
...@@ -371,7 +371,7 @@ void __init early_gart_iommu_check(void) ...@@ -371,7 +371,7 @@ void __init early_gart_iommu_check(void)
static int __initdata printed_gart_size_msg; static int __initdata printed_gart_size_msg;
void __init gart_iommu_hole_init(void) int __init gart_iommu_hole_init(void)
{ {
u32 agp_aper_base = 0, agp_aper_order = 0; u32 agp_aper_base = 0, agp_aper_order = 0;
u32 aper_size, aper_alloc = 0, aper_order = 0, last_aper_order = 0; u32 aper_size, aper_alloc = 0, aper_order = 0, last_aper_order = 0;
...@@ -381,7 +381,7 @@ void __init gart_iommu_hole_init(void) ...@@ -381,7 +381,7 @@ void __init gart_iommu_hole_init(void)
if (gart_iommu_aperture_disabled || !fix_aperture || if (gart_iommu_aperture_disabled || !fix_aperture ||
!early_pci_allowed()) !early_pci_allowed())
return; return -ENODEV;
printk(KERN_INFO "Checking aperture...\n"); printk(KERN_INFO "Checking aperture...\n");
...@@ -463,8 +463,9 @@ void __init gart_iommu_hole_init(void) ...@@ -463,8 +463,9 @@ void __init gart_iommu_hole_init(void)
unsigned long n = (32 * 1024 * 1024) << last_aper_order; unsigned long n = (32 * 1024 * 1024) << last_aper_order;
insert_aperture_resource((u32)last_aper_base, n); insert_aperture_resource((u32)last_aper_base, n);
return 1;
} }
return; return 0;
} }
if (!fallback_aper_force) { if (!fallback_aper_force) {
...@@ -500,7 +501,7 @@ void __init gart_iommu_hole_init(void) ...@@ -500,7 +501,7 @@ void __init gart_iommu_hole_init(void)
panic("Not enough memory for aperture"); panic("Not enough memory for aperture");
} }
} else { } else {
return; return 0;
} }
/* Fix up the north bridges */ /* Fix up the north bridges */
...@@ -524,4 +525,6 @@ void __init gart_iommu_hole_init(void) ...@@ -524,4 +525,6 @@ void __init gart_iommu_hole_init(void)
} }
set_up_gart_resume(aper_order, aper_alloc); set_up_gart_resume(aper_order, aper_alloc);
return 1;
} }
...@@ -1364,7 +1364,7 @@ static int __init calgary_iommu_init(void) ...@@ -1364,7 +1364,7 @@ static int __init calgary_iommu_init(void)
return 0; return 0;
} }
void __init detect_calgary(void) int __init detect_calgary(void)
{ {
int bus; int bus;
void *tbl; void *tbl;
...@@ -1378,13 +1378,13 @@ void __init detect_calgary(void) ...@@ -1378,13 +1378,13 @@ void __init detect_calgary(void)
* another HW IOMMU already, bail out. * another HW IOMMU already, bail out.
*/ */
if (no_iommu || iommu_detected) if (no_iommu || iommu_detected)
return; return -ENODEV;
if (!use_calgary) if (!use_calgary)
return; return -ENODEV;
if (!early_pci_allowed()) if (!early_pci_allowed())
return; return -ENODEV;
printk(KERN_DEBUG "Calgary: detecting Calgary via BIOS EBDA area\n"); printk(KERN_DEBUG "Calgary: detecting Calgary via BIOS EBDA area\n");
...@@ -1410,13 +1410,13 @@ void __init detect_calgary(void) ...@@ -1410,13 +1410,13 @@ void __init detect_calgary(void)
if (!rio_table_hdr) { if (!rio_table_hdr) {
printk(KERN_DEBUG "Calgary: Unable to locate Rio Grande table " printk(KERN_DEBUG "Calgary: Unable to locate Rio Grande table "
"in EBDA - bailing!\n"); "in EBDA - bailing!\n");
return; return -ENODEV;
} }
ret = build_detail_arrays(); ret = build_detail_arrays();
if (ret) { if (ret) {
printk(KERN_DEBUG "Calgary: build_detail_arrays ret %d\n", ret); printk(KERN_DEBUG "Calgary: build_detail_arrays ret %d\n", ret);
return; return -ENOMEM;
} }
specified_table_size = determine_tce_table_size((is_kdump_kernel() ? specified_table_size = determine_tce_table_size((is_kdump_kernel() ?
...@@ -1464,7 +1464,7 @@ void __init detect_calgary(void) ...@@ -1464,7 +1464,7 @@ void __init detect_calgary(void)
x86_init.iommu.iommu_init = calgary_iommu_init; x86_init.iommu.iommu_init = calgary_iommu_init;
} }
return; return calgary_found;
cleanup: cleanup:
for (--bus; bus >= 0; --bus) { for (--bus; bus >= 0; --bus) {
...@@ -1473,6 +1473,7 @@ void __init detect_calgary(void) ...@@ -1473,6 +1473,7 @@ void __init detect_calgary(void)
if (info->tce_space) if (info->tce_space)
free_tce_table(info->tce_space); free_tce_table(info->tce_space);
} }
return -ENOMEM;
} }
static int __init calgary_parse_options(char *p) static int __init calgary_parse_options(char *p)
......
...@@ -687,7 +687,7 @@ int __init check_zero_address(void) ...@@ -687,7 +687,7 @@ int __init check_zero_address(void)
return 0; return 0;
} }
void __init detect_intel_iommu(void) int __init detect_intel_iommu(void)
{ {
int ret; int ret;
...@@ -723,6 +723,8 @@ void __init detect_intel_iommu(void) ...@@ -723,6 +723,8 @@ void __init detect_intel_iommu(void)
} }
early_acpi_os_unmap_memory(dmar_tbl, dmar_tbl_size); early_acpi_os_unmap_memory(dmar_tbl, dmar_tbl_size);
dmar_tbl = NULL; dmar_tbl = NULL;
return (ret ? 1 : -ENODEV);
} }
......
...@@ -57,15 +57,15 @@ extern int dmar_table_init(void); ...@@ -57,15 +57,15 @@ extern int dmar_table_init(void);
extern int dmar_dev_scope_init(void); extern int dmar_dev_scope_init(void);
/* Intel IOMMU detection */ /* Intel IOMMU detection */
extern void detect_intel_iommu(void); extern int detect_intel_iommu(void);
extern int enable_drhd_fault_handling(void); extern int enable_drhd_fault_handling(void);
extern int parse_ioapics_under_ir(void); extern int parse_ioapics_under_ir(void);
extern int alloc_iommu(struct dmar_drhd_unit *); extern int alloc_iommu(struct dmar_drhd_unit *);
#else #else
static inline void detect_intel_iommu(void) static inline int detect_intel_iommu(void)
{ {
return; return -ENODEV;
} }
static inline int dmar_table_init(void) static inline int dmar_table_init(void)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment