1
2
3
4
5
6
7
8#include <linux/kernel.h>
9#include <linux/slab.h>
10#include <linux/list.h>
11#include <linux/bug.h>
12
13#include "dma_fifo.h"
14
15#ifdef DEBUG_TRACING
16#define df_trace(s, args...) pr_debug(s, ##args)
17#else
18#define df_trace(s, args...)
19#endif
20
21#define FAIL(fifo, condition, format...) ({ \
22 fifo->corrupt = !!(condition); \
23 WARN(fifo->corrupt, format); \
24})
25
26
27
28
29static bool addr_check(unsigned int check, unsigned int lo, unsigned int hi)
30{
31 return check - (lo + 1) < (hi - 1) - lo;
32}
33
34
35
36
37
38void dma_fifo_init(struct dma_fifo *fifo)
39{
40 memset(fifo, 0, sizeof(*fifo));
41 INIT_LIST_HEAD(&fifo->pending);
42}
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58int dma_fifo_alloc(struct dma_fifo *fifo, int size, unsigned int align,
59 int tx_limit, int open_limit, gfp_t gfp_mask)
60{
61 int capacity;
62
63 if (!is_power_of_2(align) || size < 0)
64 return -EINVAL;
65
66 size = round_up(size, align);
67 capacity = size + align * open_limit + align * DMA_FIFO_GUARD;
68 fifo->data = kmalloc(capacity, gfp_mask);
69 if (!fifo->data)
70 return -ENOMEM;
71
72 fifo->in = 0;
73 fifo->out = 0;
74 fifo->done = 0;
75 fifo->size = size;
76 fifo->avail = size;
77 fifo->align = align;
78 fifo->tx_limit = max_t(int, round_down(tx_limit, align), align);
79 fifo->open = 0;
80 fifo->open_limit = open_limit;
81 fifo->guard = size + align * open_limit;
82 fifo->capacity = capacity;
83 fifo->corrupt = 0;
84
85 return 0;
86}
87
88
89
90
91
92
93
94
95
96void dma_fifo_free(struct dma_fifo *fifo)
97{
98 struct dma_pending *pending, *next;
99
100 if (!fifo->data)
101 return;
102
103 list_for_each_entry_safe(pending, next, &fifo->pending, link)
104 list_del_init(&pending->link);
105 kfree(fifo->data);
106 fifo->data = NULL;
107}
108
109
110
111
112
113void dma_fifo_reset(struct dma_fifo *fifo)
114{
115 struct dma_pending *pending, *next;
116
117 if (!fifo->data)
118 return;
119
120 list_for_each_entry_safe(pending, next, &fifo->pending, link)
121 list_del_init(&pending->link);
122 fifo->in = 0;
123 fifo->out = 0;
124 fifo->done = 0;
125 fifo->avail = fifo->size;
126 fifo->open = 0;
127 fifo->corrupt = 0;
128}
129
130
131
132
133
134
135
136
137
138
139int dma_fifo_in(struct dma_fifo *fifo, const void *src, int n)
140{
141 int ofs, l;
142
143 if (!fifo->data)
144 return -ENOENT;
145 if (fifo->corrupt)
146 return -ENXIO;
147
148 if (n > fifo->avail)
149 n = fifo->avail;
150 if (n <= 0)
151 return 0;
152
153 ofs = fifo->in % fifo->capacity;
154 l = min(n, fifo->capacity - ofs);
155 memcpy(fifo->data + ofs, src, l);
156 memcpy(fifo->data, src + l, n - l);
157
158 if (FAIL(fifo, addr_check(fifo->done, fifo->in, fifo->in + n) ||
159 fifo->avail < n,
160 "fifo corrupt: in:%u out:%u done:%u n:%d avail:%d",
161 fifo->in, fifo->out, fifo->done, n, fifo->avail))
162 return -ENXIO;
163
164 fifo->in += n;
165 fifo->avail -= n;
166
167 df_trace("in:%u out:%u done:%u n:%d avail:%d", fifo->in, fifo->out,
168 fifo->done, n, fifo->avail);
169
170 return n;
171}
172
173
174
175
176
177
178
179
180
181
182int dma_fifo_out_pend(struct dma_fifo *fifo, struct dma_pending *pended)
183{
184 unsigned int len, n, ofs, l, limit;
185
186 if (!fifo->data)
187 return -ENOENT;
188 if (fifo->corrupt)
189 return -ENXIO;
190
191 pended->len = 0;
192 pended->data = NULL;
193 pended->out = fifo->out;
194
195 len = fifo->in - fifo->out;
196 if (!len)
197 return -ENODATA;
198 if (fifo->open == fifo->open_limit)
199 return -EAGAIN;
200
201 n = len;
202 ofs = fifo->out % fifo->capacity;
203 l = fifo->capacity - ofs;
204 limit = min_t(unsigned int, l, fifo->tx_limit);
205 if (n > limit) {
206 n = limit;
207 fifo->out += limit;
208 } else if (ofs + n > fifo->guard) {
209 fifo->out += l;
210 fifo->in = fifo->out;
211 } else {
212 fifo->out += round_up(n, fifo->align);
213 fifo->in = fifo->out;
214 }
215
216 df_trace("in: %u out: %u done: %u n: %d len: %u avail: %d", fifo->in,
217 fifo->out, fifo->done, n, len, fifo->avail);
218
219 pended->len = n;
220 pended->data = fifo->data + ofs;
221 pended->next = fifo->out;
222 list_add_tail(&pended->link, &fifo->pending);
223 ++fifo->open;
224
225 if (FAIL(fifo, fifo->open > fifo->open_limit,
226 "past open limit:%d (limit:%d)",
227 fifo->open, fifo->open_limit))
228 return -ENXIO;
229 if (FAIL(fifo, fifo->out & (fifo->align - 1),
230 "fifo out unaligned:%u (align:%u)",
231 fifo->out, fifo->align))
232 return -ENXIO;
233
234 return len - n;
235}
236
237
238
239
240
241
242int dma_fifo_out_complete(struct dma_fifo *fifo, struct dma_pending *complete)
243{
244 struct dma_pending *pending, *next, *tmp;
245
246 if (!fifo->data)
247 return -ENOENT;
248 if (fifo->corrupt)
249 return -ENXIO;
250 if (list_empty(&fifo->pending) && fifo->open == 0)
251 return -EINVAL;
252
253 if (FAIL(fifo, list_empty(&fifo->pending) != (fifo->open == 0),
254 "pending list disagrees with open count:%d",
255 fifo->open))
256 return -ENXIO;
257
258 tmp = complete->data;
259 *tmp = *complete;
260 list_replace(&complete->link, &tmp->link);
261 dp_mark_completed(tmp);
262
263
264 list_for_each_entry_safe(pending, next, &fifo->pending, link) {
265 if (!dp_is_completed(pending)) {
266 df_trace("still pending: saved out: %u len: %d",
267 pending->out, pending->len);
268 break;
269 }
270
271 if (FAIL(fifo, pending->out != fifo->done ||
272 addr_check(fifo->in, fifo->done, pending->next),
273 "in:%u out:%u done:%u saved:%u next:%u",
274 fifo->in, fifo->out, fifo->done, pending->out,
275 pending->next))
276 return -ENXIO;
277
278 list_del_init(&pending->link);
279 fifo->done = pending->next;
280 fifo->avail += pending->len;
281 --fifo->open;
282
283 df_trace("in: %u out: %u done: %u len: %u avail: %d", fifo->in,
284 fifo->out, fifo->done, pending->len, fifo->avail);
285 }
286
287 if (FAIL(fifo, fifo->open < 0, "open dma:%d < 0", fifo->open))
288 return -ENXIO;
289 if (FAIL(fifo, fifo->avail > fifo->size, "fifo avail:%d > size:%d",
290 fifo->avail, fifo->size))
291 return -ENXIO;
292
293 return 0;
294}
295