2 * Copyright 2014 Advanced Micro Devices, Inc.
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
24 #include <linux/dma-mapping.h>
27 #include "amdgpu_ih.h"
30 * amdgpu_ih_ring_init - initialize the IH state
32 * @adev: amdgpu_device pointer
33 * @ih: ih ring to initialize
34 * @ring_size: ring size to allocate
35 * @use_bus_addr: true when we can use dma_alloc_coherent
37 * Initializes the IH state and allocates a buffer
38 * for the IH ring buffer.
39 * Returns 0 for success, errors for failure.
41 int amdgpu_ih_ring_init(struct amdgpu_device
*adev
, struct amdgpu_ih_ring
*ih
,
42 unsigned ring_size
, bool use_bus_addr
)
48 rb_bufsz
= order_base_2(ring_size
/ 4);
49 ring_size
= (1 << rb_bufsz
) * 4;
50 ih
->ring_size
= ring_size
;
51 ih
->ptr_mask
= ih
->ring_size
- 1;
53 ih
->use_bus_addr
= use_bus_addr
;
61 /* add 8 bytes for the rptr/wptr shadows and
62 * add them to the end of the ring allocation.
64 ih
->ring
= dma_alloc_coherent(adev
->dev
, ih
->ring_size
+ 8,
65 &dma_addr
, GFP_KERNEL
);
69 ih
->gpu_addr
= dma_addr
;
70 ih
->wptr_addr
= dma_addr
+ ih
->ring_size
;
71 ih
->wptr_cpu
= &ih
->ring
[ih
->ring_size
/ 4];
72 ih
->rptr_addr
= dma_addr
+ ih
->ring_size
+ 4;
73 ih
->rptr_cpu
= &ih
->ring
[(ih
->ring_size
/ 4) + 1];
75 unsigned wptr_offs
, rptr_offs
;
77 r
= amdgpu_device_wb_get(adev
, &wptr_offs
);
81 r
= amdgpu_device_wb_get(adev
, &rptr_offs
);
83 amdgpu_device_wb_free(adev
, wptr_offs
);
87 r
= amdgpu_bo_create_kernel(adev
, ih
->ring_size
, PAGE_SIZE
,
88 AMDGPU_GEM_DOMAIN_GTT
,
89 &ih
->ring_obj
, &ih
->gpu_addr
,
92 amdgpu_device_wb_free(adev
, rptr_offs
);
93 amdgpu_device_wb_free(adev
, wptr_offs
);
97 ih
->wptr_addr
= adev
->wb
.gpu_addr
+ wptr_offs
* 4;
98 ih
->wptr_cpu
= &adev
->wb
.wb
[wptr_offs
];
99 ih
->rptr_addr
= adev
->wb
.gpu_addr
+ rptr_offs
* 4;
100 ih
->rptr_cpu
= &adev
->wb
.wb
[rptr_offs
];
106 * amdgpu_ih_ring_fini - tear down the IH state
108 * @adev: amdgpu_device pointer
109 * @ih: ih ring to tear down
111 * Tears down the IH state and frees buffer
112 * used for the IH ring buffer.
114 void amdgpu_ih_ring_fini(struct amdgpu_device
*adev
, struct amdgpu_ih_ring
*ih
)
116 if (ih
->use_bus_addr
) {
120 /* add 8 bytes for the rptr/wptr shadows and
121 * add them to the end of the ring allocation.
123 dma_free_coherent(adev
->dev
, ih
->ring_size
+ 8,
124 (void *)ih
->ring
, ih
->gpu_addr
);
127 amdgpu_bo_free_kernel(&ih
->ring_obj
, &ih
->gpu_addr
,
129 amdgpu_device_wb_free(adev
, (ih
->wptr_addr
- ih
->gpu_addr
) / 4);
130 amdgpu_device_wb_free(adev
, (ih
->rptr_addr
- ih
->gpu_addr
) / 4);
135 * amdgpu_ih_process - interrupt handler
137 * @adev: amdgpu_device pointer
138 * @ih: ih ring to process
140 * Interrupt hander (VI), walk the IH ring.
141 * Returns irq process return code.
143 int amdgpu_ih_process(struct amdgpu_device
*adev
, struct amdgpu_ih_ring
*ih
)
145 unsigned int count
= AMDGPU_IH_MAX_NUM_IVS
;
148 if (!ih
->enabled
|| adev
->shutdown
)
151 wptr
= amdgpu_ih_get_wptr(adev
, ih
);
154 /* is somebody else already processing irqs? */
155 if (atomic_xchg(&ih
->lock
, 1))
158 DRM_DEBUG("%s: rptr %d, wptr %d\n", __func__
, ih
->rptr
, wptr
);
160 /* Order reading of wptr vs. reading of IH ring data */
163 while (ih
->rptr
!= wptr
&& --count
) {
164 amdgpu_irq_dispatch(adev
, ih
);
165 ih
->rptr
&= ih
->ptr_mask
;
168 amdgpu_ih_set_rptr(adev
, ih
);
169 atomic_set(&ih
->lock
, 0);
171 /* make sure wptr hasn't changed while processing */
172 wptr
= amdgpu_ih_get_wptr(adev
, ih
);
173 if (wptr
!= ih
->rptr
)