blob: 7cdc89b2483c9200e8cdda24d93062f14b7364b9 [file] [log] [blame]
Fenghua Yu62fdd762008-10-17 12:14:13 -07001/*
2 * Dynamic DMA mapping support.
3 */
4
5#include <linux/types.h>
6#include <linux/mm.h>
7#include <linux/string.h>
8#include <linux/pci.h>
9#include <linux/module.h>
10#include <linux/dmar.h>
11#include <asm/iommu.h>
12#include <asm/machvec.h>
13#include <linux/dma-mapping.h>
14
Fenghua Yu62fdd762008-10-17 12:14:13 -070015
Suresh Siddhad3f13812011-08-23 17:05:25 -070016#ifdef CONFIG_INTEL_IOMMU
Fenghua Yu62fdd762008-10-17 12:14:13 -070017
18#include <linux/kernel.h>
Fenghua Yu62fdd762008-10-17 12:14:13 -070019
20#include <asm/page.h>
Fenghua Yu62fdd762008-10-17 12:14:13 -070021
22dma_addr_t bad_dma_address __read_mostly;
23EXPORT_SYMBOL(bad_dma_address);
24
25static int iommu_sac_force __read_mostly;
26
27int no_iommu __read_mostly;
28#ifdef CONFIG_IOMMU_DEBUG
29int force_iommu __read_mostly = 1;
30#else
31int force_iommu __read_mostly;
32#endif
33
Fenghua Yuaed5d5f2009-04-30 17:57:11 -070034int iommu_pass_through;
Alex Williamsonbcb71ab2011-10-21 15:56:24 -040035int iommu_group_mf;
Fenghua Yuaed5d5f2009-04-30 17:57:11 -070036
Fenghua Yu62fdd762008-10-17 12:14:13 -070037/* Dummy device used for NULL arguments (normally ISA). Better would
38 be probably a smaller DMA mask, but this is bug-to-bug compatible
39 to i386. */
40struct device fallback_dev = {
Kay Sievers48ef2bb2009-01-06 10:44:40 -080041 .init_name = "fallback device",
Yang Hongyang284901a2009-04-06 19:01:15 -070042 .coherent_dma_mask = DMA_BIT_MASK(32),
Fenghua Yu62fdd762008-10-17 12:14:13 -070043 .dma_mask = &fallback_dev.coherent_dma_mask,
44};
45
FUJITA Tomonori160c1d82009-01-05 23:59:02 +090046extern struct dma_map_ops intel_dma_ops;
Fenghua Yu62fdd762008-10-17 12:14:13 -070047
48static int __init pci_iommu_init(void)
49{
50 if (iommu_detected)
51 intel_iommu_init();
52
53 return 0;
54}
55
56/* Must execute after PCI subsystem */
57fs_initcall(pci_iommu_init);
58
59void pci_iommu_shutdown(void)
60{
61 return;
62}
63
64void __init
65iommu_dma_init(void)
66{
67 return;
68}
69
Fenghua Yu62fdd762008-10-17 12:14:13 -070070int iommu_dma_supported(struct device *dev, u64 mask)
71{
Fenghua Yu62fdd762008-10-17 12:14:13 -070072 /* Copied from i386. Doesn't make much sense, because it will
73 only work for pci_alloc_coherent.
74 The caller just has to use GFP_DMA in this case. */
Yang Hongyang2f4f27d2009-04-06 19:01:18 -070075 if (mask < DMA_BIT_MASK(24))
Fenghua Yu62fdd762008-10-17 12:14:13 -070076 return 0;
77
78 /* Tell the device to use SAC when IOMMU force is on. This
79 allows the driver to use cheaper accesses in some cases.
80
81 Problem with this is that if we overflow the IOMMU area and
82 return DAC as fallback address the device may not handle it
83 correctly.
84
85 As a special case some controllers have a 39bit address
86 mode that is as efficient as 32bit (aic79xx). Don't force
87 SAC for these. Assume all masks <= 40 bits are of this
88 type. Normally this doesn't make any difference, but gives
89 more gentle handling of IOMMU overflow. */
Yang Hongyang50cf1562009-04-06 19:01:14 -070090 if (iommu_sac_force && (mask >= DMA_BIT_MASK(40))) {
Matthew Wilcoxe088a4a2009-05-22 13:49:49 -070091 dev_info(dev, "Force SAC with mask %llx\n", mask);
Fenghua Yu62fdd762008-10-17 12:14:13 -070092 return 0;
93 }
94
95 return 1;
96}
97EXPORT_SYMBOL(iommu_dma_supported);
98
FUJITA Tomonori160c1d82009-01-05 23:59:02 +090099void __init pci_iommu_alloc(void)
100{
101 dma_ops = &intel_dma_ops;
102
103 dma_ops->sync_single_for_cpu = machvec_dma_sync_single;
104 dma_ops->sync_sg_for_cpu = machvec_dma_sync_sg;
105 dma_ops->sync_single_for_device = machvec_dma_sync_single;
106 dma_ops->sync_sg_for_device = machvec_dma_sync_sg;
107 dma_ops->dma_supported = iommu_dma_supported;
FUJITA Tomonori160c1d82009-01-05 23:59:02 +0900108
109 /*
110 * The order of these functions is important for
111 * fall-back/fail-over reasons
112 */
113 detect_intel_iommu();
114
115#ifdef CONFIG_SWIOTLB
116 pci_swiotlb_init();
117#endif
118}
119
Fenghua Yu62fdd762008-10-17 12:14:13 -0700120#endif