This source file includes following definitions.
- hl_cq_inc_ptr
- hl_eq_inc_ptr
- irq_handle_eqe
- hl_irq_handler_cq
- hl_irq_handler_eq
- hl_cq_init
- hl_cq_fini
- hl_cq_reset
- hl_eq_init
- hl_eq_fini
- hl_eq_reset
1
2
3
4
5
6
7
8 #include "habanalabs.h"
9
10 #include <linux/slab.h>
11
12
13
14
15
16
17
18
19 struct hl_eqe_work {
20 struct work_struct eq_work;
21 struct hl_device *hdev;
22 struct hl_eq_entry eq_entry;
23 };
24
25
26
27
28
29
30
31
32
33 inline u32 hl_cq_inc_ptr(u32 ptr)
34 {
35 ptr++;
36 if (unlikely(ptr == HL_CQ_LENGTH))
37 ptr = 0;
38 return ptr;
39 }
40
41
42
43
44
45
46
47
48
49 inline u32 hl_eq_inc_ptr(u32 ptr)
50 {
51 ptr++;
52 if (unlikely(ptr == HL_EQ_LENGTH))
53 ptr = 0;
54 return ptr;
55 }
56
57 static void irq_handle_eqe(struct work_struct *work)
58 {
59 struct hl_eqe_work *eqe_work = container_of(work, struct hl_eqe_work,
60 eq_work);
61 struct hl_device *hdev = eqe_work->hdev;
62
63 hdev->asic_funcs->handle_eqe(hdev, &eqe_work->eq_entry);
64
65 kfree(eqe_work);
66 }
67
68
69
70
71
72
73
74
75 irqreturn_t hl_irq_handler_cq(int irq, void *arg)
76 {
77 struct hl_cq *cq = arg;
78 struct hl_device *hdev = cq->hdev;
79 struct hl_hw_queue *queue;
80 struct hl_cs_job *job;
81 bool shadow_index_valid;
82 u16 shadow_index;
83 struct hl_cq_entry *cq_entry, *cq_base;
84
85 if (hdev->disabled) {
86 dev_dbg(hdev->dev,
87 "Device disabled but received IRQ %d for CQ %d\n",
88 irq, cq->hw_queue_id);
89 return IRQ_HANDLED;
90 }
91
92 cq_base = (struct hl_cq_entry *) (uintptr_t) cq->kernel_address;
93
94 while (1) {
95 bool entry_ready = ((le32_to_cpu(cq_base[cq->ci].data) &
96 CQ_ENTRY_READY_MASK)
97 >> CQ_ENTRY_READY_SHIFT);
98
99 if (!entry_ready)
100 break;
101
102 cq_entry = (struct hl_cq_entry *) &cq_base[cq->ci];
103
104
105
106
107 dma_rmb();
108
109 shadow_index_valid = ((le32_to_cpu(cq_entry->data) &
110 CQ_ENTRY_SHADOW_INDEX_VALID_MASK)
111 >> CQ_ENTRY_SHADOW_INDEX_VALID_SHIFT);
112
113 shadow_index = (u16) ((le32_to_cpu(cq_entry->data) &
114 CQ_ENTRY_SHADOW_INDEX_MASK)
115 >> CQ_ENTRY_SHADOW_INDEX_SHIFT);
116
117 queue = &hdev->kernel_queues[cq->hw_queue_id];
118
119 if ((shadow_index_valid) && (!hdev->disabled)) {
120 job = queue->shadow_queue[hl_pi_2_offset(shadow_index)];
121 queue_work(hdev->cq_wq, &job->finish_work);
122 }
123
124
125
126
127
128
129 queue->ci = hl_queue_inc_ptr(queue->ci);
130
131
132 cq_entry->data = cpu_to_le32(le32_to_cpu(cq_entry->data) &
133 ~CQ_ENTRY_READY_MASK);
134
135 cq->ci = hl_cq_inc_ptr(cq->ci);
136
137
138 atomic_inc(&cq->free_slots_cnt);
139 }
140
141 return IRQ_HANDLED;
142 }
143
144
145
146
147
148
149
150
151 irqreturn_t hl_irq_handler_eq(int irq, void *arg)
152 {
153 struct hl_eq *eq = arg;
154 struct hl_device *hdev = eq->hdev;
155 struct hl_eq_entry *eq_entry;
156 struct hl_eq_entry *eq_base;
157 struct hl_eqe_work *handle_eqe_work;
158
159 eq_base = (struct hl_eq_entry *) (uintptr_t) eq->kernel_address;
160
161 while (1) {
162 bool entry_ready =
163 ((le32_to_cpu(eq_base[eq->ci].hdr.ctl) &
164 EQ_CTL_READY_MASK) >> EQ_CTL_READY_SHIFT);
165
166 if (!entry_ready)
167 break;
168
169 eq_entry = &eq_base[eq->ci];
170
171
172
173
174
175 dma_rmb();
176
177 if (hdev->disabled) {
178 dev_warn(hdev->dev,
179 "Device disabled but received IRQ %d for EQ\n",
180 irq);
181 goto skip_irq;
182 }
183
184 handle_eqe_work = kmalloc(sizeof(*handle_eqe_work), GFP_ATOMIC);
185 if (handle_eqe_work) {
186 INIT_WORK(&handle_eqe_work->eq_work, irq_handle_eqe);
187 handle_eqe_work->hdev = hdev;
188
189 memcpy(&handle_eqe_work->eq_entry, eq_entry,
190 sizeof(*eq_entry));
191
192 queue_work(hdev->eq_wq, &handle_eqe_work->eq_work);
193 }
194 skip_irq:
195
196 eq_entry->hdr.ctl =
197 cpu_to_le32(le32_to_cpu(eq_entry->hdr.ctl) &
198 ~EQ_CTL_READY_MASK);
199
200 eq->ci = hl_eq_inc_ptr(eq->ci);
201
202 hdev->asic_funcs->update_eq_ci(hdev, eq->ci);
203 }
204
205 return IRQ_HANDLED;
206 }
207
208
209
210
211
212
213
214
215
216
217
218 int hl_cq_init(struct hl_device *hdev, struct hl_cq *q, u32 hw_queue_id)
219 {
220 void *p;
221
222 BUILD_BUG_ON(HL_CQ_SIZE_IN_BYTES > HL_PAGE_SIZE);
223
224 p = hdev->asic_funcs->asic_dma_alloc_coherent(hdev, HL_CQ_SIZE_IN_BYTES,
225 &q->bus_address, GFP_KERNEL | __GFP_ZERO);
226 if (!p)
227 return -ENOMEM;
228
229 q->hdev = hdev;
230 q->kernel_address = (u64) (uintptr_t) p;
231 q->hw_queue_id = hw_queue_id;
232 q->ci = 0;
233 q->pi = 0;
234
235 atomic_set(&q->free_slots_cnt, HL_CQ_LENGTH);
236
237 return 0;
238 }
239
240
241
242
243
244
245
246
247
248 void hl_cq_fini(struct hl_device *hdev, struct hl_cq *q)
249 {
250 hdev->asic_funcs->asic_dma_free_coherent(hdev, HL_CQ_SIZE_IN_BYTES,
251 (void *) (uintptr_t) q->kernel_address, q->bus_address);
252 }
253
254 void hl_cq_reset(struct hl_device *hdev, struct hl_cq *q)
255 {
256 q->ci = 0;
257 q->pi = 0;
258
259 atomic_set(&q->free_slots_cnt, HL_CQ_LENGTH);
260
261
262
263
264
265
266
267
268 memset((void *) (uintptr_t) q->kernel_address, 0, HL_CQ_SIZE_IN_BYTES);
269 }
270
271
272
273
274
275
276
277
278
279
280 int hl_eq_init(struct hl_device *hdev, struct hl_eq *q)
281 {
282 void *p;
283
284 BUILD_BUG_ON(HL_EQ_SIZE_IN_BYTES > HL_PAGE_SIZE);
285
286 p = hdev->asic_funcs->cpu_accessible_dma_pool_alloc(hdev,
287 HL_EQ_SIZE_IN_BYTES,
288 &q->bus_address);
289 if (!p)
290 return -ENOMEM;
291
292 q->hdev = hdev;
293 q->kernel_address = (u64) (uintptr_t) p;
294 q->ci = 0;
295
296 return 0;
297 }
298
299
300
301
302
303
304
305
306
307 void hl_eq_fini(struct hl_device *hdev, struct hl_eq *q)
308 {
309 flush_workqueue(hdev->eq_wq);
310
311 hdev->asic_funcs->cpu_accessible_dma_pool_free(hdev,
312 HL_EQ_SIZE_IN_BYTES,
313 (void *) (uintptr_t) q->kernel_address);
314 }
315
316 void hl_eq_reset(struct hl_device *hdev, struct hl_eq *q)
317 {
318 q->ci = 0;
319
320
321
322
323
324
325
326
327 memset((void *) (uintptr_t) q->kernel_address, 0, HL_EQ_SIZE_IN_BYTES);
328 }