Blame view

lib/swiotlb.c 29.5 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
  /*
   * Dynamic DMA mapping support.
   *
563aaf064   Jan Beulich   [IA64] swiotlb cl...
4
   * This implementation is a fallback for platforms that do not support
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
5
6
7
8
9
10
11
12
13
   * I/O TLBs (aka DMA address translation hardware).
   * Copyright (C) 2000 Asit Mallick <Asit.K.Mallick@intel.com>
   * Copyright (C) 2000 Goutham Rao <goutham.rao@intel.com>
   * Copyright (C) 2000, 2003 Hewlett-Packard Co
   *	David Mosberger-Tang <davidm@hpl.hp.com>
   *
   * 03/05/07 davidm	Switch from PCI-DMA to generic device DMA API.
   * 00/12/13 davidm	Rename to swiotlb.c and add mark_clean() to avoid
   *			unnecessary i-cache flushing.
569c8bf5d   John W. Linville   [PATCH] swiotlb: ...
14
15
16
   * 04/07/.. ak		Better overflow handling. Assorted fixes.
   * 05/09/10 linville	Add support for syncing ranges, support syncing for
   *			DMA_BIDIRECTIONAL mappings, miscellaneous cleanup.
fb05a3792   Becky Bruce   swiotlb: add supp...
17
   * 08/12/11 beckyb	Add highmem support
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18
   */
f2a4f7622   Kees Cook   swiotlb: clean up...
19
  #define pr_fmt(fmt) "software IO TLB: " fmt
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
  #include <linux/cache.h>
17e5ad6c0   Tony Luck   [PATCH] Removed r...
21
  #include <linux/dma-mapping.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
  #include <linux/mm.h>
8bc3bcc93   Paul Gortmaker   lib: reduce the u...
23
  #include <linux/export.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
24
25
  #include <linux/spinlock.h>
  #include <linux/string.h>
0016fdee9   Ian Campbell   swiotlb: move som...
26
  #include <linux/swiotlb.h>
fb05a3792   Becky Bruce   swiotlb: add supp...
27
  #include <linux/pfn.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
28
29
  #include <linux/types.h>
  #include <linux/ctype.h>
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
30
  #include <linux/highmem.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
31
  #include <linux/gfp.h>
84be456f8   Christoph Hellwig   remove <asm/scatt...
32
  #include <linux/scatterlist.h>
c7753208a   Tom Lendacky   x86, swiotlb: Add...
33
  #include <linux/mem_encrypt.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
34
35
  
  #include <asm/io.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
36
37
38
39
  #include <asm/dma.h>
  
  #include <linux/init.h>
  #include <linux/bootmem.h>
a85225092   FUJITA Tomonori   swiotlb: use iomm...
40
  #include <linux/iommu-helper.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
41

ce5be5a16   Thierry Reding   tracing/events: F...
42
  #define CREATE_TRACE_POINTS
2b2b614dd   Zoltan Kiss   tracing/events: A...
43
  #include <trace/events/swiotlb.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
44
45
  #define OFFSET(val,align) ((unsigned long)	\
  	                   ( (val) & ( (align) - 1)))
0b9afede3   Alex Williamson   [IA64] more robus...
46
47
48
49
50
51
52
53
  #define SLABS_PER_PAGE (1 << (PAGE_SHIFT - IO_TLB_SHIFT))
  
  /*
   * Minimum IO TLB size to bother booting with.  Systems with mainly
   * 64bit capable cards will only lightly use the swiotlb.  If we can't
   * allocate a contiguous 1MB, we're probably in trouble anyway.
   */
  #define IO_TLB_MIN_SLABS ((1<<20) >> IO_TLB_SHIFT)
ae7871be1   Geert Uytterhoeven   swiotlb: Convert ...
54
  enum swiotlb_force swiotlb_force;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
55
56
  
  /*
bfc5501f6   Konrad Rzeszutek Wilk   swiotlb: Make int...
57
58
   * Used to do a quick range check in swiotlb_tbl_unmap_single and
   * swiotlb_tbl_sync_single_*, to see if the memory was in fact allocated by this
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
59
60
   * API.
   */
ff7204a74   Alexander Duyck   swiotlb: Make io_...
61
  static phys_addr_t io_tlb_start, io_tlb_end;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
62
63
  
  /*
b595076a1   Uwe Kleine-König   tree-wide: fix co...
64
   * The number of IO TLB blocks (in groups of 64) between io_tlb_start and
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
65
66
67
68
69
70
71
72
   * io_tlb_end.  This is command line adjustable via setup_io_tlb_npages.
   */
  static unsigned long io_tlb_nslabs;
  
  /*
   * When the IOMMU overflows we return a fallback buffer. This sets the size.
   */
  static unsigned long io_tlb_overflow = 32*1024;
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
73
  static phys_addr_t io_tlb_overflow_buffer;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
74
75
76
77
78
79
80
81
82
  
  /*
   * This is a free list describing the number of free entries available from
   * each index
   */
  static unsigned int *io_tlb_list;
  static unsigned int io_tlb_index;
  
  /*
7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
83
84
85
86
87
88
   * Max segment that we can provide which (if pages are contingous) will
   * not be bounced (unless SWIOTLB_FORCE is set).
   */
  unsigned int max_segment;
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
89
90
91
   * We need to save away the original address corresponding to a mapped entry
   * for the sync operations.
   */
8e0629c1d   Jan Beulich   swiotlb: don't as...
92
  #define INVALID_PHYS_ADDR (~(phys_addr_t)0)
bc40ac669   Becky Bruce   swiotlb: store ph...
93
  static phys_addr_t *io_tlb_orig_addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
94
95
96
97
98
  
  /*
   * Protect the above data structures in the map and unmap calls
   */
  static DEFINE_SPINLOCK(io_tlb_lock);
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
99
  static int late_alloc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
100
101
102
103
  static int __init
  setup_io_tlb_npages(char *str)
  {
  	if (isdigit(*str)) {
e8579e72c   Alex Williamson   [IA64, X86_64] fi...
104
  		io_tlb_nslabs = simple_strtoul(str, &str, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
105
106
107
108
109
  		/* avoid tail segment of size < IO_TLB_SEGSIZE */
  		io_tlb_nslabs = ALIGN(io_tlb_nslabs, IO_TLB_SEGSIZE);
  	}
  	if (*str == ',')
  		++str;
fff5d9922   Geert Uytterhoeven   swiotlb: Add swio...
110
  	if (!strcmp(str, "force")) {
ae7871be1   Geert Uytterhoeven   swiotlb: Convert ...
111
  		swiotlb_force = SWIOTLB_FORCE;
fff5d9922   Geert Uytterhoeven   swiotlb: Add swio...
112
113
114
115
  	} else if (!strcmp(str, "noforce")) {
  		swiotlb_force = SWIOTLB_NO_FORCE;
  		io_tlb_nslabs = 1;
  	}
b18485e7a   FUJITA Tomonori   swiotlb: Remove t...
116

c729de8fc   Yinghai Lu   x86, kdump: Set c...
117
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
118
  }
c729de8fc   Yinghai Lu   x86, kdump: Set c...
119
  early_param("swiotlb", setup_io_tlb_npages);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
120
  /* make io_tlb_overflow tunable too? */
f21ffe9f6   Konrad Rzeszutek Wilk   swiotlb: Expose s...
121
  unsigned long swiotlb_nr_tbl(void)
5f98ecdbc   FUJITA Tomonori   swiotlb: Export s...
122
123
124
  {
  	return io_tlb_nslabs;
  }
f21ffe9f6   Konrad Rzeszutek Wilk   swiotlb: Expose s...
125
  EXPORT_SYMBOL_GPL(swiotlb_nr_tbl);
c729de8fc   Yinghai Lu   x86, kdump: Set c...
126

7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
127
128
129
130
131
132
133
134
135
136
137
138
139
  unsigned int swiotlb_max_segment(void)
  {
  	return max_segment;
  }
  EXPORT_SYMBOL_GPL(swiotlb_max_segment);
  
  void swiotlb_set_max_segment(unsigned int val)
  {
  	if (swiotlb_force == SWIOTLB_FORCE)
  		max_segment = 1;
  	else
  		max_segment = rounddown(val, PAGE_SIZE);
  }
c729de8fc   Yinghai Lu   x86, kdump: Set c...
140
141
142
143
144
145
146
147
148
149
  /* default to 64MB */
  #define IO_TLB_DEFAULT_SIZE (64UL<<20)
  unsigned long swiotlb_size_or_default(void)
  {
  	unsigned long size;
  
  	size = io_tlb_nslabs << IO_TLB_SHIFT;
  
  	return size ? size : (IO_TLB_DEFAULT_SIZE);
  }
c7753208a   Tom Lendacky   x86, swiotlb: Add...
150
151
152
153
154
155
156
157
  void __weak swiotlb_set_mem_attributes(void *vaddr, unsigned long size) { }
  
  /* For swiotlb, clear memory encryption mask from dma addresses */
  static dma_addr_t swiotlb_phys_to_dma(struct device *hwdev,
  				      phys_addr_t address)
  {
  	return __sme_clr(phys_to_dma(hwdev, address));
  }
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
158
  /* Note that this doesn't work with highmem page */
70a7d3cc1   Jeremy Fitzhardinge   swiotlb: add hwde...
159
160
  static dma_addr_t swiotlb_virt_to_bus(struct device *hwdev,
  				      volatile void *address)
e08e1f7ad   Ian Campbell   swiotlb: allow ar...
161
  {
862d196b2   FUJITA Tomonori   swiotlb: use phys...
162
  	return phys_to_dma(hwdev, virt_to_phys(address));
e08e1f7ad   Ian Campbell   swiotlb: allow ar...
163
  }
ac2cbab21   Yinghai Lu   x86: Don't panic ...
164
  static bool no_iotlb_memory;
ad32e8cb8   FUJITA Tomonori   swiotlb: Defer sw...
165
  void swiotlb_print_info(void)
2e5b2b86b   Ian Campbell   swiotlb: consolid...
166
  {
ad32e8cb8   FUJITA Tomonori   swiotlb: Defer sw...
167
  	unsigned long bytes = io_tlb_nslabs << IO_TLB_SHIFT;
2e5b2b86b   Ian Campbell   swiotlb: consolid...
168

ac2cbab21   Yinghai Lu   x86: Don't panic ...
169
  	if (no_iotlb_memory) {
f2a4f7622   Kees Cook   swiotlb: clean up...
170
171
  		pr_warn("No low mem
  ");
ac2cbab21   Yinghai Lu   x86: Don't panic ...
172
173
  		return;
  	}
f2a4f7622   Kees Cook   swiotlb: clean up...
174
175
  	pr_info("mapped [mem %#010llx-%#010llx] (%luMB)
  ",
ff7204a74   Alexander Duyck   swiotlb: Make io_...
176
  	       (unsigned long long)io_tlb_start,
c40dba06e   Alexander Duyck   swiotlb: Make io_...
177
  	       (unsigned long long)io_tlb_end,
f2a4f7622   Kees Cook   swiotlb: clean up...
178
  	       bytes >> 20);
2e5b2b86b   Ian Campbell   swiotlb: consolid...
179
  }
c7753208a   Tom Lendacky   x86, swiotlb: Add...
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
  /*
   * Early SWIOTLB allocation may be too early to allow an architecture to
   * perform the desired operations.  This function allows the architecture to
   * call SWIOTLB when the operations are possible.  It needs to be called
   * before the SWIOTLB memory is used.
   */
  void __init swiotlb_update_mem_attributes(void)
  {
  	void *vaddr;
  	unsigned long bytes;
  
  	if (no_iotlb_memory || late_alloc)
  		return;
  
  	vaddr = phys_to_virt(io_tlb_start);
  	bytes = PAGE_ALIGN(io_tlb_nslabs << IO_TLB_SHIFT);
  	swiotlb_set_mem_attributes(vaddr, bytes);
  	memset(vaddr, 0, bytes);
  
  	vaddr = phys_to_virt(io_tlb_overflow_buffer);
  	bytes = PAGE_ALIGN(io_tlb_overflow);
  	swiotlb_set_mem_attributes(vaddr, bytes);
  	memset(vaddr, 0, bytes);
  }
ac2cbab21   Yinghai Lu   x86: Don't panic ...
204
  int __init swiotlb_init_with_tbl(char *tlb, unsigned long nslabs, int verbose)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
205
  {
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
206
  	void *v_overflow_buffer;
563aaf064   Jan Beulich   [IA64] swiotlb cl...
207
  	unsigned long i, bytes;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
208

abbceff7d   FUJITA Tomonori   swiotlb: add the ...
209
  	bytes = nslabs << IO_TLB_SHIFT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
210

abbceff7d   FUJITA Tomonori   swiotlb: add the ...
211
  	io_tlb_nslabs = nslabs;
ff7204a74   Alexander Duyck   swiotlb: Make io_...
212
213
  	io_tlb_start = __pa(tlb);
  	io_tlb_end = io_tlb_start + bytes;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
214
215
  
  	/*
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
216
217
  	 * Get the overflow emergency buffer
  	 */
ad6492b80   Yinghai Lu   memblock, nobootm...
218
  	v_overflow_buffer = memblock_virt_alloc_low_nopanic(
457ff1de2   Santosh Shilimkar   lib/swiotlb.c: us...
219
220
  						PAGE_ALIGN(io_tlb_overflow),
  						PAGE_SIZE);
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
221
  	if (!v_overflow_buffer)
ac2cbab21   Yinghai Lu   x86: Don't panic ...
222
  		return -ENOMEM;
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
223
224
225
226
  
  	io_tlb_overflow_buffer = __pa(v_overflow_buffer);
  
  	/*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
227
228
229
230
  	 * Allocate and initialize the free list array.  This array is used
  	 * to find contiguous free memory regions of size up to IO_TLB_SEGSIZE
  	 * between io_tlb_start and io_tlb_end.
  	 */
457ff1de2   Santosh Shilimkar   lib/swiotlb.c: us...
231
232
233
  	io_tlb_list = memblock_virt_alloc(
  				PAGE_ALIGN(io_tlb_nslabs * sizeof(int)),
  				PAGE_SIZE);
457ff1de2   Santosh Shilimkar   lib/swiotlb.c: us...
234
235
236
  	io_tlb_orig_addr = memblock_virt_alloc(
  				PAGE_ALIGN(io_tlb_nslabs * sizeof(phys_addr_t)),
  				PAGE_SIZE);
8e0629c1d   Jan Beulich   swiotlb: don't as...
237
238
239
240
241
  	for (i = 0; i < io_tlb_nslabs; i++) {
  		io_tlb_list[i] = IO_TLB_SEGSIZE - OFFSET(i, IO_TLB_SEGSIZE);
  		io_tlb_orig_addr[i] = INVALID_PHYS_ADDR;
  	}
  	io_tlb_index = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
242

ad32e8cb8   FUJITA Tomonori   swiotlb: Defer sw...
243
244
  	if (verbose)
  		swiotlb_print_info();
ac2cbab21   Yinghai Lu   x86: Don't panic ...
245

7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
246
  	swiotlb_set_max_segment(io_tlb_nslabs << IO_TLB_SHIFT);
ac2cbab21   Yinghai Lu   x86: Don't panic ...
247
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248
  }
abbceff7d   FUJITA Tomonori   swiotlb: add the ...
249
250
251
252
  /*
   * Statically reserve bounce buffer space and initialize bounce buffer data
   * structures for the software IO TLB used to implement the DMA API.
   */
ac2cbab21   Yinghai Lu   x86: Don't panic ...
253
254
  void  __init
  swiotlb_init(int verbose)
abbceff7d   FUJITA Tomonori   swiotlb: add the ...
255
  {
c729de8fc   Yinghai Lu   x86, kdump: Set c...
256
  	size_t default_size = IO_TLB_DEFAULT_SIZE;
ff7204a74   Alexander Duyck   swiotlb: Make io_...
257
  	unsigned char *vstart;
abbceff7d   FUJITA Tomonori   swiotlb: add the ...
258
259
260
261
262
263
264
265
  	unsigned long bytes;
  
  	if (!io_tlb_nslabs) {
  		io_tlb_nslabs = (default_size >> IO_TLB_SHIFT);
  		io_tlb_nslabs = ALIGN(io_tlb_nslabs, IO_TLB_SEGSIZE);
  	}
  
  	bytes = io_tlb_nslabs << IO_TLB_SHIFT;
ac2cbab21   Yinghai Lu   x86: Don't panic ...
266
  	/* Get IO TLB memory from the low pages */
ad6492b80   Yinghai Lu   memblock, nobootm...
267
  	vstart = memblock_virt_alloc_low_nopanic(PAGE_ALIGN(bytes), PAGE_SIZE);
ac2cbab21   Yinghai Lu   x86: Don't panic ...
268
269
  	if (vstart && !swiotlb_init_with_tbl(vstart, io_tlb_nslabs, verbose))
  		return;
abbceff7d   FUJITA Tomonori   swiotlb: add the ...
270

ac2cbab21   Yinghai Lu   x86: Don't panic ...
271
  	if (io_tlb_start)
457ff1de2   Santosh Shilimkar   lib/swiotlb.c: us...
272
273
  		memblock_free_early(io_tlb_start,
  				    PAGE_ALIGN(io_tlb_nslabs << IO_TLB_SHIFT));
f2a4f7622   Kees Cook   swiotlb: clean up...
274
  	pr_warn("Cannot allocate buffer");
ac2cbab21   Yinghai Lu   x86: Don't panic ...
275
  	no_iotlb_memory = true;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
276
  }
0b9afede3   Alex Williamson   [IA64] more robus...
277
278
279
280
281
282
  /*
   * Systems with larger DMA zones (those that don't support ISA) can
   * initialize the swiotlb later using the slab allocator if needed.
   * This should be just like above, but with some error catching.
   */
  int
563aaf064   Jan Beulich   [IA64] swiotlb cl...
283
  swiotlb_late_init_with_default_size(size_t default_size)
0b9afede3   Alex Williamson   [IA64] more robus...
284
  {
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
285
  	unsigned long bytes, req_nslabs = io_tlb_nslabs;
ff7204a74   Alexander Duyck   swiotlb: Make io_...
286
  	unsigned char *vstart = NULL;
0b9afede3   Alex Williamson   [IA64] more robus...
287
  	unsigned int order;
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
288
  	int rc = 0;
0b9afede3   Alex Williamson   [IA64] more robus...
289
290
291
292
293
294
295
296
297
  
  	if (!io_tlb_nslabs) {
  		io_tlb_nslabs = (default_size >> IO_TLB_SHIFT);
  		io_tlb_nslabs = ALIGN(io_tlb_nslabs, IO_TLB_SEGSIZE);
  	}
  
  	/*
  	 * Get IO TLB memory from the low pages
  	 */
563aaf064   Jan Beulich   [IA64] swiotlb cl...
298
  	order = get_order(io_tlb_nslabs << IO_TLB_SHIFT);
0b9afede3   Alex Williamson   [IA64] more robus...
299
  	io_tlb_nslabs = SLABS_PER_PAGE << order;
563aaf064   Jan Beulich   [IA64] swiotlb cl...
300
  	bytes = io_tlb_nslabs << IO_TLB_SHIFT;
0b9afede3   Alex Williamson   [IA64] more robus...
301
302
  
  	while ((SLABS_PER_PAGE << order) > IO_TLB_MIN_SLABS) {
ff7204a74   Alexander Duyck   swiotlb: Make io_...
303
304
305
  		vstart = (void *)__get_free_pages(GFP_DMA | __GFP_NOWARN,
  						  order);
  		if (vstart)
0b9afede3   Alex Williamson   [IA64] more robus...
306
307
308
  			break;
  		order--;
  	}
ff7204a74   Alexander Duyck   swiotlb: Make io_...
309
  	if (!vstart) {
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
310
311
312
  		io_tlb_nslabs = req_nslabs;
  		return -ENOMEM;
  	}
563aaf064   Jan Beulich   [IA64] swiotlb cl...
313
  	if (order != get_order(bytes)) {
f2a4f7622   Kees Cook   swiotlb: clean up...
314
315
316
  		pr_warn("only able to allocate %ld MB
  ",
  			(PAGE_SIZE << order) >> 20);
0b9afede3   Alex Williamson   [IA64] more robus...
317
318
  		io_tlb_nslabs = SLABS_PER_PAGE << order;
  	}
ff7204a74   Alexander Duyck   swiotlb: Make io_...
319
  	rc = swiotlb_late_init_with_tbl(vstart, io_tlb_nslabs);
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
320
  	if (rc)
ff7204a74   Alexander Duyck   swiotlb: Make io_...
321
  		free_pages((unsigned long)vstart, order);
7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
322

74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
323
324
325
326
327
328
329
  	return rc;
  }
  
  int
  swiotlb_late_init_with_tbl(char *tlb, unsigned long nslabs)
  {
  	unsigned long i, bytes;
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
330
  	unsigned char *v_overflow_buffer;
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
331
332
333
334
  
  	bytes = nslabs << IO_TLB_SHIFT;
  
  	io_tlb_nslabs = nslabs;
ff7204a74   Alexander Duyck   swiotlb: Make io_...
335
336
  	io_tlb_start = virt_to_phys(tlb);
  	io_tlb_end = io_tlb_start + bytes;
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
337

c7753208a   Tom Lendacky   x86, swiotlb: Add...
338
  	swiotlb_set_mem_attributes(tlb, bytes);
ff7204a74   Alexander Duyck   swiotlb: Make io_...
339
  	memset(tlb, 0, bytes);
0b9afede3   Alex Williamson   [IA64] more robus...
340
341
  
  	/*
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
342
343
344
345
346
347
  	 * Get the overflow emergency buffer
  	 */
  	v_overflow_buffer = (void *)__get_free_pages(GFP_DMA,
  						     get_order(io_tlb_overflow));
  	if (!v_overflow_buffer)
  		goto cleanup2;
c7753208a   Tom Lendacky   x86, swiotlb: Add...
348
349
  	swiotlb_set_mem_attributes(v_overflow_buffer, io_tlb_overflow);
  	memset(v_overflow_buffer, 0, io_tlb_overflow);
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
350
351
352
  	io_tlb_overflow_buffer = virt_to_phys(v_overflow_buffer);
  
  	/*
0b9afede3   Alex Williamson   [IA64] more robus...
353
354
355
356
357
358
359
  	 * Allocate and initialize the free list array.  This array is used
  	 * to find contiguous free memory regions of size up to IO_TLB_SEGSIZE
  	 * between io_tlb_start and io_tlb_end.
  	 */
  	io_tlb_list = (unsigned int *)__get_free_pages(GFP_KERNEL,
  	                              get_order(io_tlb_nslabs * sizeof(int)));
  	if (!io_tlb_list)
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
360
  		goto cleanup3;
0b9afede3   Alex Williamson   [IA64] more robus...
361

bc40ac669   Becky Bruce   swiotlb: store ph...
362
363
364
365
  	io_tlb_orig_addr = (phys_addr_t *)
  		__get_free_pages(GFP_KERNEL,
  				 get_order(io_tlb_nslabs *
  					   sizeof(phys_addr_t)));
0b9afede3   Alex Williamson   [IA64] more robus...
366
  	if (!io_tlb_orig_addr)
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
367
  		goto cleanup4;
0b9afede3   Alex Williamson   [IA64] more robus...
368

8e0629c1d   Jan Beulich   swiotlb: don't as...
369
370
371
372
373
  	for (i = 0; i < io_tlb_nslabs; i++) {
  		io_tlb_list[i] = IO_TLB_SEGSIZE - OFFSET(i, IO_TLB_SEGSIZE);
  		io_tlb_orig_addr[i] = INVALID_PHYS_ADDR;
  	}
  	io_tlb_index = 0;
0b9afede3   Alex Williamson   [IA64] more robus...
374

ad32e8cb8   FUJITA Tomonori   swiotlb: Defer sw...
375
  	swiotlb_print_info();
0b9afede3   Alex Williamson   [IA64] more robus...
376

5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
377
  	late_alloc = 1;
7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
378
  	swiotlb_set_max_segment(io_tlb_nslabs << IO_TLB_SHIFT);
0b9afede3   Alex Williamson   [IA64] more robus...
379
380
381
  	return 0;
  
  cleanup4:
25667d675   Tony Luck   Revert "[IA64] sw...
382
383
  	free_pages((unsigned long)io_tlb_list, get_order(io_tlb_nslabs *
  	                                                 sizeof(int)));
0b9afede3   Alex Williamson   [IA64] more robus...
384
  	io_tlb_list = NULL;
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
385
386
387
388
  cleanup3:
  	free_pages((unsigned long)v_overflow_buffer,
  		   get_order(io_tlb_overflow));
  	io_tlb_overflow_buffer = 0;
0b9afede3   Alex Williamson   [IA64] more robus...
389
  cleanup2:
c40dba06e   Alexander Duyck   swiotlb: Make io_...
390
  	io_tlb_end = 0;
ff7204a74   Alexander Duyck   swiotlb: Make io_...
391
  	io_tlb_start = 0;
74838b753   Konrad Rzeszutek Wilk   swiotlb: add the ...
392
  	io_tlb_nslabs = 0;
7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
393
  	max_segment = 0;
0b9afede3   Alex Williamson   [IA64] more robus...
394
395
  	return -ENOMEM;
  }
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
396
397
  void __init swiotlb_free(void)
  {
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
398
  	if (!io_tlb_orig_addr)
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
399
400
401
  		return;
  
  	if (late_alloc) {
ee3f6ba89   Alexander Duyck   swiotlb: Make io_...
402
  		free_pages((unsigned long)phys_to_virt(io_tlb_overflow_buffer),
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
403
404
405
406
407
  			   get_order(io_tlb_overflow));
  		free_pages((unsigned long)io_tlb_orig_addr,
  			   get_order(io_tlb_nslabs * sizeof(phys_addr_t)));
  		free_pages((unsigned long)io_tlb_list, get_order(io_tlb_nslabs *
  								 sizeof(int)));
ff7204a74   Alexander Duyck   swiotlb: Make io_...
408
  		free_pages((unsigned long)phys_to_virt(io_tlb_start),
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
409
410
  			   get_order(io_tlb_nslabs << IO_TLB_SHIFT));
  	} else {
457ff1de2   Santosh Shilimkar   lib/swiotlb.c: us...
411
412
413
414
415
416
417
418
  		memblock_free_late(io_tlb_overflow_buffer,
  				   PAGE_ALIGN(io_tlb_overflow));
  		memblock_free_late(__pa(io_tlb_orig_addr),
  				   PAGE_ALIGN(io_tlb_nslabs * sizeof(phys_addr_t)));
  		memblock_free_late(__pa(io_tlb_list),
  				   PAGE_ALIGN(io_tlb_nslabs * sizeof(int)));
  		memblock_free_late(io_tlb_start,
  				   PAGE_ALIGN(io_tlb_nslabs << IO_TLB_SHIFT));
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
419
  	}
f21ffe9f6   Konrad Rzeszutek Wilk   swiotlb: Expose s...
420
  	io_tlb_nslabs = 0;
7453c549f   Konrad Rzeszutek Wilk   swiotlb: Export s...
421
  	max_segment = 0;
5740afdb6   FUJITA Tomonori   swiotlb: Add swio...
422
  }
9c5a36214   Akinobu Mita   x86: enable DMA C...
423
  int is_swiotlb_buffer(phys_addr_t paddr)
640aebfe0   FUJITA Tomonori   swiotlb: add is_s...
424
  {
ff7204a74   Alexander Duyck   swiotlb: Make io_...
425
  	return paddr >= io_tlb_start && paddr < io_tlb_end;
640aebfe0   FUJITA Tomonori   swiotlb: add is_s...
426
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
427
  /*
fb05a3792   Becky Bruce   swiotlb: add supp...
428
429
   * Bounce: copy the swiotlb buffer back to the original dma location
   */
af51a9f18   Alexander Duyck   swiotlb: Do not e...
430
431
  static void swiotlb_bounce(phys_addr_t orig_addr, phys_addr_t tlb_addr,
  			   size_t size, enum dma_data_direction dir)
fb05a3792   Becky Bruce   swiotlb: add supp...
432
  {
af51a9f18   Alexander Duyck   swiotlb: Do not e...
433
434
  	unsigned long pfn = PFN_DOWN(orig_addr);
  	unsigned char *vaddr = phys_to_virt(tlb_addr);
fb05a3792   Becky Bruce   swiotlb: add supp...
435
436
437
  
  	if (PageHighMem(pfn_to_page(pfn))) {
  		/* The buffer does not have a mapping.  Map it in and copy */
af51a9f18   Alexander Duyck   swiotlb: Do not e...
438
  		unsigned int offset = orig_addr & ~PAGE_MASK;
fb05a3792   Becky Bruce   swiotlb: add supp...
439
440
441
442
443
  		char *buffer;
  		unsigned int sz = 0;
  		unsigned long flags;
  
  		while (size) {
67131ad05   Becky Bruce   swiotlb: fix comp...
444
  			sz = min_t(size_t, PAGE_SIZE - offset, size);
fb05a3792   Becky Bruce   swiotlb: add supp...
445
446
  
  			local_irq_save(flags);
c3eede8e0   Cong Wang   lib: remove the s...
447
  			buffer = kmap_atomic(pfn_to_page(pfn));
fb05a3792   Becky Bruce   swiotlb: add supp...
448
  			if (dir == DMA_TO_DEVICE)
af51a9f18   Alexander Duyck   swiotlb: Do not e...
449
  				memcpy(vaddr, buffer + offset, sz);
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
450
  			else
af51a9f18   Alexander Duyck   swiotlb: Do not e...
451
  				memcpy(buffer + offset, vaddr, sz);
c3eede8e0   Cong Wang   lib: remove the s...
452
  			kunmap_atomic(buffer);
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
453
  			local_irq_restore(flags);
fb05a3792   Becky Bruce   swiotlb: add supp...
454
455
456
  
  			size -= sz;
  			pfn++;
af51a9f18   Alexander Duyck   swiotlb: Do not e...
457
  			vaddr += sz;
fb05a3792   Becky Bruce   swiotlb: add supp...
458
  			offset = 0;
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
459
  		}
af51a9f18   Alexander Duyck   swiotlb: Do not e...
460
461
  	} else if (dir == DMA_TO_DEVICE) {
  		memcpy(vaddr, phys_to_virt(orig_addr), size);
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
462
  	} else {
af51a9f18   Alexander Duyck   swiotlb: Do not e...
463
  		memcpy(phys_to_virt(orig_addr), vaddr, size);
ef9b18935   Jeremy Fitzhardinge   swiotlb: support ...
464
  	}
1b548f667   Jeremy Fitzhardinge   swiotlb: factor o...
465
  }
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
466
467
468
  phys_addr_t swiotlb_tbl_map_single(struct device *hwdev,
  				   dma_addr_t tbl_dma_addr,
  				   phys_addr_t orig_addr, size_t size,
0443fa003   Alexander Duyck   swiotlb: Add supp...
469
470
  				   enum dma_data_direction dir,
  				   unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
471
472
  {
  	unsigned long flags;
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
473
  	phys_addr_t tlb_addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
474
475
  	unsigned int nslots, stride, index, wrap;
  	int i;
681cc5cd3   FUJITA Tomonori   iommu sg merging:...
476
477
478
  	unsigned long mask;
  	unsigned long offset_slots;
  	unsigned long max_slots;
ac2cbab21   Yinghai Lu   x86: Don't panic ...
479
480
  	if (no_iotlb_memory)
  		panic("Can not allocate SWIOTLB buffer earlier and can't now provide you with the DMA bounce buffer");
648babb70   Tom Lendacky   swiotlb: Add warn...
481
482
483
  	if (sme_active())
  		pr_warn_once("SME is active and system is using DMA bounce buffers
  ");
681cc5cd3   FUJITA Tomonori   iommu sg merging:...
484
  	mask = dma_get_seg_boundary(hwdev);
681cc5cd3   FUJITA Tomonori   iommu sg merging:...
485

eb605a575   FUJITA Tomonori   swiotlb: add swio...
486
487
488
  	tbl_dma_addr &= mask;
  
  	offset_slots = ALIGN(tbl_dma_addr, 1 << IO_TLB_SHIFT) >> IO_TLB_SHIFT;
a5ddde4a5   Ian Campbell   swiotlb: add comm...
489
490
491
492
  
  	/*
   	 * Carefully handle integer overflow which can occur when mask == ~0UL.
   	 */
b15a3891c   Jan Beulich   avoid endless loo...
493
494
495
  	max_slots = mask + 1
  		    ? ALIGN(mask + 1, 1 << IO_TLB_SHIFT) >> IO_TLB_SHIFT
  		    : 1UL << (BITS_PER_LONG - IO_TLB_SHIFT);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
496
497
  
  	/*
602d9858f   Nikita Yushchenko   swiotlb: ensure t...
498
499
  	 * For mappings greater than or equal to a page, we limit the stride
  	 * (and hence alignment) to a page size.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
500
501
  	 */
  	nslots = ALIGN(size, 1 << IO_TLB_SHIFT) >> IO_TLB_SHIFT;
602d9858f   Nikita Yushchenko   swiotlb: ensure t...
502
  	if (size >= PAGE_SIZE)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
503
504
505
  		stride = (1 << (PAGE_SHIFT - IO_TLB_SHIFT));
  	else
  		stride = 1;
348145458   Eric Sesterhenn   BUG_ON() Conversi...
506
  	BUG_ON(!nslots);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
507
508
509
510
511
512
  
  	/*
  	 * Find suitable number of IO TLB entries size that will fit this
  	 * request and allocate a buffer from that IO TLB pool.
  	 */
  	spin_lock_irqsave(&io_tlb_lock, flags);
a7133a155   Andrew Morton   lib/swiotlb.c: cl...
513
514
515
516
517
518
  	index = ALIGN(io_tlb_index, stride);
  	if (index >= io_tlb_nslabs)
  		index = 0;
  	wrap = index;
  
  	do {
a85225092   FUJITA Tomonori   swiotlb: use iomm...
519
520
  		while (iommu_is_span_boundary(index, nslots, offset_slots,
  					      max_slots)) {
b15a3891c   Jan Beulich   avoid endless loo...
521
522
523
  			index += stride;
  			if (index >= io_tlb_nslabs)
  				index = 0;
a7133a155   Andrew Morton   lib/swiotlb.c: cl...
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
  			if (index == wrap)
  				goto not_found;
  		}
  
  		/*
  		 * If we find a slot that indicates we have 'nslots' number of
  		 * contiguous buffers, we allocate the buffers from that slot
  		 * and mark the entries as '0' indicating unavailable.
  		 */
  		if (io_tlb_list[index] >= nslots) {
  			int count = 0;
  
  			for (i = index; i < (int) (index + nslots); i++)
  				io_tlb_list[i] = 0;
  			for (i = index - 1; (OFFSET(i, IO_TLB_SEGSIZE) != IO_TLB_SEGSIZE - 1) && io_tlb_list[i]; i--)
  				io_tlb_list[i] = ++count;
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
540
  			tlb_addr = io_tlb_start + (index << IO_TLB_SHIFT);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
541

a7133a155   Andrew Morton   lib/swiotlb.c: cl...
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
  			/*
  			 * Update the indices to avoid searching in the next
  			 * round.
  			 */
  			io_tlb_index = ((index + nslots) < io_tlb_nslabs
  					? (index + nslots) : 0);
  
  			goto found;
  		}
  		index += stride;
  		if (index >= io_tlb_nslabs)
  			index = 0;
  	} while (index != wrap);
  
  not_found:
  	spin_unlock_irqrestore(&io_tlb_lock, flags);
37efa60e1   Christian König   swiotlb: suppress...
558
  	if (!(attrs & DMA_ATTR_NO_WARN) && printk_ratelimit())
0cb637bff   Konrad Rzeszutek Wilk   swiotlb: Don't Do...
559
560
  		dev_warn(hwdev, "swiotlb buffer is full (sz: %zd bytes)
  ", size);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
561
  	return SWIOTLB_MAP_ERROR;
a7133a155   Andrew Morton   lib/swiotlb.c: cl...
562
  found:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
563
564
565
566
567
568
569
  	spin_unlock_irqrestore(&io_tlb_lock, flags);
  
  	/*
  	 * Save away the mapping from the original address to the DMA address.
  	 * This is needed when we sync the memory.  Then we sync the buffer if
  	 * needed.
  	 */
bc40ac669   Becky Bruce   swiotlb: store ph...
570
  	for (i = 0; i < nslots; i++)
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
571
  		io_tlb_orig_addr[index+i] = orig_addr + (i << IO_TLB_SHIFT);
0443fa003   Alexander Duyck   swiotlb: Add supp...
572
573
  	if (!(attrs & DMA_ATTR_SKIP_CPU_SYNC) &&
  	    (dir == DMA_TO_DEVICE || dir == DMA_BIDIRECTIONAL))
af51a9f18   Alexander Duyck   swiotlb: Do not e...
574
  		swiotlb_bounce(orig_addr, tlb_addr, size, DMA_TO_DEVICE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
575

e05ed4d1f   Alexander Duyck   swiotlb: Return p...
576
  	return tlb_addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
577
  }
d7ef1533a   Konrad Rzeszutek Wilk   swiotlb: Make swi...
578
  EXPORT_SYMBOL_GPL(swiotlb_tbl_map_single);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
579
580
  
  /*
eb605a575   FUJITA Tomonori   swiotlb: add swio...
581
582
   * Allocates bounce buffer and returns its kernel virtual address.
   */
023600f19   Alexandre Courbot   swiotlb: do not e...
583
584
  static phys_addr_t
  map_single(struct device *hwdev, phys_addr_t phys, size_t size,
0443fa003   Alexander Duyck   swiotlb: Add supp...
585
  	   enum dma_data_direction dir, unsigned long attrs)
eb605a575   FUJITA Tomonori   swiotlb: add swio...
586
  {
fff5d9922   Geert Uytterhoeven   swiotlb: Add swio...
587
588
589
590
591
592
593
594
  	dma_addr_t start_dma_addr;
  
  	if (swiotlb_force == SWIOTLB_NO_FORCE) {
  		dev_warn_ratelimited(hwdev, "Cannot do DMA to address %pa
  ",
  				     &phys);
  		return SWIOTLB_MAP_ERROR;
  	}
eb605a575   FUJITA Tomonori   swiotlb: add swio...
595

c7753208a   Tom Lendacky   x86, swiotlb: Add...
596
  	start_dma_addr = swiotlb_phys_to_dma(hwdev, io_tlb_start);
0443fa003   Alexander Duyck   swiotlb: Add supp...
597
598
  	return swiotlb_tbl_map_single(hwdev, start_dma_addr, phys, size,
  				      dir, attrs);
eb605a575   FUJITA Tomonori   swiotlb: add swio...
599
600
601
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
602
603
   * dma_addr is the kernel virtual address of the bounce buffer to unmap.
   */
61ca08c32   Alexander Duyck   swiotlb: Use phys...
604
  void swiotlb_tbl_unmap_single(struct device *hwdev, phys_addr_t tlb_addr,
0443fa003   Alexander Duyck   swiotlb: Add supp...
605
606
  			      size_t size, enum dma_data_direction dir,
  			      unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
607
608
609
  {
  	unsigned long flags;
  	int i, count, nslots = ALIGN(size, 1 << IO_TLB_SHIFT) >> IO_TLB_SHIFT;
61ca08c32   Alexander Duyck   swiotlb: Use phys...
610
611
  	int index = (tlb_addr - io_tlb_start) >> IO_TLB_SHIFT;
  	phys_addr_t orig_addr = io_tlb_orig_addr[index];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
612
613
614
615
  
  	/*
  	 * First, sync the memory before unmapping the entry
  	 */
8e0629c1d   Jan Beulich   swiotlb: don't as...
616
  	if (orig_addr != INVALID_PHYS_ADDR &&
0443fa003   Alexander Duyck   swiotlb: Add supp...
617
  	    !(attrs & DMA_ATTR_SKIP_CPU_SYNC) &&
8e0629c1d   Jan Beulich   swiotlb: don't as...
618
  	    ((dir == DMA_FROM_DEVICE) || (dir == DMA_BIDIRECTIONAL)))
af51a9f18   Alexander Duyck   swiotlb: Do not e...
619
  		swiotlb_bounce(orig_addr, tlb_addr, size, DMA_FROM_DEVICE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
620
621
622
  
  	/*
  	 * Return the buffer to the free list by setting the corresponding
af901ca18   André Goddard Rosa   tree-wide: fix as...
623
  	 * entries to indicate the number of contiguous entries available.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
624
625
626
627
628
629
630
631
632
633
634
  	 * While returning the entries to the free list, we merge the entries
  	 * with slots below and above the pool being returned.
  	 */
  	spin_lock_irqsave(&io_tlb_lock, flags);
  	{
  		count = ((index + nslots) < ALIGN(index + 1, IO_TLB_SEGSIZE) ?
  			 io_tlb_list[index + nslots] : 0);
  		/*
  		 * Step 1: return the slots to the free list, merging the
  		 * slots with superceeding slots
  		 */
8e0629c1d   Jan Beulich   swiotlb: don't as...
635
  		for (i = index + nslots - 1; i >= index; i--) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
636
  			io_tlb_list[i] = ++count;
8e0629c1d   Jan Beulich   swiotlb: don't as...
637
638
  			io_tlb_orig_addr[i] = INVALID_PHYS_ADDR;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
639
640
641
642
643
644
645
646
647
  		/*
  		 * Step 2: merge the returned slots with the preceding slots,
  		 * if available (non zero)
  		 */
  		for (i = index - 1; (OFFSET(i, IO_TLB_SEGSIZE) != IO_TLB_SEGSIZE -1) && io_tlb_list[i]; i--)
  			io_tlb_list[i] = ++count;
  	}
  	spin_unlock_irqrestore(&io_tlb_lock, flags);
  }
d7ef1533a   Konrad Rzeszutek Wilk   swiotlb: Make swi...
648
  EXPORT_SYMBOL_GPL(swiotlb_tbl_unmap_single);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
649

fbfda893e   Alexander Duyck   swiotlb: Use phys...
650
651
652
  void swiotlb_tbl_sync_single(struct device *hwdev, phys_addr_t tlb_addr,
  			     size_t size, enum dma_data_direction dir,
  			     enum dma_sync_target target)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
653
  {
fbfda893e   Alexander Duyck   swiotlb: Use phys...
654
655
  	int index = (tlb_addr - io_tlb_start) >> IO_TLB_SHIFT;
  	phys_addr_t orig_addr = io_tlb_orig_addr[index];
bc40ac669   Becky Bruce   swiotlb: store ph...
656

8e0629c1d   Jan Beulich   swiotlb: don't as...
657
658
  	if (orig_addr == INVALID_PHYS_ADDR)
  		return;
fbfda893e   Alexander Duyck   swiotlb: Use phys...
659
  	orig_addr += (unsigned long)tlb_addr & ((1 << IO_TLB_SHIFT) - 1);
df336d1c7   Keir Fraser   Fix swiotlb_sync_...
660

de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
661
662
663
  	switch (target) {
  	case SYNC_FOR_CPU:
  		if (likely(dir == DMA_FROM_DEVICE || dir == DMA_BIDIRECTIONAL))
af51a9f18   Alexander Duyck   swiotlb: Do not e...
664
  			swiotlb_bounce(orig_addr, tlb_addr,
fbfda893e   Alexander Duyck   swiotlb: Use phys...
665
  				       size, DMA_FROM_DEVICE);
348145458   Eric Sesterhenn   BUG_ON() Conversi...
666
667
  		else
  			BUG_ON(dir != DMA_TO_DEVICE);
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
668
669
670
  		break;
  	case SYNC_FOR_DEVICE:
  		if (likely(dir == DMA_TO_DEVICE || dir == DMA_BIDIRECTIONAL))
af51a9f18   Alexander Duyck   swiotlb: Do not e...
671
  			swiotlb_bounce(orig_addr, tlb_addr,
fbfda893e   Alexander Duyck   swiotlb: Use phys...
672
  				       size, DMA_TO_DEVICE);
348145458   Eric Sesterhenn   BUG_ON() Conversi...
673
674
  		else
  			BUG_ON(dir != DMA_FROM_DEVICE);
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
675
676
  		break;
  	default:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
677
  		BUG();
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
678
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
679
  }
d7ef1533a   Konrad Rzeszutek Wilk   swiotlb: Make swi...
680
  EXPORT_SYMBOL_GPL(swiotlb_tbl_sync_single);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
681
682
683
  
  void *
  swiotlb_alloc_coherent(struct device *hwdev, size_t size,
06a544971   Al Viro   [PATCH] gfp_t: dm...
684
  		       dma_addr_t *dma_handle, gfp_t flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
685
  {
37efa60e1   Christian König   swiotlb: suppress...
686
  	bool warn = !(flags & __GFP_NOWARN);
563aaf064   Jan Beulich   [IA64] swiotlb cl...
687
  	dma_addr_t dev_addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
688
689
  	void *ret;
  	int order = get_order(size);
284901a90   Yang Hongyang   dma-mapping: repl...
690
  	u64 dma_mask = DMA_BIT_MASK(32);
1e74f3000   FUJITA Tomonori   swiotlb: use cohe...
691
692
693
  
  	if (hwdev && hwdev->coherent_dma_mask)
  		dma_mask = hwdev->coherent_dma_mask;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
694

25667d675   Tony Luck   Revert "[IA64] sw...
695
  	ret = (void *)__get_free_pages(flags, order);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
696
697
698
699
700
701
702
703
704
  	if (ret) {
  		dev_addr = swiotlb_virt_to_bus(hwdev, ret);
  		if (dev_addr + size - 1 > dma_mask) {
  			/*
  			 * The allocated memory isn't reachable by the device.
  			 */
  			free_pages((unsigned long) ret, order);
  			ret = NULL;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
705
706
707
  	}
  	if (!ret) {
  		/*
bfc5501f6   Konrad Rzeszutek Wilk   swiotlb: Make int...
708
709
  		 * We are either out of memory or the device can't DMA to
  		 * GFP_DMA memory; fall back on map_single(), which
ceb5ac326   Becky Bruce   swiotlb: comment ...
710
  		 * will grab memory from the lowest available address range.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
711
  		 */
37efa60e1   Christian König   swiotlb: suppress...
712
713
  		phys_addr_t paddr = map_single(hwdev, 0, size, DMA_FROM_DEVICE,
  					       warn ? 0 : DMA_ATTR_NO_WARN);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
714
  		if (paddr == SWIOTLB_MAP_ERROR)
94cc81f9a   Joerg Roedel   swiotlb: Warn on ...
715
  			goto err_warn;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
716

e05ed4d1f   Alexander Duyck   swiotlb: Return p...
717
  		ret = phys_to_virt(paddr);
c7753208a   Tom Lendacky   x86, swiotlb: Add...
718
  		dev_addr = swiotlb_phys_to_dma(hwdev, paddr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
719

61ca08c32   Alexander Duyck   swiotlb: Use phys...
720
721
722
723
724
725
  		/* Confirm address can be DMA'd by device */
  		if (dev_addr + size - 1 > dma_mask) {
  			printk("hwdev DMA mask = 0x%016Lx, dev_addr = 0x%016Lx
  ",
  			       (unsigned long long)dma_mask,
  			       (unsigned long long)dev_addr);
a2b89b596   FUJITA Tomonori   swiotlb: remove p...
726

0443fa003   Alexander Duyck   swiotlb: Add supp...
727
728
729
730
  			/*
  			 * DMA_TO_DEVICE to avoid memcpy in unmap_single.
  			 * The DMA_ATTR_SKIP_CPU_SYNC is optional.
  			 */
61ca08c32   Alexander Duyck   swiotlb: Use phys...
731
  			swiotlb_tbl_unmap_single(hwdev, paddr,
0443fa003   Alexander Duyck   swiotlb: Add supp...
732
733
  						 size, DMA_TO_DEVICE,
  						 DMA_ATTR_SKIP_CPU_SYNC);
94cc81f9a   Joerg Roedel   swiotlb: Warn on ...
734
  			goto err_warn;
61ca08c32   Alexander Duyck   swiotlb: Use phys...
735
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
736
  	}
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
737

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
738
  	*dma_handle = dev_addr;
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
739
  	memset(ret, 0, size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
740
  	return ret;
94cc81f9a   Joerg Roedel   swiotlb: Warn on ...
741
742
  
  err_warn:
37efa60e1   Christian König   swiotlb: suppress...
743
  	if (warn && printk_ratelimit()) {
f2a4f7622   Kees Cook   swiotlb: clean up...
744
745
  		pr_warn("coherent allocation failed for device %s size=%zu
  ",
37efa60e1   Christian König   swiotlb: suppress...
746
747
748
  			dev_name(hwdev), size);
  		dump_stack();
  	}
94cc81f9a   Joerg Roedel   swiotlb: Warn on ...
749
750
  
  	return NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
751
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
752
  EXPORT_SYMBOL(swiotlb_alloc_coherent);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
753
754
755
  
  void
  swiotlb_free_coherent(struct device *hwdev, size_t size, void *vaddr,
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
756
  		      dma_addr_t dev_addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
757
  {
862d196b2   FUJITA Tomonori   swiotlb: use phys...
758
  	phys_addr_t paddr = dma_to_phys(hwdev, dev_addr);
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
759

aa24886e3   David Brownell   dma_free_coherent...
760
  	WARN_ON(irqs_disabled());
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
761
762
  	if (!is_swiotlb_buffer(paddr))
  		free_pages((unsigned long)vaddr, get_order(size));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
763
  	else
0443fa003   Alexander Duyck   swiotlb: Add supp...
764
765
766
767
768
769
  		/*
  		 * DMA_TO_DEVICE to avoid memcpy in swiotlb_tbl_unmap_single.
  		 * DMA_ATTR_SKIP_CPU_SYNC is optional.
  		 */
  		swiotlb_tbl_unmap_single(hwdev, paddr, size, DMA_TO_DEVICE,
  					 DMA_ATTR_SKIP_CPU_SYNC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
770
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
771
  EXPORT_SYMBOL(swiotlb_free_coherent);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
772
773
  
  static void
22d482699   Konrad Rzeszutek Wilk   swiotlb: search a...
774
775
  swiotlb_full(struct device *dev, size_t size, enum dma_data_direction dir,
  	     int do_panic)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
776
  {
fff5d9922   Geert Uytterhoeven   swiotlb: Add swio...
777
778
  	if (swiotlb_force == SWIOTLB_NO_FORCE)
  		return;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
779
780
781
  	/*
  	 * Ran out of IOMMU space for this operation. This is very bad.
  	 * Unfortunately the drivers cannot handle this operation properly.
17e5ad6c0   Tony Luck   [PATCH] Removed r...
782
  	 * unless they check for dma_mapping_error (most don't)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
783
784
785
  	 * When the mapping is small enough return a static buffer to limit
  	 * the damage, or panic when the transfer is too big.
  	 */
0d2e18985   Geert Uytterhoeven   swiotlb: Rate-lim...
786
787
788
  	dev_err_ratelimited(dev, "DMA: Out of SW-IOMMU space for %zu bytes
  ",
  			    size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
789

c7084b35e   Casey Dahlin   lib/swiotlb.c: Fi...
790
791
792
793
794
795
796
797
798
799
800
801
  	if (size <= io_tlb_overflow || !do_panic)
  		return;
  
  	if (dir == DMA_BIDIRECTIONAL)
  		panic("DMA: Random memory could be DMA accessed
  ");
  	if (dir == DMA_FROM_DEVICE)
  		panic("DMA: Random memory could be DMA written
  ");
  	if (dir == DMA_TO_DEVICE)
  		panic("DMA: Random memory could be DMA read
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
802
803
804
805
  }
  
  /*
   * Map a single buffer of the indicated size for DMA in streaming mode.  The
17e5ad6c0   Tony Luck   [PATCH] Removed r...
806
   * physical address to use is returned.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
807
808
   *
   * Once the device is given the dma address, the device owns this memory until
ceb5ac326   Becky Bruce   swiotlb: comment ...
809
   * either swiotlb_unmap_page or swiotlb_dma_sync_single is performed.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
810
   */
f98eee8ea   FUJITA Tomonori   x86, ia64: remove...
811
812
813
  dma_addr_t swiotlb_map_page(struct device *dev, struct page *page,
  			    unsigned long offset, size_t size,
  			    enum dma_data_direction dir,
00085f1ef   Krzysztof Kozlowski   dma-mapping: use ...
814
  			    unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
815
  {
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
816
  	phys_addr_t map, phys = page_to_phys(page) + offset;
862d196b2   FUJITA Tomonori   swiotlb: use phys...
817
  	dma_addr_t dev_addr = phys_to_dma(dev, phys);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
818

348145458   Eric Sesterhenn   BUG_ON() Conversi...
819
  	BUG_ON(dir == DMA_NONE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
820
  	/*
ceb5ac326   Becky Bruce   swiotlb: comment ...
821
  	 * If the address happens to be in the device's DMA window,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
822
823
824
  	 * we can safely return the device addr and not worry about bounce
  	 * buffering it.
  	 */
ae7871be1   Geert Uytterhoeven   swiotlb: Convert ...
825
  	if (dma_capable(dev, dev_addr, size) && swiotlb_force != SWIOTLB_FORCE)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
826
  		return dev_addr;
2b2b614dd   Zoltan Kiss   tracing/events: A...
827
  	trace_swiotlb_bounced(dev, dev_addr, size, swiotlb_force);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
828
  	/* Oh well, have to allocate and map a bounce buffer. */
0443fa003   Alexander Duyck   swiotlb: Add supp...
829
  	map = map_single(dev, phys, size, dir, attrs);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
830
  	if (map == SWIOTLB_MAP_ERROR) {
f98eee8ea   FUJITA Tomonori   x86, ia64: remove...
831
  		swiotlb_full(dev, size, dir, 1);
c7753208a   Tom Lendacky   x86, swiotlb: Add...
832
  		return swiotlb_phys_to_dma(dev, io_tlb_overflow_buffer);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
833
  	}
c7753208a   Tom Lendacky   x86, swiotlb: Add...
834
  	dev_addr = swiotlb_phys_to_dma(dev, map);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
835

e05ed4d1f   Alexander Duyck   swiotlb: Return p...
836
  	/* Ensure that the address returned is DMA'ble */
0443fa003   Alexander Duyck   swiotlb: Add supp...
837
838
  	if (dma_capable(dev, dev_addr, size))
  		return dev_addr;
d29fa0cb7   Alexander Duyck   swiotlb: Minor fi...
839
840
  	attrs |= DMA_ATTR_SKIP_CPU_SYNC;
  	swiotlb_tbl_unmap_single(dev, map, size, dir, attrs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
841

c7753208a   Tom Lendacky   x86, swiotlb: Add...
842
  	return swiotlb_phys_to_dma(dev, io_tlb_overflow_buffer);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
843
  }
f98eee8ea   FUJITA Tomonori   x86, ia64: remove...
844
  EXPORT_SYMBOL_GPL(swiotlb_map_page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
845
846
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
847
   * Unmap a single streaming mode DMA translation.  The dma_addr and size must
ceb5ac326   Becky Bruce   swiotlb: comment ...
848
   * match what was provided for in a previous swiotlb_map_page call.  All
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
849
850
851
852
853
   * other usages are undefined.
   *
   * After this call, reads by the cpu to the buffer are guaranteed to see
   * whatever the device wrote there.
   */
7fcebbd2d   Becky Bruce   swiotlb: rename u...
854
  static void unmap_single(struct device *hwdev, dma_addr_t dev_addr,
0443fa003   Alexander Duyck   swiotlb: Add supp...
855
856
  			 size_t size, enum dma_data_direction dir,
  			 unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
857
  {
862d196b2   FUJITA Tomonori   swiotlb: use phys...
858
  	phys_addr_t paddr = dma_to_phys(hwdev, dev_addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
859

348145458   Eric Sesterhenn   BUG_ON() Conversi...
860
  	BUG_ON(dir == DMA_NONE);
7fcebbd2d   Becky Bruce   swiotlb: rename u...
861

02ca646e7   FUJITA Tomonori   swiotlb: remove u...
862
  	if (is_swiotlb_buffer(paddr)) {
0443fa003   Alexander Duyck   swiotlb: Add supp...
863
  		swiotlb_tbl_unmap_single(hwdev, paddr, size, dir, attrs);
7fcebbd2d   Becky Bruce   swiotlb: rename u...
864
865
866
867
868
  		return;
  	}
  
  	if (dir != DMA_FROM_DEVICE)
  		return;
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
869
870
871
872
873
874
875
  	/*
  	 * phys_to_virt doesn't work with hihgmem page but we could
  	 * call dma_mark_clean() with hihgmem page here. However, we
  	 * are fine since dma_mark_clean() is null on POWERPC. We can
  	 * make dma_mark_clean() take a physical address if necessary.
  	 */
  	dma_mark_clean(phys_to_virt(paddr), size);
7fcebbd2d   Becky Bruce   swiotlb: rename u...
876
877
878
879
  }
  
  void swiotlb_unmap_page(struct device *hwdev, dma_addr_t dev_addr,
  			size_t size, enum dma_data_direction dir,
00085f1ef   Krzysztof Kozlowski   dma-mapping: use ...
880
  			unsigned long attrs)
7fcebbd2d   Becky Bruce   swiotlb: rename u...
881
  {
0443fa003   Alexander Duyck   swiotlb: Add supp...
882
  	unmap_single(hwdev, dev_addr, size, dir, attrs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
883
  }
f98eee8ea   FUJITA Tomonori   x86, ia64: remove...
884
  EXPORT_SYMBOL_GPL(swiotlb_unmap_page);
874d6a955   FUJITA Tomonori   swiotlb: clean up...
885

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
886
887
888
889
  /*
   * Make physical memory consistent for a single streaming mode DMA translation
   * after a transfer.
   *
ceb5ac326   Becky Bruce   swiotlb: comment ...
890
   * If you perform a swiotlb_map_page() but wish to interrogate the buffer
17e5ad6c0   Tony Luck   [PATCH] Removed r...
891
892
   * using the cpu, yet do not wish to teardown the dma mapping, you must
   * call this function before doing so.  At the next point you give the dma
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
893
894
895
   * address back to the card, you must first perform a
   * swiotlb_dma_sync_for_device, and then the device again owns the buffer
   */
be6b02678   Andrew Morton   [PATCH] swiotlb u...
896
  static void
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
897
  swiotlb_sync_single(struct device *hwdev, dma_addr_t dev_addr,
d7ef1533a   Konrad Rzeszutek Wilk   swiotlb: Make swi...
898
899
  		    size_t size, enum dma_data_direction dir,
  		    enum dma_sync_target target)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
900
  {
862d196b2   FUJITA Tomonori   swiotlb: use phys...
901
  	phys_addr_t paddr = dma_to_phys(hwdev, dev_addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
902

348145458   Eric Sesterhenn   BUG_ON() Conversi...
903
  	BUG_ON(dir == DMA_NONE);
380d68783   Becky Bruce   swiotlb: use swio...
904

02ca646e7   FUJITA Tomonori   swiotlb: remove u...
905
  	if (is_swiotlb_buffer(paddr)) {
fbfda893e   Alexander Duyck   swiotlb: Use phys...
906
  		swiotlb_tbl_sync_single(hwdev, paddr, size, dir, target);
380d68783   Becky Bruce   swiotlb: use swio...
907
908
909
910
911
  		return;
  	}
  
  	if (dir != DMA_FROM_DEVICE)
  		return;
02ca646e7   FUJITA Tomonori   swiotlb: remove u...
912
  	dma_mark_clean(phys_to_virt(paddr), size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
913
914
915
  }
  
  void
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
916
  swiotlb_sync_single_for_cpu(struct device *hwdev, dma_addr_t dev_addr,
160c1d8e4   FUJITA Tomonori   x86, ia64: conver...
917
  			    size_t size, enum dma_data_direction dir)
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
918
  {
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
919
  	swiotlb_sync_single(hwdev, dev_addr, size, dir, SYNC_FOR_CPU);
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
920
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
921
  EXPORT_SYMBOL(swiotlb_sync_single_for_cpu);
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
922
923
  
  void
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
924
  swiotlb_sync_single_for_device(struct device *hwdev, dma_addr_t dev_addr,
160c1d8e4   FUJITA Tomonori   x86, ia64: conver...
925
  			       size_t size, enum dma_data_direction dir)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
926
  {
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
927
  	swiotlb_sync_single(hwdev, dev_addr, size, dir, SYNC_FOR_DEVICE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
928
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
929
  EXPORT_SYMBOL(swiotlb_sync_single_for_device);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
930
931
932
  
  /*
   * Map a set of buffers described by scatterlist in streaming mode for DMA.
ceb5ac326   Becky Bruce   swiotlb: comment ...
933
   * This is the scatter-gather version of the above swiotlb_map_page
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
934
935
936
937
938
939
940
941
942
943
   * interface.  Here the scatter gather list elements are each tagged with the
   * appropriate dma address and length.  They are obtained via
   * sg_dma_{address,length}(SG).
   *
   * NOTE: An implementation may be able to use a smaller number of
   *       DMA address/length pairs than there are SG table elements.
   *       (for example via virtual mapping capabilities)
   *       The routine returns the number of addr/length pairs actually
   *       used, at most nents.
   *
ceb5ac326   Becky Bruce   swiotlb: comment ...
944
   * Device ownership issues as mentioned above for swiotlb_map_page are the
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
945
946
947
   * same here.
   */
  int
309df0c50   Arthur Kepner   dma/ia64: update ...
948
  swiotlb_map_sg_attrs(struct device *hwdev, struct scatterlist *sgl, int nelems,
00085f1ef   Krzysztof Kozlowski   dma-mapping: use ...
949
  		     enum dma_data_direction dir, unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
950
  {
dbfd49fe9   Jens Axboe   swiotlb: sg chain...
951
  	struct scatterlist *sg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
952
  	int i;
348145458   Eric Sesterhenn   BUG_ON() Conversi...
953
  	BUG_ON(dir == DMA_NONE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
954

dbfd49fe9   Jens Axboe   swiotlb: sg chain...
955
  	for_each_sg(sgl, sg, nelems, i) {
961d7d0ee   Ian Campbell   swiotlb: do not u...
956
  		phys_addr_t paddr = sg_phys(sg);
862d196b2   FUJITA Tomonori   swiotlb: use phys...
957
  		dma_addr_t dev_addr = phys_to_dma(hwdev, paddr);
bc40ac669   Becky Bruce   swiotlb: store ph...
958

ae7871be1   Geert Uytterhoeven   swiotlb: Convert ...
959
  		if (swiotlb_force == SWIOTLB_FORCE ||
b9394647a   FUJITA Tomonori   swiotlb: use dma_...
960
  		    !dma_capable(hwdev, dev_addr, sg->length)) {
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
961
  			phys_addr_t map = map_single(hwdev, sg_phys(sg),
0443fa003   Alexander Duyck   swiotlb: Add supp...
962
  						     sg->length, dir, attrs);
e05ed4d1f   Alexander Duyck   swiotlb: Return p...
963
  			if (map == SWIOTLB_MAP_ERROR) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
964
965
966
  				/* Don't panic here, we expect map_sg users
  				   to do proper error handling. */
  				swiotlb_full(hwdev, sg->length, dir, 0);
d29fa0cb7   Alexander Duyck   swiotlb: Minor fi...
967
  				attrs |= DMA_ATTR_SKIP_CPU_SYNC;
309df0c50   Arthur Kepner   dma/ia64: update ...
968
969
  				swiotlb_unmap_sg_attrs(hwdev, sgl, i, dir,
  						       attrs);
4d86ec7a8   EunBong Song   swiotlb: replace ...
970
  				sg_dma_len(sgl) = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
971
972
  				return 0;
  			}
c7753208a   Tom Lendacky   x86, swiotlb: Add...
973
  			sg->dma_address = swiotlb_phys_to_dma(hwdev, map);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
974
975
  		} else
  			sg->dma_address = dev_addr;
4d86ec7a8   EunBong Song   swiotlb: replace ...
976
  		sg_dma_len(sg) = sg->length;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
977
978
979
  	}
  	return nelems;
  }
309df0c50   Arthur Kepner   dma/ia64: update ...
980
  EXPORT_SYMBOL(swiotlb_map_sg_attrs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
981
982
  /*
   * Unmap a set of streaming mode DMA translations.  Again, cpu read rules
ceb5ac326   Becky Bruce   swiotlb: comment ...
983
   * concerning calls here are the same as for swiotlb_unmap_page() above.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
984
985
   */
  void
309df0c50   Arthur Kepner   dma/ia64: update ...
986
  swiotlb_unmap_sg_attrs(struct device *hwdev, struct scatterlist *sgl,
00085f1ef   Krzysztof Kozlowski   dma-mapping: use ...
987
988
  		       int nelems, enum dma_data_direction dir,
  		       unsigned long attrs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
989
  {
dbfd49fe9   Jens Axboe   swiotlb: sg chain...
990
  	struct scatterlist *sg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
991
  	int i;
348145458   Eric Sesterhenn   BUG_ON() Conversi...
992
  	BUG_ON(dir == DMA_NONE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
993

7fcebbd2d   Becky Bruce   swiotlb: rename u...
994
  	for_each_sg(sgl, sg, nelems, i)
0443fa003   Alexander Duyck   swiotlb: Add supp...
995
996
  		unmap_single(hwdev, sg->dma_address, sg_dma_len(sg), dir,
  			     attrs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
997
  }
309df0c50   Arthur Kepner   dma/ia64: update ...
998
  EXPORT_SYMBOL(swiotlb_unmap_sg_attrs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
999
1000
1001
1002
1003
1004
1005
  /*
   * Make physical memory consistent for a set of streaming mode DMA translations
   * after a transfer.
   *
   * The same as swiotlb_sync_single_* but for a scatter-gather list, same rules
   * and usage.
   */
be6b02678   Andrew Morton   [PATCH] swiotlb u...
1006
  static void
dbfd49fe9   Jens Axboe   swiotlb: sg chain...
1007
  swiotlb_sync_sg(struct device *hwdev, struct scatterlist *sgl,
d7ef1533a   Konrad Rzeszutek Wilk   swiotlb: Make swi...
1008
1009
  		int nelems, enum dma_data_direction dir,
  		enum dma_sync_target target)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1010
  {
dbfd49fe9   Jens Axboe   swiotlb: sg chain...
1011
  	struct scatterlist *sg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1012
  	int i;
380d68783   Becky Bruce   swiotlb: use swio...
1013
1014
  	for_each_sg(sgl, sg, nelems, i)
  		swiotlb_sync_single(hwdev, sg->dma_address,
4d86ec7a8   EunBong Song   swiotlb: replace ...
1015
  				    sg_dma_len(sg), dir, target);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1016
1017
1018
  }
  
  void
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
1019
  swiotlb_sync_sg_for_cpu(struct device *hwdev, struct scatterlist *sg,
160c1d8e4   FUJITA Tomonori   x86, ia64: conver...
1020
  			int nelems, enum dma_data_direction dir)
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
1021
  {
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
1022
  	swiotlb_sync_sg(hwdev, sg, nelems, dir, SYNC_FOR_CPU);
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
1023
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
1024
  EXPORT_SYMBOL(swiotlb_sync_sg_for_cpu);
8270f3f1a   John W. Linville   [PATCH] swiotlb: ...
1025
1026
  
  void
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1027
  swiotlb_sync_sg_for_device(struct device *hwdev, struct scatterlist *sg,
160c1d8e4   FUJITA Tomonori   x86, ia64: conver...
1028
  			   int nelems, enum dma_data_direction dir)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1029
  {
de69e0f0b   John W. Linville   [PATCH] swiotlb: ...
1030
  	swiotlb_sync_sg(hwdev, sg, nelems, dir, SYNC_FOR_DEVICE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1031
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
1032
  EXPORT_SYMBOL(swiotlb_sync_sg_for_device);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1033
1034
  
  int
8d8bb39b9   FUJITA Tomonori   dma-mapping: add ...
1035
  swiotlb_dma_mapping_error(struct device *hwdev, dma_addr_t dma_addr)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1036
  {
c7753208a   Tom Lendacky   x86, swiotlb: Add...
1037
  	return (dma_addr == swiotlb_phys_to_dma(hwdev, io_tlb_overflow_buffer));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1038
  }
874d6a955   FUJITA Tomonori   swiotlb: clean up...
1039
  EXPORT_SYMBOL(swiotlb_dma_mapping_error);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1040
1041
  
  /*
17e5ad6c0   Tony Luck   [PATCH] Removed r...
1042
   * Return whether the given device DMA address mask can be supported
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1043
   * properly.  For example, if your device can only drive the low 24-bits
17e5ad6c0   Tony Luck   [PATCH] Removed r...
1044
   * during bus mastering, then you would pass 0x00ffffff as the mask to
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1045
1046
1047
   * this function.
   */
  int
563aaf064   Jan Beulich   [IA64] swiotlb cl...
1048
  swiotlb_dma_supported(struct device *hwdev, u64 mask)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1049
  {
c7753208a   Tom Lendacky   x86, swiotlb: Add...
1050
  	return swiotlb_phys_to_dma(hwdev, io_tlb_end - 1) <= mask;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1051
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1052
  EXPORT_SYMBOL(swiotlb_dma_supported);