Blame view

arch/microblaze/kernel/dma.c 5.26 KB
ccfe27d70   Michal Simek   microblaze: Suppo...
1
2
3
4
5
6
7
8
9
10
  /*
   * Copyright (C) 2009-2010 PetaLogix
   * Copyright (C) 2006 Benjamin Herrenschmidt, IBM Corporation
   *
   * Provide default implementations of the DMA mapping callbacks for
   * directly mapped busses.
   */
  
  #include <linux/device.h>
  #include <linux/dma-mapping.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
11
  #include <linux/gfp.h>
ccfe27d70   Michal Simek   microblaze: Suppo...
12
  #include <linux/dma-debug.h>
66421a648   Paul Gortmaker   microblaze: Add e...
13
  #include <linux/export.h>
6bd55f0bb   Michal Simek   microblaze: Fix c...
14
  #include <linux/bug.h>
ccfe27d70   Michal Simek   microblaze: Suppo...
15
16
17
18
19
20
21
22
23
  
  /*
   * Generic direct DMA implementation
   *
   * This implementation supports a per-device offset that can be applied if
   * the address at which memory is visible to devices is not 0. Platform code
   * can set archdata.dma_data to an unsigned long holding the offset. By
   * default the offset is PCI_DRAM_OFFSET.
   */
2549edd35   Michal Simek   microblaze: Imple...
24

ccfe27d70   Michal Simek   microblaze: Suppo...
25
26
  static unsigned long get_dma_direct_offset(struct device *dev)
  {
78ebfa884   Michal Simek   microblaze: Addin...
27
  	if (likely(dev))
ccfe27d70   Michal Simek   microblaze: Suppo...
28
29
30
31
  		return (unsigned long)dev->archdata.dma_data;
  
  	return PCI_DRAM_OFFSET; /* FIXME Not sure if is correct */
  }
1be53e084   Michal Simek   microblaze: Fix d...
32
33
34
  #define NOT_COHERENT_CACHE
  
  static void *dma_direct_alloc_coherent(struct device *dev, size_t size,
988624ec1   Andrzej Pietrasiewicz   Microblaze: adapt...
35
36
  				       dma_addr_t *dma_handle, gfp_t flag,
  				       struct dma_attrs *attrs)
ccfe27d70   Michal Simek   microblaze: Suppo...
37
  {
1be53e084   Michal Simek   microblaze: Fix d...
38
39
40
  #ifdef NOT_COHERENT_CACHE
  	return consistent_alloc(flag, size, dma_handle);
  #else
ccfe27d70   Michal Simek   microblaze: Suppo...
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
  	void *ret;
  	struct page *page;
  	int node = dev_to_node(dev);
  
  	/* ignore region specifiers */
  	flag  &= ~(__GFP_HIGHMEM);
  
  	page = alloc_pages_node(node, flag, get_order(size));
  	if (page == NULL)
  		return NULL;
  	ret = page_address(page);
  	memset(ret, 0, size);
  	*dma_handle = virt_to_phys(ret) + get_dma_direct_offset(dev);
  
  	return ret;
1be53e084   Michal Simek   microblaze: Fix d...
56
  #endif
ccfe27d70   Michal Simek   microblaze: Suppo...
57
  }
1be53e084   Michal Simek   microblaze: Fix d...
58
  static void dma_direct_free_coherent(struct device *dev, size_t size,
988624ec1   Andrzej Pietrasiewicz   Microblaze: adapt...
59
60
  				     void *vaddr, dma_addr_t dma_handle,
  				     struct dma_attrs *attrs)
ccfe27d70   Michal Simek   microblaze: Suppo...
61
  {
1be53e084   Michal Simek   microblaze: Fix d...
62
  #ifdef NOT_COHERENT_CACHE
f1525765f   Michal Simek   microblaze: Fix c...
63
  	consistent_free(size, vaddr);
1be53e084   Michal Simek   microblaze: Fix d...
64
  #else
ccfe27d70   Michal Simek   microblaze: Suppo...
65
  	free_pages((unsigned long)vaddr, get_order(size));
1be53e084   Michal Simek   microblaze: Fix d...
66
  #endif
ccfe27d70   Michal Simek   microblaze: Suppo...
67
68
69
70
71
72
73
74
  }
  
  static int dma_direct_map_sg(struct device *dev, struct scatterlist *sgl,
  			     int nents, enum dma_data_direction direction,
  			     struct dma_attrs *attrs)
  {
  	struct scatterlist *sg;
  	int i;
d79f3b06a   Michal Simek   microblaze: Preli...
75
  	/* FIXME this part of code is untested */
ccfe27d70   Michal Simek   microblaze: Suppo...
76
77
  	for_each_sg(sgl, sg, nents, i) {
  		sg->dma_address = sg_phys(sg) + get_dma_direct_offset(dev);
cf560c180   Eli Billauer   microblaze: Moved...
78
  		__dma_sync(page_to_phys(sg_page(sg)) + sg->offset,
d79f3b06a   Michal Simek   microblaze: Preli...
79
  							sg->length, direction);
ccfe27d70   Michal Simek   microblaze: Suppo...
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
  	}
  
  	return nents;
  }
  
  static void dma_direct_unmap_sg(struct device *dev, struct scatterlist *sg,
  				int nents, enum dma_data_direction direction,
  				struct dma_attrs *attrs)
  {
  }
  
  static int dma_direct_dma_supported(struct device *dev, u64 mask)
  {
  	return 1;
  }
  
  static inline dma_addr_t dma_direct_map_page(struct device *dev,
  					     struct page *page,
  					     unsigned long offset,
  					     size_t size,
2549edd35   Michal Simek   microblaze: Imple...
100
  					     enum dma_data_direction direction,
ccfe27d70   Michal Simek   microblaze: Suppo...
101
102
  					     struct dma_attrs *attrs)
  {
cf560c180   Eli Billauer   microblaze: Moved...
103
  	__dma_sync(page_to_phys(page) + offset, size, direction);
ccfe27d70   Michal Simek   microblaze: Suppo...
104
105
106
107
108
109
110
111
112
  	return page_to_phys(page) + offset + get_dma_direct_offset(dev);
  }
  
  static inline void dma_direct_unmap_page(struct device *dev,
  					 dma_addr_t dma_address,
  					 size_t size,
  					 enum dma_data_direction direction,
  					 struct dma_attrs *attrs)
  {
d79f3b06a   Michal Simek   microblaze: Preli...
113
114
115
116
117
  /* There is not necessary to do cache cleanup
   *
   * phys_to_virt is here because in __dma_sync_page is __virt_to_phys and
   * dma_address is physical address
   */
cf560c180   Eli Billauer   microblaze: Moved...
118
  	__dma_sync(dma_address, size, direction);
ccfe27d70   Michal Simek   microblaze: Suppo...
119
  }
0fb2a6f28   Eli Billauer   microblaze: Added...
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
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
165
166
167
168
169
170
171
172
173
174
  static inline void
  dma_direct_sync_single_for_cpu(struct device *dev,
  			       dma_addr_t dma_handle, size_t size,
  			       enum dma_data_direction direction)
  {
  	/*
  	 * It's pointless to flush the cache as the memory segment
  	 * is given to the CPU
  	 */
  
  	if (direction == DMA_FROM_DEVICE)
  		__dma_sync(dma_handle, size, direction);
  }
  
  static inline void
  dma_direct_sync_single_for_device(struct device *dev,
  				  dma_addr_t dma_handle, size_t size,
  				  enum dma_data_direction direction)
  {
  	/*
  	 * It's pointless to invalidate the cache if the device isn't
  	 * supposed to write to the relevant region
  	 */
  
  	if (direction == DMA_TO_DEVICE)
  		__dma_sync(dma_handle, size, direction);
  }
  
  static inline void
  dma_direct_sync_sg_for_cpu(struct device *dev,
  			   struct scatterlist *sgl, int nents,
  			   enum dma_data_direction direction)
  {
  	struct scatterlist *sg;
  	int i;
  
  	/* FIXME this part of code is untested */
  	if (direction == DMA_FROM_DEVICE)
  		for_each_sg(sgl, sg, nents, i)
  			__dma_sync(sg->dma_address, sg->length, direction);
  }
  
  static inline void
  dma_direct_sync_sg_for_device(struct device *dev,
  			      struct scatterlist *sgl, int nents,
  			      enum dma_data_direction direction)
  {
  	struct scatterlist *sg;
  	int i;
  
  	/* FIXME this part of code is untested */
  	if (direction == DMA_TO_DEVICE)
  		for_each_sg(sgl, sg, nents, i)
  			__dma_sync(sg->dma_address, sg->length, direction);
  }
ccfe27d70   Michal Simek   microblaze: Suppo...
175
  struct dma_map_ops dma_direct_ops = {
988624ec1   Andrzej Pietrasiewicz   Microblaze: adapt...
176
177
  	.alloc		= dma_direct_alloc_coherent,
  	.free		= dma_direct_free_coherent,
ccfe27d70   Michal Simek   microblaze: Suppo...
178
179
180
181
182
  	.map_sg		= dma_direct_map_sg,
  	.unmap_sg	= dma_direct_unmap_sg,
  	.dma_supported	= dma_direct_dma_supported,
  	.map_page	= dma_direct_map_page,
  	.unmap_page	= dma_direct_unmap_page,
0fb2a6f28   Eli Billauer   microblaze: Added...
183
184
185
186
  	.sync_single_for_cpu		= dma_direct_sync_single_for_cpu,
  	.sync_single_for_device		= dma_direct_sync_single_for_device,
  	.sync_sg_for_cpu		= dma_direct_sync_sg_for_cpu,
  	.sync_sg_for_device		= dma_direct_sync_sg_for_device,
ccfe27d70   Michal Simek   microblaze: Suppo...
187
188
189
190
191
192
193
194
  };
  EXPORT_SYMBOL(dma_direct_ops);
  
  /* Number of entries preallocated for DMA-API debugging */
  #define PREALLOC_DMA_DEBUG_ENTRIES (1 << 16)
  
  static int __init dma_init(void)
  {
6bd55f0bb   Michal Simek   microblaze: Fix c...
195
  	dma_debug_init(PREALLOC_DMA_DEBUG_ENTRIES);
ccfe27d70   Michal Simek   microblaze: Suppo...
196

6bd55f0bb   Michal Simek   microblaze: Fix c...
197
  	return 0;
ccfe27d70   Michal Simek   microblaze: Suppo...
198
199
  }
  fs_initcall(dma_init);