blob: 1c85dcb168a1a8870df3ba6d3fa70c9dfba0a993 [file] [log] [blame]
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001/*
2 * Copyright 2008 Advanced Micro Devices, Inc.
3 * Copyright 2008 Red Hat Inc.
4 * Copyright 2009 Jerome Glisse.
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
20 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
21 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
22 * OTHER DEALINGS IN THE SOFTWARE.
23 *
24 * Authors: Dave Airlie
25 * Alex Deucher
26 * Jerome Glisse
27 */
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090028#include <linux/slab.h>
Jerome Glisse3ce0a232009-09-08 10:10:24 +100029#include <linux/seq_file.h>
30#include <linux/firmware.h>
31#include <linux/platform_device.h>
Jerome Glisse771fe6b2009-06-05 14:42:42 +020032#include "drmP.h"
Jerome Glisse3ce0a232009-09-08 10:10:24 +100033#include "radeon_drm.h"
Jerome Glisse771fe6b2009-06-05 14:42:42 +020034#include "radeon.h"
Daniel Vettere6990372010-03-11 21:19:17 +000035#include "radeon_asic.h"
Jerome Glisse3ce0a232009-09-08 10:10:24 +100036#include "radeon_mode.h"
Jerome Glisse3ce0a232009-09-08 10:10:24 +100037#include "r600d.h"
Jerome Glisse3ce0a232009-09-08 10:10:24 +100038#include "atom.h"
Jerome Glissed39c3b82009-09-28 18:34:43 +020039#include "avivod.h"
Jerome Glisse771fe6b2009-06-05 14:42:42 +020040
Jerome Glisse3ce0a232009-09-08 10:10:24 +100041#define PFP_UCODE_SIZE 576
42#define PM4_UCODE_SIZE 1792
Alex Deucherd8f60cf2009-12-01 13:43:46 -050043#define RLC_UCODE_SIZE 768
Jerome Glisse3ce0a232009-09-08 10:10:24 +100044#define R700_PFP_UCODE_SIZE 848
45#define R700_PM4_UCODE_SIZE 1360
Alex Deucherd8f60cf2009-12-01 13:43:46 -050046#define R700_RLC_UCODE_SIZE 1024
Alex Deucherfe251e22010-03-24 13:36:43 -040047#define EVERGREEN_PFP_UCODE_SIZE 1120
48#define EVERGREEN_PM4_UCODE_SIZE 1376
Alex Deucher45f9a392010-03-24 13:55:51 -040049#define EVERGREEN_RLC_UCODE_SIZE 768
Jerome Glisse3ce0a232009-09-08 10:10:24 +100050
51/* Firmware Names */
52MODULE_FIRMWARE("radeon/R600_pfp.bin");
53MODULE_FIRMWARE("radeon/R600_me.bin");
54MODULE_FIRMWARE("radeon/RV610_pfp.bin");
55MODULE_FIRMWARE("radeon/RV610_me.bin");
56MODULE_FIRMWARE("radeon/RV630_pfp.bin");
57MODULE_FIRMWARE("radeon/RV630_me.bin");
58MODULE_FIRMWARE("radeon/RV620_pfp.bin");
59MODULE_FIRMWARE("radeon/RV620_me.bin");
60MODULE_FIRMWARE("radeon/RV635_pfp.bin");
61MODULE_FIRMWARE("radeon/RV635_me.bin");
62MODULE_FIRMWARE("radeon/RV670_pfp.bin");
63MODULE_FIRMWARE("radeon/RV670_me.bin");
64MODULE_FIRMWARE("radeon/RS780_pfp.bin");
65MODULE_FIRMWARE("radeon/RS780_me.bin");
66MODULE_FIRMWARE("radeon/RV770_pfp.bin");
67MODULE_FIRMWARE("radeon/RV770_me.bin");
68MODULE_FIRMWARE("radeon/RV730_pfp.bin");
69MODULE_FIRMWARE("radeon/RV730_me.bin");
70MODULE_FIRMWARE("radeon/RV710_pfp.bin");
71MODULE_FIRMWARE("radeon/RV710_me.bin");
Alex Deucherd8f60cf2009-12-01 13:43:46 -050072MODULE_FIRMWARE("radeon/R600_rlc.bin");
73MODULE_FIRMWARE("radeon/R700_rlc.bin");
Alex Deucherfe251e22010-03-24 13:36:43 -040074MODULE_FIRMWARE("radeon/CEDAR_pfp.bin");
75MODULE_FIRMWARE("radeon/CEDAR_me.bin");
Alex Deucher45f9a392010-03-24 13:55:51 -040076MODULE_FIRMWARE("radeon/CEDAR_rlc.bin");
Alex Deucherfe251e22010-03-24 13:36:43 -040077MODULE_FIRMWARE("radeon/REDWOOD_pfp.bin");
78MODULE_FIRMWARE("radeon/REDWOOD_me.bin");
Alex Deucher45f9a392010-03-24 13:55:51 -040079MODULE_FIRMWARE("radeon/REDWOOD_rlc.bin");
Alex Deucherfe251e22010-03-24 13:36:43 -040080MODULE_FIRMWARE("radeon/JUNIPER_pfp.bin");
81MODULE_FIRMWARE("radeon/JUNIPER_me.bin");
Alex Deucher45f9a392010-03-24 13:55:51 -040082MODULE_FIRMWARE("radeon/JUNIPER_rlc.bin");
Dave Airliea7433742010-04-09 15:31:09 +100083MODULE_FIRMWARE("radeon/CYPRESS_pfp.bin");
Alex Deucherfe251e22010-03-24 13:36:43 -040084MODULE_FIRMWARE("radeon/CYPRESS_me.bin");
Alex Deucher45f9a392010-03-24 13:55:51 -040085MODULE_FIRMWARE("radeon/CYPRESS_rlc.bin");
Jerome Glisse3ce0a232009-09-08 10:10:24 +100086
87int r600_debugfs_mc_info_init(struct radeon_device *rdev);
Jerome Glisse771fe6b2009-06-05 14:42:42 +020088
Jerome Glisse1a029b72009-10-06 19:04:30 +020089/* r600,rv610,rv630,rv620,rv635,rv670 */
Jerome Glisse771fe6b2009-06-05 14:42:42 +020090int r600_mc_wait_for_idle(struct radeon_device *rdev);
91void r600_gpu_init(struct radeon_device *rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +100092void r600_fini(struct radeon_device *rdev);
Alex Deucher45f9a392010-03-24 13:55:51 -040093void r600_irq_disable(struct radeon_device *rdev);
Jerome Glisse771fe6b2009-06-05 14:42:42 +020094
Alex Deucherdef9ba92010-04-22 12:39:58 -040095bool r600_gui_idle(struct radeon_device *rdev)
96{
97 if (RREG32(GRBM_STATUS) & GUI_ACTIVE)
98 return false;
99 else
100 return true;
101}
102
Alex Deuchere0df1ac2009-12-04 15:12:21 -0500103/* hpd for digital panel detect/disconnect */
104bool r600_hpd_sense(struct radeon_device *rdev, enum radeon_hpd_id hpd)
105{
106 bool connected = false;
107
108 if (ASIC_IS_DCE3(rdev)) {
109 switch (hpd) {
110 case RADEON_HPD_1:
111 if (RREG32(DC_HPD1_INT_STATUS) & DC_HPDx_SENSE)
112 connected = true;
113 break;
114 case RADEON_HPD_2:
115 if (RREG32(DC_HPD2_INT_STATUS) & DC_HPDx_SENSE)
116 connected = true;
117 break;
118 case RADEON_HPD_3:
119 if (RREG32(DC_HPD3_INT_STATUS) & DC_HPDx_SENSE)
120 connected = true;
121 break;
122 case RADEON_HPD_4:
123 if (RREG32(DC_HPD4_INT_STATUS) & DC_HPDx_SENSE)
124 connected = true;
125 break;
126 /* DCE 3.2 */
127 case RADEON_HPD_5:
128 if (RREG32(DC_HPD5_INT_STATUS) & DC_HPDx_SENSE)
129 connected = true;
130 break;
131 case RADEON_HPD_6:
132 if (RREG32(DC_HPD6_INT_STATUS) & DC_HPDx_SENSE)
133 connected = true;
134 break;
135 default:
136 break;
137 }
138 } else {
139 switch (hpd) {
140 case RADEON_HPD_1:
141 if (RREG32(DC_HOT_PLUG_DETECT1_INT_STATUS) & DC_HOT_PLUG_DETECTx_SENSE)
142 connected = true;
143 break;
144 case RADEON_HPD_2:
145 if (RREG32(DC_HOT_PLUG_DETECT2_INT_STATUS) & DC_HOT_PLUG_DETECTx_SENSE)
146 connected = true;
147 break;
148 case RADEON_HPD_3:
149 if (RREG32(DC_HOT_PLUG_DETECT3_INT_STATUS) & DC_HOT_PLUG_DETECTx_SENSE)
150 connected = true;
151 break;
152 default:
153 break;
154 }
155 }
156 return connected;
157}
158
159void r600_hpd_set_polarity(struct radeon_device *rdev,
Alex Deucher429770b2009-12-04 15:26:55 -0500160 enum radeon_hpd_id hpd)
Alex Deuchere0df1ac2009-12-04 15:12:21 -0500161{
162 u32 tmp;
163 bool connected = r600_hpd_sense(rdev, hpd);
164
165 if (ASIC_IS_DCE3(rdev)) {
166 switch (hpd) {
167 case RADEON_HPD_1:
168 tmp = RREG32(DC_HPD1_INT_CONTROL);
169 if (connected)
170 tmp &= ~DC_HPDx_INT_POLARITY;
171 else
172 tmp |= DC_HPDx_INT_POLARITY;
173 WREG32(DC_HPD1_INT_CONTROL, tmp);
174 break;
175 case RADEON_HPD_2:
176 tmp = RREG32(DC_HPD2_INT_CONTROL);
177 if (connected)
178 tmp &= ~DC_HPDx_INT_POLARITY;
179 else
180 tmp |= DC_HPDx_INT_POLARITY;
181 WREG32(DC_HPD2_INT_CONTROL, tmp);
182 break;
183 case RADEON_HPD_3:
184 tmp = RREG32(DC_HPD3_INT_CONTROL);
185 if (connected)
186 tmp &= ~DC_HPDx_INT_POLARITY;
187 else
188 tmp |= DC_HPDx_INT_POLARITY;
189 WREG32(DC_HPD3_INT_CONTROL, tmp);
190 break;
191 case RADEON_HPD_4:
192 tmp = RREG32(DC_HPD4_INT_CONTROL);
193 if (connected)
194 tmp &= ~DC_HPDx_INT_POLARITY;
195 else
196 tmp |= DC_HPDx_INT_POLARITY;
197 WREG32(DC_HPD4_INT_CONTROL, tmp);
198 break;
199 case RADEON_HPD_5:
200 tmp = RREG32(DC_HPD5_INT_CONTROL);
201 if (connected)
202 tmp &= ~DC_HPDx_INT_POLARITY;
203 else
204 tmp |= DC_HPDx_INT_POLARITY;
205 WREG32(DC_HPD5_INT_CONTROL, tmp);
206 break;
207 /* DCE 3.2 */
208 case RADEON_HPD_6:
209 tmp = RREG32(DC_HPD6_INT_CONTROL);
210 if (connected)
211 tmp &= ~DC_HPDx_INT_POLARITY;
212 else
213 tmp |= DC_HPDx_INT_POLARITY;
214 WREG32(DC_HPD6_INT_CONTROL, tmp);
215 break;
216 default:
217 break;
218 }
219 } else {
220 switch (hpd) {
221 case RADEON_HPD_1:
222 tmp = RREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL);
223 if (connected)
224 tmp &= ~DC_HOT_PLUG_DETECTx_INT_POLARITY;
225 else
226 tmp |= DC_HOT_PLUG_DETECTx_INT_POLARITY;
227 WREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL, tmp);
228 break;
229 case RADEON_HPD_2:
230 tmp = RREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL);
231 if (connected)
232 tmp &= ~DC_HOT_PLUG_DETECTx_INT_POLARITY;
233 else
234 tmp |= DC_HOT_PLUG_DETECTx_INT_POLARITY;
235 WREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL, tmp);
236 break;
237 case RADEON_HPD_3:
238 tmp = RREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL);
239 if (connected)
240 tmp &= ~DC_HOT_PLUG_DETECTx_INT_POLARITY;
241 else
242 tmp |= DC_HOT_PLUG_DETECTx_INT_POLARITY;
243 WREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL, tmp);
244 break;
245 default:
246 break;
247 }
248 }
249}
250
251void r600_hpd_init(struct radeon_device *rdev)
252{
253 struct drm_device *dev = rdev->ddev;
254 struct drm_connector *connector;
255
256 if (ASIC_IS_DCE3(rdev)) {
257 u32 tmp = DC_HPDx_CONNECTION_TIMER(0x9c4) | DC_HPDx_RX_INT_TIMER(0xfa);
258 if (ASIC_IS_DCE32(rdev))
259 tmp |= DC_HPDx_EN;
260
261 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
262 struct radeon_connector *radeon_connector = to_radeon_connector(connector);
263 switch (radeon_connector->hpd.hpd) {
264 case RADEON_HPD_1:
265 WREG32(DC_HPD1_CONTROL, tmp);
266 rdev->irq.hpd[0] = true;
267 break;
268 case RADEON_HPD_2:
269 WREG32(DC_HPD2_CONTROL, tmp);
270 rdev->irq.hpd[1] = true;
271 break;
272 case RADEON_HPD_3:
273 WREG32(DC_HPD3_CONTROL, tmp);
274 rdev->irq.hpd[2] = true;
275 break;
276 case RADEON_HPD_4:
277 WREG32(DC_HPD4_CONTROL, tmp);
278 rdev->irq.hpd[3] = true;
279 break;
280 /* DCE 3.2 */
281 case RADEON_HPD_5:
282 WREG32(DC_HPD5_CONTROL, tmp);
283 rdev->irq.hpd[4] = true;
284 break;
285 case RADEON_HPD_6:
286 WREG32(DC_HPD6_CONTROL, tmp);
287 rdev->irq.hpd[5] = true;
288 break;
289 default:
290 break;
291 }
292 }
293 } else {
294 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
295 struct radeon_connector *radeon_connector = to_radeon_connector(connector);
296 switch (radeon_connector->hpd.hpd) {
297 case RADEON_HPD_1:
298 WREG32(DC_HOT_PLUG_DETECT1_CONTROL, DC_HOT_PLUG_DETECTx_EN);
299 rdev->irq.hpd[0] = true;
300 break;
301 case RADEON_HPD_2:
302 WREG32(DC_HOT_PLUG_DETECT2_CONTROL, DC_HOT_PLUG_DETECTx_EN);
303 rdev->irq.hpd[1] = true;
304 break;
305 case RADEON_HPD_3:
306 WREG32(DC_HOT_PLUG_DETECT3_CONTROL, DC_HOT_PLUG_DETECTx_EN);
307 rdev->irq.hpd[2] = true;
308 break;
309 default:
310 break;
311 }
312 }
313 }
Jerome Glisse003e69f2010-01-07 15:39:14 +0100314 if (rdev->irq.installed)
315 r600_irq_set(rdev);
Alex Deuchere0df1ac2009-12-04 15:12:21 -0500316}
317
318void r600_hpd_fini(struct radeon_device *rdev)
319{
320 struct drm_device *dev = rdev->ddev;
321 struct drm_connector *connector;
322
323 if (ASIC_IS_DCE3(rdev)) {
324 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
325 struct radeon_connector *radeon_connector = to_radeon_connector(connector);
326 switch (radeon_connector->hpd.hpd) {
327 case RADEON_HPD_1:
328 WREG32(DC_HPD1_CONTROL, 0);
329 rdev->irq.hpd[0] = false;
330 break;
331 case RADEON_HPD_2:
332 WREG32(DC_HPD2_CONTROL, 0);
333 rdev->irq.hpd[1] = false;
334 break;
335 case RADEON_HPD_3:
336 WREG32(DC_HPD3_CONTROL, 0);
337 rdev->irq.hpd[2] = false;
338 break;
339 case RADEON_HPD_4:
340 WREG32(DC_HPD4_CONTROL, 0);
341 rdev->irq.hpd[3] = false;
342 break;
343 /* DCE 3.2 */
344 case RADEON_HPD_5:
345 WREG32(DC_HPD5_CONTROL, 0);
346 rdev->irq.hpd[4] = false;
347 break;
348 case RADEON_HPD_6:
349 WREG32(DC_HPD6_CONTROL, 0);
350 rdev->irq.hpd[5] = false;
351 break;
352 default:
353 break;
354 }
355 }
356 } else {
357 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
358 struct radeon_connector *radeon_connector = to_radeon_connector(connector);
359 switch (radeon_connector->hpd.hpd) {
360 case RADEON_HPD_1:
361 WREG32(DC_HOT_PLUG_DETECT1_CONTROL, 0);
362 rdev->irq.hpd[0] = false;
363 break;
364 case RADEON_HPD_2:
365 WREG32(DC_HOT_PLUG_DETECT2_CONTROL, 0);
366 rdev->irq.hpd[1] = false;
367 break;
368 case RADEON_HPD_3:
369 WREG32(DC_HOT_PLUG_DETECT3_CONTROL, 0);
370 rdev->irq.hpd[2] = false;
371 break;
372 default:
373 break;
374 }
375 }
376 }
377}
378
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200379/*
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000380 * R600 PCIE GART
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200381 */
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000382void r600_pcie_gart_tlb_flush(struct radeon_device *rdev)
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200383{
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000384 unsigned i;
385 u32 tmp;
386
Dave Airlie2e98f102010-02-15 15:54:45 +1000387 /* flush hdp cache so updates hit vram */
388 WREG32(R_005480_HDP_MEM_COHERENCY_FLUSH_CNTL, 0x1);
389
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000390 WREG32(VM_CONTEXT0_INVALIDATION_LOW_ADDR, rdev->mc.gtt_start >> 12);
391 WREG32(VM_CONTEXT0_INVALIDATION_HIGH_ADDR, (rdev->mc.gtt_end - 1) >> 12);
392 WREG32(VM_CONTEXT0_REQUEST_RESPONSE, REQUEST_TYPE(1));
393 for (i = 0; i < rdev->usec_timeout; i++) {
394 /* read MC_STATUS */
395 tmp = RREG32(VM_CONTEXT0_REQUEST_RESPONSE);
396 tmp = (tmp & RESPONSE_TYPE_MASK) >> RESPONSE_TYPE_SHIFT;
397 if (tmp == 2) {
398 printk(KERN_WARNING "[drm] r600 flush TLB failed\n");
399 return;
400 }
401 if (tmp) {
402 return;
403 }
404 udelay(1);
405 }
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200406}
407
Jerome Glisse4aac0472009-09-14 18:29:49 +0200408int r600_pcie_gart_init(struct radeon_device *rdev)
409{
410 int r;
411
412 if (rdev->gart.table.vram.robj) {
413 WARN(1, "R600 PCIE GART already initialized.\n");
414 return 0;
415 }
416 /* Initialize common gart structure */
417 r = radeon_gart_init(rdev);
418 if (r)
419 return r;
420 rdev->gart.table_size = rdev->gart.num_gpu_pages * 8;
421 return radeon_gart_table_vram_alloc(rdev);
422}
423
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000424int r600_pcie_gart_enable(struct radeon_device *rdev)
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200425{
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000426 u32 tmp;
427 int r, i;
428
Jerome Glisse4aac0472009-09-14 18:29:49 +0200429 if (rdev->gart.table.vram.robj == NULL) {
430 dev_err(rdev->dev, "No VRAM object for PCIE GART.\n");
431 return -EINVAL;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000432 }
Jerome Glisse4aac0472009-09-14 18:29:49 +0200433 r = radeon_gart_table_vram_pin(rdev);
434 if (r)
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000435 return r;
Dave Airlie82568562010-02-05 16:00:07 +1000436 radeon_gart_restore(rdev);
Dave Airliebc1a6312009-09-15 11:07:52 +1000437
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000438 /* Setup L2 cache */
439 WREG32(VM_L2_CNTL, ENABLE_L2_CACHE | ENABLE_L2_FRAGMENT_PROCESSING |
440 ENABLE_L2_PTE_CACHE_LRU_UPDATE_BY_WRITE |
441 EFFECTIVE_L2_QUEUE_SIZE(7));
442 WREG32(VM_L2_CNTL2, 0);
443 WREG32(VM_L2_CNTL3, BANK_SELECT_0(0) | BANK_SELECT_1(1));
444 /* Setup TLB control */
445 tmp = ENABLE_L1_TLB | ENABLE_L1_FRAGMENT_PROCESSING |
446 SYSTEM_ACCESS_MODE_NOT_IN_SYS |
447 EFFECTIVE_L1_TLB_SIZE(5) | EFFECTIVE_L1_QUEUE_SIZE(5) |
448 ENABLE_WAIT_L2_QUERY;
449 WREG32(MC_VM_L1_TLB_MCB_RD_SYS_CNTL, tmp);
450 WREG32(MC_VM_L1_TLB_MCB_WR_SYS_CNTL, tmp);
451 WREG32(MC_VM_L1_TLB_MCB_RD_HDP_CNTL, tmp | ENABLE_L1_STRICT_ORDERING);
452 WREG32(MC_VM_L1_TLB_MCB_WR_HDP_CNTL, tmp);
453 WREG32(MC_VM_L1_TLB_MCD_RD_A_CNTL, tmp);
454 WREG32(MC_VM_L1_TLB_MCD_WR_A_CNTL, tmp);
455 WREG32(MC_VM_L1_TLB_MCD_RD_B_CNTL, tmp);
456 WREG32(MC_VM_L1_TLB_MCD_WR_B_CNTL, tmp);
457 WREG32(MC_VM_L1_TLB_MCB_RD_GFX_CNTL, tmp);
458 WREG32(MC_VM_L1_TLB_MCB_WR_GFX_CNTL, tmp);
459 WREG32(MC_VM_L1_TLB_MCB_RD_PDMA_CNTL, tmp);
460 WREG32(MC_VM_L1_TLB_MCB_WR_PDMA_CNTL, tmp);
461 WREG32(MC_VM_L1_TLB_MCB_RD_SEM_CNTL, tmp | ENABLE_SEMAPHORE_MODE);
462 WREG32(MC_VM_L1_TLB_MCB_WR_SEM_CNTL, tmp | ENABLE_SEMAPHORE_MODE);
463 WREG32(VM_CONTEXT0_PAGE_TABLE_START_ADDR, rdev->mc.gtt_start >> 12);
Jerome Glisse1a029b72009-10-06 19:04:30 +0200464 WREG32(VM_CONTEXT0_PAGE_TABLE_END_ADDR, rdev->mc.gtt_end >> 12);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000465 WREG32(VM_CONTEXT0_PAGE_TABLE_BASE_ADDR, rdev->gart.table_addr >> 12);
466 WREG32(VM_CONTEXT0_CNTL, ENABLE_CONTEXT | PAGE_TABLE_DEPTH(0) |
467 RANGE_PROTECTION_FAULT_ENABLE_DEFAULT);
468 WREG32(VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_ADDR,
469 (u32)(rdev->dummy_page.addr >> 12));
470 for (i = 1; i < 7; i++)
471 WREG32(VM_CONTEXT0_CNTL + (i * 4), 0);
472
473 r600_pcie_gart_tlb_flush(rdev);
474 rdev->gart.ready = true;
475 return 0;
476}
477
478void r600_pcie_gart_disable(struct radeon_device *rdev)
479{
480 u32 tmp;
Jerome Glisse4c788672009-11-20 14:29:23 +0100481 int i, r;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000482
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000483 /* Disable all tables */
484 for (i = 0; i < 7; i++)
485 WREG32(VM_CONTEXT0_CNTL + (i * 4), 0);
486
487 /* Disable L2 cache */
488 WREG32(VM_L2_CNTL, ENABLE_L2_FRAGMENT_PROCESSING |
489 EFFECTIVE_L2_QUEUE_SIZE(7));
490 WREG32(VM_L2_CNTL3, BANK_SELECT_0(0) | BANK_SELECT_1(1));
491 /* Setup L1 TLB control */
492 tmp = EFFECTIVE_L1_TLB_SIZE(5) | EFFECTIVE_L1_QUEUE_SIZE(5) |
493 ENABLE_WAIT_L2_QUERY;
494 WREG32(MC_VM_L1_TLB_MCD_RD_A_CNTL, tmp);
495 WREG32(MC_VM_L1_TLB_MCD_WR_A_CNTL, tmp);
496 WREG32(MC_VM_L1_TLB_MCD_RD_B_CNTL, tmp);
497 WREG32(MC_VM_L1_TLB_MCD_WR_B_CNTL, tmp);
498 WREG32(MC_VM_L1_TLB_MCB_RD_GFX_CNTL, tmp);
499 WREG32(MC_VM_L1_TLB_MCB_WR_GFX_CNTL, tmp);
500 WREG32(MC_VM_L1_TLB_MCB_RD_PDMA_CNTL, tmp);
501 WREG32(MC_VM_L1_TLB_MCB_WR_PDMA_CNTL, tmp);
502 WREG32(MC_VM_L1_TLB_MCB_RD_SEM_CNTL, tmp);
503 WREG32(MC_VM_L1_TLB_MCB_WR_SEM_CNTL, tmp);
504 WREG32(MC_VM_L1_TLB_MCB_RD_SYS_CNTL, tmp);
505 WREG32(MC_VM_L1_TLB_MCB_WR_SYS_CNTL, tmp);
506 WREG32(MC_VM_L1_TLB_MCB_RD_HDP_CNTL, tmp);
507 WREG32(MC_VM_L1_TLB_MCB_WR_HDP_CNTL, tmp);
Jerome Glisse4aac0472009-09-14 18:29:49 +0200508 if (rdev->gart.table.vram.robj) {
Jerome Glisse4c788672009-11-20 14:29:23 +0100509 r = radeon_bo_reserve(rdev->gart.table.vram.robj, false);
510 if (likely(r == 0)) {
511 radeon_bo_kunmap(rdev->gart.table.vram.robj);
512 radeon_bo_unpin(rdev->gart.table.vram.robj);
513 radeon_bo_unreserve(rdev->gart.table.vram.robj);
514 }
Jerome Glisse4aac0472009-09-14 18:29:49 +0200515 }
516}
517
518void r600_pcie_gart_fini(struct radeon_device *rdev)
519{
Jerome Glissef9274562010-03-17 14:44:29 +0000520 radeon_gart_fini(rdev);
Jerome Glisse4aac0472009-09-14 18:29:49 +0200521 r600_pcie_gart_disable(rdev);
522 radeon_gart_table_vram_free(rdev);
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200523}
524
Jerome Glisse1a029b72009-10-06 19:04:30 +0200525void r600_agp_enable(struct radeon_device *rdev)
526{
527 u32 tmp;
528 int i;
529
530 /* Setup L2 cache */
531 WREG32(VM_L2_CNTL, ENABLE_L2_CACHE | ENABLE_L2_FRAGMENT_PROCESSING |
532 ENABLE_L2_PTE_CACHE_LRU_UPDATE_BY_WRITE |
533 EFFECTIVE_L2_QUEUE_SIZE(7));
534 WREG32(VM_L2_CNTL2, 0);
535 WREG32(VM_L2_CNTL3, BANK_SELECT_0(0) | BANK_SELECT_1(1));
536 /* Setup TLB control */
537 tmp = ENABLE_L1_TLB | ENABLE_L1_FRAGMENT_PROCESSING |
538 SYSTEM_ACCESS_MODE_NOT_IN_SYS |
539 EFFECTIVE_L1_TLB_SIZE(5) | EFFECTIVE_L1_QUEUE_SIZE(5) |
540 ENABLE_WAIT_L2_QUERY;
541 WREG32(MC_VM_L1_TLB_MCB_RD_SYS_CNTL, tmp);
542 WREG32(MC_VM_L1_TLB_MCB_WR_SYS_CNTL, tmp);
543 WREG32(MC_VM_L1_TLB_MCB_RD_HDP_CNTL, tmp | ENABLE_L1_STRICT_ORDERING);
544 WREG32(MC_VM_L1_TLB_MCB_WR_HDP_CNTL, tmp);
545 WREG32(MC_VM_L1_TLB_MCD_RD_A_CNTL, tmp);
546 WREG32(MC_VM_L1_TLB_MCD_WR_A_CNTL, tmp);
547 WREG32(MC_VM_L1_TLB_MCD_RD_B_CNTL, tmp);
548 WREG32(MC_VM_L1_TLB_MCD_WR_B_CNTL, tmp);
549 WREG32(MC_VM_L1_TLB_MCB_RD_GFX_CNTL, tmp);
550 WREG32(MC_VM_L1_TLB_MCB_WR_GFX_CNTL, tmp);
551 WREG32(MC_VM_L1_TLB_MCB_RD_PDMA_CNTL, tmp);
552 WREG32(MC_VM_L1_TLB_MCB_WR_PDMA_CNTL, tmp);
553 WREG32(MC_VM_L1_TLB_MCB_RD_SEM_CNTL, tmp | ENABLE_SEMAPHORE_MODE);
554 WREG32(MC_VM_L1_TLB_MCB_WR_SEM_CNTL, tmp | ENABLE_SEMAPHORE_MODE);
555 for (i = 0; i < 7; i++)
556 WREG32(VM_CONTEXT0_CNTL + (i * 4), 0);
557}
558
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200559int r600_mc_wait_for_idle(struct radeon_device *rdev)
560{
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000561 unsigned i;
562 u32 tmp;
563
564 for (i = 0; i < rdev->usec_timeout; i++) {
565 /* read MC_STATUS */
566 tmp = RREG32(R_000E50_SRBM_STATUS) & 0x3F00;
567 if (!tmp)
568 return 0;
569 udelay(1);
570 }
571 return -1;
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200572}
573
Jerome Glissea3c19452009-10-01 18:02:13 +0200574static void r600_mc_program(struct radeon_device *rdev)
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200575{
Jerome Glissea3c19452009-10-01 18:02:13 +0200576 struct rv515_mc_save save;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000577 u32 tmp;
578 int i, j;
579
580 /* Initialize HDP */
581 for (i = 0, j = 0; i < 32; i++, j += 0x18) {
582 WREG32((0x2c14 + j), 0x00000000);
583 WREG32((0x2c18 + j), 0x00000000);
584 WREG32((0x2c1c + j), 0x00000000);
585 WREG32((0x2c20 + j), 0x00000000);
586 WREG32((0x2c24 + j), 0x00000000);
587 }
588 WREG32(HDP_REG_COHERENCY_FLUSH_CNTL, 0);
589
Jerome Glissea3c19452009-10-01 18:02:13 +0200590 rv515_mc_stop(rdev, &save);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000591 if (r600_mc_wait_for_idle(rdev)) {
Jerome Glissea3c19452009-10-01 18:02:13 +0200592 dev_warn(rdev->dev, "Wait for MC idle timedout !\n");
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000593 }
Jerome Glissea3c19452009-10-01 18:02:13 +0200594 /* Lockout access through VGA aperture (doesn't exist before R600) */
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000595 WREG32(VGA_HDP_CONTROL, VGA_MEMORY_DISABLE);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000596 /* Update configuration */
Jerome Glisse1a029b72009-10-06 19:04:30 +0200597 if (rdev->flags & RADEON_IS_AGP) {
598 if (rdev->mc.vram_start < rdev->mc.gtt_start) {
599 /* VRAM before AGP */
600 WREG32(MC_VM_SYSTEM_APERTURE_LOW_ADDR,
601 rdev->mc.vram_start >> 12);
602 WREG32(MC_VM_SYSTEM_APERTURE_HIGH_ADDR,
603 rdev->mc.gtt_end >> 12);
604 } else {
605 /* VRAM after AGP */
606 WREG32(MC_VM_SYSTEM_APERTURE_LOW_ADDR,
607 rdev->mc.gtt_start >> 12);
608 WREG32(MC_VM_SYSTEM_APERTURE_HIGH_ADDR,
609 rdev->mc.vram_end >> 12);
610 }
611 } else {
612 WREG32(MC_VM_SYSTEM_APERTURE_LOW_ADDR, rdev->mc.vram_start >> 12);
613 WREG32(MC_VM_SYSTEM_APERTURE_HIGH_ADDR, rdev->mc.vram_end >> 12);
614 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000615 WREG32(MC_VM_SYSTEM_APERTURE_DEFAULT_ADDR, 0);
Jerome Glisse1a029b72009-10-06 19:04:30 +0200616 tmp = ((rdev->mc.vram_end >> 24) & 0xFFFF) << 16;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000617 tmp |= ((rdev->mc.vram_start >> 24) & 0xFFFF);
618 WREG32(MC_VM_FB_LOCATION, tmp);
619 WREG32(HDP_NONSURFACE_BASE, (rdev->mc.vram_start >> 8));
620 WREG32(HDP_NONSURFACE_INFO, (2 << 7));
Jerome Glisse1a029b72009-10-06 19:04:30 +0200621 WREG32(HDP_NONSURFACE_SIZE, rdev->mc.mc_vram_size | 0x3FF);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000622 if (rdev->flags & RADEON_IS_AGP) {
Jerome Glisse1a029b72009-10-06 19:04:30 +0200623 WREG32(MC_VM_AGP_TOP, rdev->mc.gtt_end >> 22);
624 WREG32(MC_VM_AGP_BOT, rdev->mc.gtt_start >> 22);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000625 WREG32(MC_VM_AGP_BASE, rdev->mc.agp_base >> 22);
626 } else {
627 WREG32(MC_VM_AGP_BASE, 0);
628 WREG32(MC_VM_AGP_TOP, 0x0FFFFFFF);
629 WREG32(MC_VM_AGP_BOT, 0x0FFFFFFF);
630 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000631 if (r600_mc_wait_for_idle(rdev)) {
Jerome Glissea3c19452009-10-01 18:02:13 +0200632 dev_warn(rdev->dev, "Wait for MC idle timedout !\n");
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000633 }
Jerome Glissea3c19452009-10-01 18:02:13 +0200634 rv515_mc_resume(rdev, &save);
Dave Airlie698443d2009-09-18 14:16:38 +1000635 /* we need to own VRAM, so turn off the VGA renderer here
636 * to stop it overwriting our objects */
Jerome Glissed39c3b82009-09-28 18:34:43 +0200637 rv515_vga_render_disable(rdev);
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200638}
639
Jerome Glissed594e462010-02-17 21:54:29 +0000640/**
641 * r600_vram_gtt_location - try to find VRAM & GTT location
642 * @rdev: radeon device structure holding all necessary informations
643 * @mc: memory controller structure holding memory informations
644 *
645 * Function will place try to place VRAM at same place as in CPU (PCI)
646 * address space as some GPU seems to have issue when we reprogram at
647 * different address space.
648 *
649 * If there is not enough space to fit the unvisible VRAM after the
650 * aperture then we limit the VRAM size to the aperture.
651 *
652 * If we are using AGP then place VRAM adjacent to AGP aperture are we need
653 * them to be in one from GPU point of view so that we can program GPU to
654 * catch access outside them (weird GPU policy see ??).
655 *
656 * This function will never fails, worst case are limiting VRAM or GTT.
657 *
658 * Note: GTT start, end, size should be initialized before calling this
659 * function on AGP platform.
660 */
661void r600_vram_gtt_location(struct radeon_device *rdev, struct radeon_mc *mc)
662{
663 u64 size_bf, size_af;
664
665 if (mc->mc_vram_size > 0xE0000000) {
666 /* leave room for at least 512M GTT */
667 dev_warn(rdev->dev, "limiting VRAM\n");
668 mc->real_vram_size = 0xE0000000;
669 mc->mc_vram_size = 0xE0000000;
670 }
671 if (rdev->flags & RADEON_IS_AGP) {
672 size_bf = mc->gtt_start;
673 size_af = 0xFFFFFFFF - mc->gtt_end + 1;
674 if (size_bf > size_af) {
675 if (mc->mc_vram_size > size_bf) {
676 dev_warn(rdev->dev, "limiting VRAM\n");
677 mc->real_vram_size = size_bf;
678 mc->mc_vram_size = size_bf;
679 }
680 mc->vram_start = mc->gtt_start - mc->mc_vram_size;
681 } else {
682 if (mc->mc_vram_size > size_af) {
683 dev_warn(rdev->dev, "limiting VRAM\n");
684 mc->real_vram_size = size_af;
685 mc->mc_vram_size = size_af;
686 }
687 mc->vram_start = mc->gtt_end;
688 }
689 mc->vram_end = mc->vram_start + mc->mc_vram_size - 1;
690 dev_info(rdev->dev, "VRAM: %lluM 0x%08llX - 0x%08llX (%lluM used)\n",
691 mc->mc_vram_size >> 20, mc->vram_start,
692 mc->vram_end, mc->real_vram_size >> 20);
693 } else {
694 u64 base = 0;
695 if (rdev->flags & RADEON_IS_IGP)
696 base = (RREG32(MC_VM_FB_LOCATION) & 0xFFFF) << 24;
697 radeon_vram_location(rdev, &rdev->mc, base);
698 radeon_gtt_location(rdev, mc);
699 }
700}
701
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000702int r600_mc_init(struct radeon_device *rdev)
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200703{
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000704 u32 tmp;
Alex Deucher5885b7a2009-10-19 17:23:33 -0400705 int chansize, numchan;
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200706
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000707 /* Get VRAM informations */
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200708 rdev->mc.vram_is_ddr = true;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000709 tmp = RREG32(RAMCFG);
710 if (tmp & CHANSIZE_OVERRIDE) {
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200711 chansize = 16;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000712 } else if (tmp & CHANSIZE_MASK) {
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200713 chansize = 64;
714 } else {
715 chansize = 32;
716 }
Alex Deucher5885b7a2009-10-19 17:23:33 -0400717 tmp = RREG32(CHMAP);
718 switch ((tmp & NOOFCHAN_MASK) >> NOOFCHAN_SHIFT) {
719 case 0:
720 default:
721 numchan = 1;
722 break;
723 case 1:
724 numchan = 2;
725 break;
726 case 2:
727 numchan = 4;
728 break;
729 case 3:
730 numchan = 8;
731 break;
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200732 }
Alex Deucher5885b7a2009-10-19 17:23:33 -0400733 rdev->mc.vram_width = numchan * chansize;
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200734 /* Could aper size report 0 ? */
735 rdev->mc.aper_base = drm_get_resource_start(rdev->ddev, 0);
736 rdev->mc.aper_size = drm_get_resource_len(rdev->ddev, 0);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000737 /* Setup GPU memory space */
738 rdev->mc.mc_vram_size = RREG32(CONFIG_MEMSIZE);
739 rdev->mc.real_vram_size = RREG32(CONFIG_MEMSIZE);
Jerome Glisse51e5fcd2010-02-19 14:33:54 +0000740 rdev->mc.visible_vram_size = rdev->mc.aper_size;
Jerome Glissed594e462010-02-17 21:54:29 +0000741 r600_vram_gtt_location(rdev, &rdev->mc);
Alex Deucherf47299c2010-03-16 20:54:38 -0400742
Alex Deucher06b64762010-01-05 11:27:29 -0500743 if (rdev->flags & RADEON_IS_IGP)
744 rdev->mc.igp_sideport_enabled = radeon_atombios_sideport_present(rdev);
Alex Deucherf47299c2010-03-16 20:54:38 -0400745 radeon_update_bandwidth_info(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000746 return 0;
Jerome Glisse771fe6b2009-06-05 14:42:42 +0200747}
748
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000749/* We doesn't check that the GPU really needs a reset we simply do the
750 * reset, it's up to the caller to determine if the GPU needs one. We
751 * might add an helper function to check that.
752 */
753int r600_gpu_soft_reset(struct radeon_device *rdev)
754{
Jerome Glissea3c19452009-10-01 18:02:13 +0200755 struct rv515_mc_save save;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000756 u32 grbm_busy_mask = S_008010_VC_BUSY(1) | S_008010_VGT_BUSY_NO_DMA(1) |
757 S_008010_VGT_BUSY(1) | S_008010_TA03_BUSY(1) |
758 S_008010_TC_BUSY(1) | S_008010_SX_BUSY(1) |
759 S_008010_SH_BUSY(1) | S_008010_SPI03_BUSY(1) |
760 S_008010_SMX_BUSY(1) | S_008010_SC_BUSY(1) |
761 S_008010_PA_BUSY(1) | S_008010_DB03_BUSY(1) |
762 S_008010_CR_BUSY(1) | S_008010_CB03_BUSY(1) |
763 S_008010_GUI_ACTIVE(1);
764 u32 grbm2_busy_mask = S_008014_SPI0_BUSY(1) | S_008014_SPI1_BUSY(1) |
765 S_008014_SPI2_BUSY(1) | S_008014_SPI3_BUSY(1) |
766 S_008014_TA0_BUSY(1) | S_008014_TA1_BUSY(1) |
767 S_008014_TA2_BUSY(1) | S_008014_TA3_BUSY(1) |
768 S_008014_DB0_BUSY(1) | S_008014_DB1_BUSY(1) |
769 S_008014_DB2_BUSY(1) | S_008014_DB3_BUSY(1) |
770 S_008014_CB0_BUSY(1) | S_008014_CB1_BUSY(1) |
771 S_008014_CB2_BUSY(1) | S_008014_CB3_BUSY(1);
Jerome Glissea3c19452009-10-01 18:02:13 +0200772 u32 tmp;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000773
Jerome Glisse1a029b72009-10-06 19:04:30 +0200774 dev_info(rdev->dev, "GPU softreset \n");
775 dev_info(rdev->dev, " R_008010_GRBM_STATUS=0x%08X\n",
776 RREG32(R_008010_GRBM_STATUS));
777 dev_info(rdev->dev, " R_008014_GRBM_STATUS2=0x%08X\n",
Jerome Glissea3c19452009-10-01 18:02:13 +0200778 RREG32(R_008014_GRBM_STATUS2));
Jerome Glisse1a029b72009-10-06 19:04:30 +0200779 dev_info(rdev->dev, " R_000E50_SRBM_STATUS=0x%08X\n",
780 RREG32(R_000E50_SRBM_STATUS));
Jerome Glissea3c19452009-10-01 18:02:13 +0200781 rv515_mc_stop(rdev, &save);
782 if (r600_mc_wait_for_idle(rdev)) {
783 dev_warn(rdev->dev, "Wait for MC idle timedout !\n");
784 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000785 /* Disable CP parsing/prefetching */
Jerome Glisse90aca4d2010-03-09 14:45:12 +0000786 WREG32(R_0086D8_CP_ME_CNTL, S_0086D8_CP_ME_HALT(1));
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000787 /* Check if any of the rendering block is busy and reset it */
788 if ((RREG32(R_008010_GRBM_STATUS) & grbm_busy_mask) ||
789 (RREG32(R_008014_GRBM_STATUS2) & grbm2_busy_mask)) {
Jerome Glissea3c19452009-10-01 18:02:13 +0200790 tmp = S_008020_SOFT_RESET_CR(1) |
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000791 S_008020_SOFT_RESET_DB(1) |
792 S_008020_SOFT_RESET_CB(1) |
793 S_008020_SOFT_RESET_PA(1) |
794 S_008020_SOFT_RESET_SC(1) |
795 S_008020_SOFT_RESET_SMX(1) |
796 S_008020_SOFT_RESET_SPI(1) |
797 S_008020_SOFT_RESET_SX(1) |
798 S_008020_SOFT_RESET_SH(1) |
799 S_008020_SOFT_RESET_TC(1) |
800 S_008020_SOFT_RESET_TA(1) |
801 S_008020_SOFT_RESET_VC(1) |
Jerome Glissea3c19452009-10-01 18:02:13 +0200802 S_008020_SOFT_RESET_VGT(1);
Jerome Glisse1a029b72009-10-06 19:04:30 +0200803 dev_info(rdev->dev, " R_008020_GRBM_SOFT_RESET=0x%08X\n", tmp);
Jerome Glissea3c19452009-10-01 18:02:13 +0200804 WREG32(R_008020_GRBM_SOFT_RESET, tmp);
Jerome Glisse90aca4d2010-03-09 14:45:12 +0000805 RREG32(R_008020_GRBM_SOFT_RESET);
806 mdelay(15);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000807 WREG32(R_008020_GRBM_SOFT_RESET, 0);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000808 }
809 /* Reset CP (we always reset CP) */
Jerome Glissea3c19452009-10-01 18:02:13 +0200810 tmp = S_008020_SOFT_RESET_CP(1);
811 dev_info(rdev->dev, "R_008020_GRBM_SOFT_RESET=0x%08X\n", tmp);
812 WREG32(R_008020_GRBM_SOFT_RESET, tmp);
Jerome Glisse90aca4d2010-03-09 14:45:12 +0000813 RREG32(R_008020_GRBM_SOFT_RESET);
814 mdelay(15);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000815 WREG32(R_008020_GRBM_SOFT_RESET, 0);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000816 /* Wait a little for things to settle down */
Jerome Glisse225758d2010-03-09 14:45:10 +0000817 mdelay(1);
Jerome Glisse1a029b72009-10-06 19:04:30 +0200818 dev_info(rdev->dev, " R_008010_GRBM_STATUS=0x%08X\n",
819 RREG32(R_008010_GRBM_STATUS));
820 dev_info(rdev->dev, " R_008014_GRBM_STATUS2=0x%08X\n",
821 RREG32(R_008014_GRBM_STATUS2));
822 dev_info(rdev->dev, " R_000E50_SRBM_STATUS=0x%08X\n",
823 RREG32(R_000E50_SRBM_STATUS));
Jerome Glissea3c19452009-10-01 18:02:13 +0200824 rv515_mc_resume(rdev, &save);
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000825 return 0;
826}
827
Jerome Glisse225758d2010-03-09 14:45:10 +0000828bool r600_gpu_is_lockup(struct radeon_device *rdev)
829{
830 u32 srbm_status;
831 u32 grbm_status;
832 u32 grbm_status2;
833 int r;
834
835 srbm_status = RREG32(R_000E50_SRBM_STATUS);
836 grbm_status = RREG32(R_008010_GRBM_STATUS);
837 grbm_status2 = RREG32(R_008014_GRBM_STATUS2);
838 if (!G_008010_GUI_ACTIVE(grbm_status)) {
839 r100_gpu_lockup_update(&rdev->config.r300.lockup, &rdev->cp);
840 return false;
841 }
842 /* force CP activities */
843 r = radeon_ring_lock(rdev, 2);
844 if (!r) {
845 /* PACKET2 NOP */
846 radeon_ring_write(rdev, 0x80000000);
847 radeon_ring_write(rdev, 0x80000000);
848 radeon_ring_unlock_commit(rdev);
849 }
850 rdev->cp.rptr = RREG32(R600_CP_RB_RPTR);
851 return r100_gpu_cp_is_lockup(rdev, &rdev->config.r300.lockup, &rdev->cp);
852}
853
Jerome Glissea2d07b72010-03-09 14:45:11 +0000854int r600_asic_reset(struct radeon_device *rdev)
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000855{
856 return r600_gpu_soft_reset(rdev);
857}
858
859static u32 r600_get_tile_pipe_to_backend_map(u32 num_tile_pipes,
860 u32 num_backends,
861 u32 backend_disable_mask)
862{
863 u32 backend_map = 0;
864 u32 enabled_backends_mask;
865 u32 enabled_backends_count;
866 u32 cur_pipe;
867 u32 swizzle_pipe[R6XX_MAX_PIPES];
868 u32 cur_backend;
869 u32 i;
870
871 if (num_tile_pipes > R6XX_MAX_PIPES)
872 num_tile_pipes = R6XX_MAX_PIPES;
873 if (num_tile_pipes < 1)
874 num_tile_pipes = 1;
875 if (num_backends > R6XX_MAX_BACKENDS)
876 num_backends = R6XX_MAX_BACKENDS;
877 if (num_backends < 1)
878 num_backends = 1;
879
880 enabled_backends_mask = 0;
881 enabled_backends_count = 0;
882 for (i = 0; i < R6XX_MAX_BACKENDS; ++i) {
883 if (((backend_disable_mask >> i) & 1) == 0) {
884 enabled_backends_mask |= (1 << i);
885 ++enabled_backends_count;
886 }
887 if (enabled_backends_count == num_backends)
888 break;
889 }
890
891 if (enabled_backends_count == 0) {
892 enabled_backends_mask = 1;
893 enabled_backends_count = 1;
894 }
895
896 if (enabled_backends_count != num_backends)
897 num_backends = enabled_backends_count;
898
899 memset((uint8_t *)&swizzle_pipe[0], 0, sizeof(u32) * R6XX_MAX_PIPES);
900 switch (num_tile_pipes) {
901 case 1:
902 swizzle_pipe[0] = 0;
903 break;
904 case 2:
905 swizzle_pipe[0] = 0;
906 swizzle_pipe[1] = 1;
907 break;
908 case 3:
909 swizzle_pipe[0] = 0;
910 swizzle_pipe[1] = 1;
911 swizzle_pipe[2] = 2;
912 break;
913 case 4:
914 swizzle_pipe[0] = 0;
915 swizzle_pipe[1] = 1;
916 swizzle_pipe[2] = 2;
917 swizzle_pipe[3] = 3;
918 break;
919 case 5:
920 swizzle_pipe[0] = 0;
921 swizzle_pipe[1] = 1;
922 swizzle_pipe[2] = 2;
923 swizzle_pipe[3] = 3;
924 swizzle_pipe[4] = 4;
925 break;
926 case 6:
927 swizzle_pipe[0] = 0;
928 swizzle_pipe[1] = 2;
929 swizzle_pipe[2] = 4;
930 swizzle_pipe[3] = 5;
931 swizzle_pipe[4] = 1;
932 swizzle_pipe[5] = 3;
933 break;
934 case 7:
935 swizzle_pipe[0] = 0;
936 swizzle_pipe[1] = 2;
937 swizzle_pipe[2] = 4;
938 swizzle_pipe[3] = 6;
939 swizzle_pipe[4] = 1;
940 swizzle_pipe[5] = 3;
941 swizzle_pipe[6] = 5;
942 break;
943 case 8:
944 swizzle_pipe[0] = 0;
945 swizzle_pipe[1] = 2;
946 swizzle_pipe[2] = 4;
947 swizzle_pipe[3] = 6;
948 swizzle_pipe[4] = 1;
949 swizzle_pipe[5] = 3;
950 swizzle_pipe[6] = 5;
951 swizzle_pipe[7] = 7;
952 break;
953 }
954
955 cur_backend = 0;
956 for (cur_pipe = 0; cur_pipe < num_tile_pipes; ++cur_pipe) {
957 while (((1 << cur_backend) & enabled_backends_mask) == 0)
958 cur_backend = (cur_backend + 1) % R6XX_MAX_BACKENDS;
959
960 backend_map |= (u32)(((cur_backend & 3) << (swizzle_pipe[cur_pipe] * 2)));
961
962 cur_backend = (cur_backend + 1) % R6XX_MAX_BACKENDS;
963 }
964
965 return backend_map;
966}
967
968int r600_count_pipe_bits(uint32_t val)
969{
970 int i, ret = 0;
971
972 for (i = 0; i < 32; i++) {
973 ret += val & 1;
974 val >>= 1;
975 }
976 return ret;
977}
978
979void r600_gpu_init(struct radeon_device *rdev)
980{
981 u32 tiling_config;
982 u32 ramcfg;
Alex Deucherd03f5d52010-02-19 16:22:31 -0500983 u32 backend_map;
984 u32 cc_rb_backend_disable;
985 u32 cc_gc_shader_pipe_config;
Jerome Glisse3ce0a232009-09-08 10:10:24 +1000986 u32 tmp;
987 int i, j;
988 u32 sq_config;
989 u32 sq_gpr_resource_mgmt_1 = 0;
990 u32 sq_gpr_resource_mgmt_2 = 0;
991 u32 sq_thread_resource_mgmt = 0;
992 u32 sq_stack_resource_mgmt_1 = 0;
993 u32 sq_stack_resource_mgmt_2 = 0;
994
995 /* FIXME: implement */
996 switch (rdev->family) {
997 case CHIP_R600:
998 rdev->config.r600.max_pipes = 4;
999 rdev->config.r600.max_tile_pipes = 8;
1000 rdev->config.r600.max_simds = 4;
1001 rdev->config.r600.max_backends = 4;
1002 rdev->config.r600.max_gprs = 256;
1003 rdev->config.r600.max_threads = 192;
1004 rdev->config.r600.max_stack_entries = 256;
1005 rdev->config.r600.max_hw_contexts = 8;
1006 rdev->config.r600.max_gs_threads = 16;
1007 rdev->config.r600.sx_max_export_size = 128;
1008 rdev->config.r600.sx_max_export_pos_size = 16;
1009 rdev->config.r600.sx_max_export_smx_size = 128;
1010 rdev->config.r600.sq_num_cf_insts = 2;
1011 break;
1012 case CHIP_RV630:
1013 case CHIP_RV635:
1014 rdev->config.r600.max_pipes = 2;
1015 rdev->config.r600.max_tile_pipes = 2;
1016 rdev->config.r600.max_simds = 3;
1017 rdev->config.r600.max_backends = 1;
1018 rdev->config.r600.max_gprs = 128;
1019 rdev->config.r600.max_threads = 192;
1020 rdev->config.r600.max_stack_entries = 128;
1021 rdev->config.r600.max_hw_contexts = 8;
1022 rdev->config.r600.max_gs_threads = 4;
1023 rdev->config.r600.sx_max_export_size = 128;
1024 rdev->config.r600.sx_max_export_pos_size = 16;
1025 rdev->config.r600.sx_max_export_smx_size = 128;
1026 rdev->config.r600.sq_num_cf_insts = 2;
1027 break;
1028 case CHIP_RV610:
1029 case CHIP_RV620:
1030 case CHIP_RS780:
1031 case CHIP_RS880:
1032 rdev->config.r600.max_pipes = 1;
1033 rdev->config.r600.max_tile_pipes = 1;
1034 rdev->config.r600.max_simds = 2;
1035 rdev->config.r600.max_backends = 1;
1036 rdev->config.r600.max_gprs = 128;
1037 rdev->config.r600.max_threads = 192;
1038 rdev->config.r600.max_stack_entries = 128;
1039 rdev->config.r600.max_hw_contexts = 4;
1040 rdev->config.r600.max_gs_threads = 4;
1041 rdev->config.r600.sx_max_export_size = 128;
1042 rdev->config.r600.sx_max_export_pos_size = 16;
1043 rdev->config.r600.sx_max_export_smx_size = 128;
1044 rdev->config.r600.sq_num_cf_insts = 1;
1045 break;
1046 case CHIP_RV670:
1047 rdev->config.r600.max_pipes = 4;
1048 rdev->config.r600.max_tile_pipes = 4;
1049 rdev->config.r600.max_simds = 4;
1050 rdev->config.r600.max_backends = 4;
1051 rdev->config.r600.max_gprs = 192;
1052 rdev->config.r600.max_threads = 192;
1053 rdev->config.r600.max_stack_entries = 256;
1054 rdev->config.r600.max_hw_contexts = 8;
1055 rdev->config.r600.max_gs_threads = 16;
1056 rdev->config.r600.sx_max_export_size = 128;
1057 rdev->config.r600.sx_max_export_pos_size = 16;
1058 rdev->config.r600.sx_max_export_smx_size = 128;
1059 rdev->config.r600.sq_num_cf_insts = 2;
1060 break;
1061 default:
1062 break;
1063 }
1064
1065 /* Initialize HDP */
1066 for (i = 0, j = 0; i < 32; i++, j += 0x18) {
1067 WREG32((0x2c14 + j), 0x00000000);
1068 WREG32((0x2c18 + j), 0x00000000);
1069 WREG32((0x2c1c + j), 0x00000000);
1070 WREG32((0x2c20 + j), 0x00000000);
1071 WREG32((0x2c24 + j), 0x00000000);
1072 }
1073
1074 WREG32(GRBM_CNTL, GRBM_READ_TIMEOUT(0xff));
1075
1076 /* Setup tiling */
1077 tiling_config = 0;
1078 ramcfg = RREG32(RAMCFG);
1079 switch (rdev->config.r600.max_tile_pipes) {
1080 case 1:
1081 tiling_config |= PIPE_TILING(0);
1082 break;
1083 case 2:
1084 tiling_config |= PIPE_TILING(1);
1085 break;
1086 case 4:
1087 tiling_config |= PIPE_TILING(2);
1088 break;
1089 case 8:
1090 tiling_config |= PIPE_TILING(3);
1091 break;
1092 default:
1093 break;
1094 }
Alex Deucherd03f5d52010-02-19 16:22:31 -05001095 rdev->config.r600.tiling_npipes = rdev->config.r600.max_tile_pipes;
Jerome Glisse961fb592010-02-10 22:30:05 +00001096 rdev->config.r600.tiling_nbanks = 4 << ((ramcfg & NOOFBANK_MASK) >> NOOFBANK_SHIFT);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001097 tiling_config |= BANK_TILING((ramcfg & NOOFBANK_MASK) >> NOOFBANK_SHIFT);
1098 tiling_config |= GROUP_SIZE(0);
Jerome Glisse961fb592010-02-10 22:30:05 +00001099 rdev->config.r600.tiling_group_size = 256;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001100 tmp = (ramcfg & NOOFROWS_MASK) >> NOOFROWS_SHIFT;
1101 if (tmp > 3) {
1102 tiling_config |= ROW_TILING(3);
1103 tiling_config |= SAMPLE_SPLIT(3);
1104 } else {
1105 tiling_config |= ROW_TILING(tmp);
1106 tiling_config |= SAMPLE_SPLIT(tmp);
1107 }
1108 tiling_config |= BANK_SWAPS(1);
Alex Deucherd03f5d52010-02-19 16:22:31 -05001109
1110 cc_rb_backend_disable = RREG32(CC_RB_BACKEND_DISABLE) & 0x00ff0000;
1111 cc_rb_backend_disable |=
1112 BACKEND_DISABLE((R6XX_MAX_BACKENDS_MASK << rdev->config.r600.max_backends) & R6XX_MAX_BACKENDS_MASK);
1113
1114 cc_gc_shader_pipe_config = RREG32(CC_GC_SHADER_PIPE_CONFIG) & 0xffffff00;
1115 cc_gc_shader_pipe_config |=
1116 INACTIVE_QD_PIPES((R6XX_MAX_PIPES_MASK << rdev->config.r600.max_pipes) & R6XX_MAX_PIPES_MASK);
1117 cc_gc_shader_pipe_config |=
1118 INACTIVE_SIMDS((R6XX_MAX_SIMDS_MASK << rdev->config.r600.max_simds) & R6XX_MAX_SIMDS_MASK);
1119
1120 backend_map = r600_get_tile_pipe_to_backend_map(rdev->config.r600.max_tile_pipes,
1121 (R6XX_MAX_BACKENDS -
1122 r600_count_pipe_bits((cc_rb_backend_disable &
1123 R6XX_MAX_BACKENDS_MASK) >> 16)),
1124 (cc_rb_backend_disable >> 16));
1125
1126 tiling_config |= BACKEND_MAP(backend_map);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001127 WREG32(GB_TILING_CONFIG, tiling_config);
1128 WREG32(DCP_TILING_CONFIG, tiling_config & 0xffff);
1129 WREG32(HDP_TILING_CONFIG, tiling_config & 0xffff);
1130
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001131 /* Setup pipes */
Alex Deucherd03f5d52010-02-19 16:22:31 -05001132 WREG32(CC_RB_BACKEND_DISABLE, cc_rb_backend_disable);
1133 WREG32(CC_GC_SHADER_PIPE_CONFIG, cc_gc_shader_pipe_config);
Alex Deucherf867c60d2010-03-05 14:50:37 -05001134 WREG32(GC_USER_SHADER_PIPE_CONFIG, cc_gc_shader_pipe_config);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001135
Alex Deucherd03f5d52010-02-19 16:22:31 -05001136 tmp = R6XX_MAX_PIPES - r600_count_pipe_bits((cc_gc_shader_pipe_config & INACTIVE_QD_PIPES_MASK) >> 8);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001137 WREG32(VGT_OUT_DEALLOC_CNTL, (tmp * 4) & DEALLOC_DIST_MASK);
1138 WREG32(VGT_VERTEX_REUSE_BLOCK_CNTL, ((tmp * 4) - 2) & VTX_REUSE_DEPTH_MASK);
1139
1140 /* Setup some CP states */
1141 WREG32(CP_QUEUE_THRESHOLDS, (ROQ_IB1_START(0x16) | ROQ_IB2_START(0x2b)));
1142 WREG32(CP_MEQ_THRESHOLDS, (MEQ_END(0x40) | ROQ_END(0x40)));
1143
1144 WREG32(TA_CNTL_AUX, (DISABLE_CUBE_ANISO | SYNC_GRADIENT |
1145 SYNC_WALKER | SYNC_ALIGNER));
1146 /* Setup various GPU states */
1147 if (rdev->family == CHIP_RV670)
1148 WREG32(ARB_GDEC_RD_CNTL, 0x00000021);
1149
1150 tmp = RREG32(SX_DEBUG_1);
1151 tmp |= SMX_EVENT_RELEASE;
1152 if ((rdev->family > CHIP_R600))
1153 tmp |= ENABLE_NEW_SMX_ADDRESS;
1154 WREG32(SX_DEBUG_1, tmp);
1155
1156 if (((rdev->family) == CHIP_R600) ||
1157 ((rdev->family) == CHIP_RV630) ||
1158 ((rdev->family) == CHIP_RV610) ||
1159 ((rdev->family) == CHIP_RV620) ||
Alex Deucheree59f2b2009-11-05 13:11:46 -05001160 ((rdev->family) == CHIP_RS780) ||
1161 ((rdev->family) == CHIP_RS880)) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001162 WREG32(DB_DEBUG, PREZ_MUST_WAIT_FOR_POSTZ_DONE);
1163 } else {
1164 WREG32(DB_DEBUG, 0);
1165 }
1166 WREG32(DB_WATERMARKS, (DEPTH_FREE(4) | DEPTH_CACHELINE_FREE(16) |
1167 DEPTH_FLUSH(16) | DEPTH_PENDING_FREE(4)));
1168
1169 WREG32(PA_SC_MULTI_CHIP_CNTL, 0);
1170 WREG32(VGT_NUM_INSTANCES, 0);
1171
1172 WREG32(SPI_CONFIG_CNTL, GPR_WRITE_PRIORITY(0));
1173 WREG32(SPI_CONFIG_CNTL_1, VTX_DONE_DELAY(0));
1174
1175 tmp = RREG32(SQ_MS_FIFO_SIZES);
1176 if (((rdev->family) == CHIP_RV610) ||
1177 ((rdev->family) == CHIP_RV620) ||
Alex Deucheree59f2b2009-11-05 13:11:46 -05001178 ((rdev->family) == CHIP_RS780) ||
1179 ((rdev->family) == CHIP_RS880)) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001180 tmp = (CACHE_FIFO_SIZE(0xa) |
1181 FETCH_FIFO_HIWATER(0xa) |
1182 DONE_FIFO_HIWATER(0xe0) |
1183 ALU_UPDATE_FIFO_HIWATER(0x8));
1184 } else if (((rdev->family) == CHIP_R600) ||
1185 ((rdev->family) == CHIP_RV630)) {
1186 tmp &= ~DONE_FIFO_HIWATER(0xff);
1187 tmp |= DONE_FIFO_HIWATER(0x4);
1188 }
1189 WREG32(SQ_MS_FIFO_SIZES, tmp);
1190
1191 /* SQ_CONFIG, SQ_GPR_RESOURCE_MGMT, SQ_THREAD_RESOURCE_MGMT, SQ_STACK_RESOURCE_MGMT
1192 * should be adjusted as needed by the 2D/3D drivers. This just sets default values
1193 */
1194 sq_config = RREG32(SQ_CONFIG);
1195 sq_config &= ~(PS_PRIO(3) |
1196 VS_PRIO(3) |
1197 GS_PRIO(3) |
1198 ES_PRIO(3));
1199 sq_config |= (DX9_CONSTS |
1200 VC_ENABLE |
1201 PS_PRIO(0) |
1202 VS_PRIO(1) |
1203 GS_PRIO(2) |
1204 ES_PRIO(3));
1205
1206 if ((rdev->family) == CHIP_R600) {
1207 sq_gpr_resource_mgmt_1 = (NUM_PS_GPRS(124) |
1208 NUM_VS_GPRS(124) |
1209 NUM_CLAUSE_TEMP_GPRS(4));
1210 sq_gpr_resource_mgmt_2 = (NUM_GS_GPRS(0) |
1211 NUM_ES_GPRS(0));
1212 sq_thread_resource_mgmt = (NUM_PS_THREADS(136) |
1213 NUM_VS_THREADS(48) |
1214 NUM_GS_THREADS(4) |
1215 NUM_ES_THREADS(4));
1216 sq_stack_resource_mgmt_1 = (NUM_PS_STACK_ENTRIES(128) |
1217 NUM_VS_STACK_ENTRIES(128));
1218 sq_stack_resource_mgmt_2 = (NUM_GS_STACK_ENTRIES(0) |
1219 NUM_ES_STACK_ENTRIES(0));
1220 } else if (((rdev->family) == CHIP_RV610) ||
1221 ((rdev->family) == CHIP_RV620) ||
Alex Deucheree59f2b2009-11-05 13:11:46 -05001222 ((rdev->family) == CHIP_RS780) ||
1223 ((rdev->family) == CHIP_RS880)) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001224 /* no vertex cache */
1225 sq_config &= ~VC_ENABLE;
1226
1227 sq_gpr_resource_mgmt_1 = (NUM_PS_GPRS(44) |
1228 NUM_VS_GPRS(44) |
1229 NUM_CLAUSE_TEMP_GPRS(2));
1230 sq_gpr_resource_mgmt_2 = (NUM_GS_GPRS(17) |
1231 NUM_ES_GPRS(17));
1232 sq_thread_resource_mgmt = (NUM_PS_THREADS(79) |
1233 NUM_VS_THREADS(78) |
1234 NUM_GS_THREADS(4) |
1235 NUM_ES_THREADS(31));
1236 sq_stack_resource_mgmt_1 = (NUM_PS_STACK_ENTRIES(40) |
1237 NUM_VS_STACK_ENTRIES(40));
1238 sq_stack_resource_mgmt_2 = (NUM_GS_STACK_ENTRIES(32) |
1239 NUM_ES_STACK_ENTRIES(16));
1240 } else if (((rdev->family) == CHIP_RV630) ||
1241 ((rdev->family) == CHIP_RV635)) {
1242 sq_gpr_resource_mgmt_1 = (NUM_PS_GPRS(44) |
1243 NUM_VS_GPRS(44) |
1244 NUM_CLAUSE_TEMP_GPRS(2));
1245 sq_gpr_resource_mgmt_2 = (NUM_GS_GPRS(18) |
1246 NUM_ES_GPRS(18));
1247 sq_thread_resource_mgmt = (NUM_PS_THREADS(79) |
1248 NUM_VS_THREADS(78) |
1249 NUM_GS_THREADS(4) |
1250 NUM_ES_THREADS(31));
1251 sq_stack_resource_mgmt_1 = (NUM_PS_STACK_ENTRIES(40) |
1252 NUM_VS_STACK_ENTRIES(40));
1253 sq_stack_resource_mgmt_2 = (NUM_GS_STACK_ENTRIES(32) |
1254 NUM_ES_STACK_ENTRIES(16));
1255 } else if ((rdev->family) == CHIP_RV670) {
1256 sq_gpr_resource_mgmt_1 = (NUM_PS_GPRS(44) |
1257 NUM_VS_GPRS(44) |
1258 NUM_CLAUSE_TEMP_GPRS(2));
1259 sq_gpr_resource_mgmt_2 = (NUM_GS_GPRS(17) |
1260 NUM_ES_GPRS(17));
1261 sq_thread_resource_mgmt = (NUM_PS_THREADS(79) |
1262 NUM_VS_THREADS(78) |
1263 NUM_GS_THREADS(4) |
1264 NUM_ES_THREADS(31));
1265 sq_stack_resource_mgmt_1 = (NUM_PS_STACK_ENTRIES(64) |
1266 NUM_VS_STACK_ENTRIES(64));
1267 sq_stack_resource_mgmt_2 = (NUM_GS_STACK_ENTRIES(64) |
1268 NUM_ES_STACK_ENTRIES(64));
1269 }
1270
1271 WREG32(SQ_CONFIG, sq_config);
1272 WREG32(SQ_GPR_RESOURCE_MGMT_1, sq_gpr_resource_mgmt_1);
1273 WREG32(SQ_GPR_RESOURCE_MGMT_2, sq_gpr_resource_mgmt_2);
1274 WREG32(SQ_THREAD_RESOURCE_MGMT, sq_thread_resource_mgmt);
1275 WREG32(SQ_STACK_RESOURCE_MGMT_1, sq_stack_resource_mgmt_1);
1276 WREG32(SQ_STACK_RESOURCE_MGMT_2, sq_stack_resource_mgmt_2);
1277
1278 if (((rdev->family) == CHIP_RV610) ||
1279 ((rdev->family) == CHIP_RV620) ||
Alex Deucheree59f2b2009-11-05 13:11:46 -05001280 ((rdev->family) == CHIP_RS780) ||
1281 ((rdev->family) == CHIP_RS880)) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001282 WREG32(VGT_CACHE_INVALIDATION, CACHE_INVALIDATION(TC_ONLY));
1283 } else {
1284 WREG32(VGT_CACHE_INVALIDATION, CACHE_INVALIDATION(VC_AND_TC));
1285 }
1286
1287 /* More default values. 2D/3D driver should adjust as needed */
1288 WREG32(PA_SC_AA_SAMPLE_LOCS_2S, (S0_X(0xc) | S0_Y(0x4) |
1289 S1_X(0x4) | S1_Y(0xc)));
1290 WREG32(PA_SC_AA_SAMPLE_LOCS_4S, (S0_X(0xe) | S0_Y(0xe) |
1291 S1_X(0x2) | S1_Y(0x2) |
1292 S2_X(0xa) | S2_Y(0x6) |
1293 S3_X(0x6) | S3_Y(0xa)));
1294 WREG32(PA_SC_AA_SAMPLE_LOCS_8S_WD0, (S0_X(0xe) | S0_Y(0xb) |
1295 S1_X(0x4) | S1_Y(0xc) |
1296 S2_X(0x1) | S2_Y(0x6) |
1297 S3_X(0xa) | S3_Y(0xe)));
1298 WREG32(PA_SC_AA_SAMPLE_LOCS_8S_WD1, (S4_X(0x6) | S4_Y(0x1) |
1299 S5_X(0x0) | S5_Y(0x0) |
1300 S6_X(0xb) | S6_Y(0x4) |
1301 S7_X(0x7) | S7_Y(0x8)));
1302
1303 WREG32(VGT_STRMOUT_EN, 0);
1304 tmp = rdev->config.r600.max_pipes * 16;
1305 switch (rdev->family) {
1306 case CHIP_RV610:
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001307 case CHIP_RV620:
Alex Deucheree59f2b2009-11-05 13:11:46 -05001308 case CHIP_RS780:
1309 case CHIP_RS880:
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001310 tmp += 32;
1311 break;
1312 case CHIP_RV670:
1313 tmp += 128;
1314 break;
1315 default:
1316 break;
1317 }
1318 if (tmp > 256) {
1319 tmp = 256;
1320 }
1321 WREG32(VGT_ES_PER_GS, 128);
1322 WREG32(VGT_GS_PER_ES, tmp);
1323 WREG32(VGT_GS_PER_VS, 2);
1324 WREG32(VGT_GS_VERTEX_REUSE, 16);
1325
1326 /* more default values. 2D/3D driver should adjust as needed */
1327 WREG32(PA_SC_LINE_STIPPLE_STATE, 0);
1328 WREG32(VGT_STRMOUT_EN, 0);
1329 WREG32(SX_MISC, 0);
1330 WREG32(PA_SC_MODE_CNTL, 0);
1331 WREG32(PA_SC_AA_CONFIG, 0);
1332 WREG32(PA_SC_LINE_STIPPLE, 0);
1333 WREG32(SPI_INPUT_Z, 0);
1334 WREG32(SPI_PS_IN_CONTROL_0, NUM_INTERP(2));
1335 WREG32(CB_COLOR7_FRAG, 0);
1336
1337 /* Clear render buffer base addresses */
1338 WREG32(CB_COLOR0_BASE, 0);
1339 WREG32(CB_COLOR1_BASE, 0);
1340 WREG32(CB_COLOR2_BASE, 0);
1341 WREG32(CB_COLOR3_BASE, 0);
1342 WREG32(CB_COLOR4_BASE, 0);
1343 WREG32(CB_COLOR5_BASE, 0);
1344 WREG32(CB_COLOR6_BASE, 0);
1345 WREG32(CB_COLOR7_BASE, 0);
1346 WREG32(CB_COLOR7_FRAG, 0);
1347
1348 switch (rdev->family) {
1349 case CHIP_RV610:
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001350 case CHIP_RV620:
Alex Deucheree59f2b2009-11-05 13:11:46 -05001351 case CHIP_RS780:
1352 case CHIP_RS880:
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001353 tmp = TC_L2_SIZE(8);
1354 break;
1355 case CHIP_RV630:
1356 case CHIP_RV635:
1357 tmp = TC_L2_SIZE(4);
1358 break;
1359 case CHIP_R600:
1360 tmp = TC_L2_SIZE(0) | L2_DISABLE_LATE_HIT;
1361 break;
1362 default:
1363 tmp = TC_L2_SIZE(0);
1364 break;
1365 }
1366 WREG32(TC_CNTL, tmp);
1367
1368 tmp = RREG32(HDP_HOST_PATH_CNTL);
1369 WREG32(HDP_HOST_PATH_CNTL, tmp);
1370
1371 tmp = RREG32(ARB_POP);
1372 tmp |= ENABLE_TC128;
1373 WREG32(ARB_POP, tmp);
1374
1375 WREG32(PA_SC_MULTI_CHIP_CNTL, 0);
1376 WREG32(PA_CL_ENHANCE, (CLIP_VTX_REORDER_ENA |
1377 NUM_CLIP_SEQ(3)));
1378 WREG32(PA_SC_ENHANCE, FORCE_EOV_MAX_CLK_CNT(4095));
1379}
1380
1381
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001382/*
1383 * Indirect registers accessor
1384 */
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001385u32 r600_pciep_rreg(struct radeon_device *rdev, u32 reg)
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001386{
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001387 u32 r;
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001388
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001389 WREG32(PCIE_PORT_INDEX, ((reg) & 0xff));
1390 (void)RREG32(PCIE_PORT_INDEX);
1391 r = RREG32(PCIE_PORT_DATA);
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001392 return r;
1393}
1394
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001395void r600_pciep_wreg(struct radeon_device *rdev, u32 reg, u32 v)
Jerome Glisse771fe6b2009-06-05 14:42:42 +02001396{
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001397 WREG32(PCIE_PORT_INDEX, ((reg) & 0xff));
1398 (void)RREG32(PCIE_PORT_INDEX);
1399 WREG32(PCIE_PORT_DATA, (v));
1400 (void)RREG32(PCIE_PORT_DATA);
1401}
1402
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001403/*
1404 * CP & Ring
1405 */
1406void r600_cp_stop(struct radeon_device *rdev)
1407{
1408 WREG32(R_0086D8_CP_ME_CNTL, S_0086D8_CP_ME_HALT(1));
1409}
1410
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001411int r600_init_microcode(struct radeon_device *rdev)
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001412{
1413 struct platform_device *pdev;
1414 const char *chip_name;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001415 const char *rlc_chip_name;
1416 size_t pfp_req_size, me_req_size, rlc_req_size;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001417 char fw_name[30];
1418 int err;
1419
1420 DRM_DEBUG("\n");
1421
1422 pdev = platform_device_register_simple("radeon_cp", 0, NULL, 0);
1423 err = IS_ERR(pdev);
1424 if (err) {
1425 printk(KERN_ERR "radeon_cp: Failed to register firmware\n");
1426 return -EINVAL;
1427 }
1428
1429 switch (rdev->family) {
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001430 case CHIP_R600:
1431 chip_name = "R600";
1432 rlc_chip_name = "R600";
1433 break;
1434 case CHIP_RV610:
1435 chip_name = "RV610";
1436 rlc_chip_name = "R600";
1437 break;
1438 case CHIP_RV630:
1439 chip_name = "RV630";
1440 rlc_chip_name = "R600";
1441 break;
1442 case CHIP_RV620:
1443 chip_name = "RV620";
1444 rlc_chip_name = "R600";
1445 break;
1446 case CHIP_RV635:
1447 chip_name = "RV635";
1448 rlc_chip_name = "R600";
1449 break;
1450 case CHIP_RV670:
1451 chip_name = "RV670";
1452 rlc_chip_name = "R600";
1453 break;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001454 case CHIP_RS780:
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001455 case CHIP_RS880:
1456 chip_name = "RS780";
1457 rlc_chip_name = "R600";
1458 break;
1459 case CHIP_RV770:
1460 chip_name = "RV770";
1461 rlc_chip_name = "R700";
1462 break;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001463 case CHIP_RV730:
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001464 case CHIP_RV740:
1465 chip_name = "RV730";
1466 rlc_chip_name = "R700";
1467 break;
1468 case CHIP_RV710:
1469 chip_name = "RV710";
1470 rlc_chip_name = "R700";
1471 break;
Alex Deucherfe251e22010-03-24 13:36:43 -04001472 case CHIP_CEDAR:
1473 chip_name = "CEDAR";
Alex Deucher45f9a392010-03-24 13:55:51 -04001474 rlc_chip_name = "CEDAR";
Alex Deucherfe251e22010-03-24 13:36:43 -04001475 break;
1476 case CHIP_REDWOOD:
1477 chip_name = "REDWOOD";
Alex Deucher45f9a392010-03-24 13:55:51 -04001478 rlc_chip_name = "REDWOOD";
Alex Deucherfe251e22010-03-24 13:36:43 -04001479 break;
1480 case CHIP_JUNIPER:
1481 chip_name = "JUNIPER";
Alex Deucher45f9a392010-03-24 13:55:51 -04001482 rlc_chip_name = "JUNIPER";
Alex Deucherfe251e22010-03-24 13:36:43 -04001483 break;
1484 case CHIP_CYPRESS:
1485 case CHIP_HEMLOCK:
1486 chip_name = "CYPRESS";
Alex Deucher45f9a392010-03-24 13:55:51 -04001487 rlc_chip_name = "CYPRESS";
Alex Deucherfe251e22010-03-24 13:36:43 -04001488 break;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001489 default: BUG();
1490 }
1491
Alex Deucherfe251e22010-03-24 13:36:43 -04001492 if (rdev->family >= CHIP_CEDAR) {
1493 pfp_req_size = EVERGREEN_PFP_UCODE_SIZE * 4;
1494 me_req_size = EVERGREEN_PM4_UCODE_SIZE * 4;
Alex Deucher45f9a392010-03-24 13:55:51 -04001495 rlc_req_size = EVERGREEN_RLC_UCODE_SIZE * 4;
Alex Deucherfe251e22010-03-24 13:36:43 -04001496 } else if (rdev->family >= CHIP_RV770) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001497 pfp_req_size = R700_PFP_UCODE_SIZE * 4;
1498 me_req_size = R700_PM4_UCODE_SIZE * 4;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001499 rlc_req_size = R700_RLC_UCODE_SIZE * 4;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001500 } else {
1501 pfp_req_size = PFP_UCODE_SIZE * 4;
1502 me_req_size = PM4_UCODE_SIZE * 12;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001503 rlc_req_size = RLC_UCODE_SIZE * 4;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001504 }
1505
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001506 DRM_INFO("Loading %s Microcode\n", chip_name);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001507
1508 snprintf(fw_name, sizeof(fw_name), "radeon/%s_pfp.bin", chip_name);
1509 err = request_firmware(&rdev->pfp_fw, fw_name, &pdev->dev);
1510 if (err)
1511 goto out;
1512 if (rdev->pfp_fw->size != pfp_req_size) {
1513 printk(KERN_ERR
1514 "r600_cp: Bogus length %zu in firmware \"%s\"\n",
1515 rdev->pfp_fw->size, fw_name);
1516 err = -EINVAL;
1517 goto out;
1518 }
1519
1520 snprintf(fw_name, sizeof(fw_name), "radeon/%s_me.bin", chip_name);
1521 err = request_firmware(&rdev->me_fw, fw_name, &pdev->dev);
1522 if (err)
1523 goto out;
1524 if (rdev->me_fw->size != me_req_size) {
1525 printk(KERN_ERR
1526 "r600_cp: Bogus length %zu in firmware \"%s\"\n",
1527 rdev->me_fw->size, fw_name);
1528 err = -EINVAL;
1529 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001530
1531 snprintf(fw_name, sizeof(fw_name), "radeon/%s_rlc.bin", rlc_chip_name);
1532 err = request_firmware(&rdev->rlc_fw, fw_name, &pdev->dev);
1533 if (err)
1534 goto out;
1535 if (rdev->rlc_fw->size != rlc_req_size) {
1536 printk(KERN_ERR
1537 "r600_rlc: Bogus length %zu in firmware \"%s\"\n",
1538 rdev->rlc_fw->size, fw_name);
1539 err = -EINVAL;
1540 }
1541
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001542out:
1543 platform_device_unregister(pdev);
1544
1545 if (err) {
1546 if (err != -EINVAL)
1547 printk(KERN_ERR
1548 "r600_cp: Failed to load firmware \"%s\"\n",
1549 fw_name);
1550 release_firmware(rdev->pfp_fw);
1551 rdev->pfp_fw = NULL;
1552 release_firmware(rdev->me_fw);
1553 rdev->me_fw = NULL;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001554 release_firmware(rdev->rlc_fw);
1555 rdev->rlc_fw = NULL;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001556 }
1557 return err;
1558}
1559
1560static int r600_cp_load_microcode(struct radeon_device *rdev)
1561{
1562 const __be32 *fw_data;
1563 int i;
1564
1565 if (!rdev->me_fw || !rdev->pfp_fw)
1566 return -EINVAL;
1567
1568 r600_cp_stop(rdev);
1569
1570 WREG32(CP_RB_CNTL, RB_NO_UPDATE | RB_BLKSZ(15) | RB_BUFSZ(3));
1571
1572 /* Reset cp */
1573 WREG32(GRBM_SOFT_RESET, SOFT_RESET_CP);
1574 RREG32(GRBM_SOFT_RESET);
1575 mdelay(15);
1576 WREG32(GRBM_SOFT_RESET, 0);
1577
1578 WREG32(CP_ME_RAM_WADDR, 0);
1579
1580 fw_data = (const __be32 *)rdev->me_fw->data;
1581 WREG32(CP_ME_RAM_WADDR, 0);
1582 for (i = 0; i < PM4_UCODE_SIZE * 3; i++)
1583 WREG32(CP_ME_RAM_DATA,
1584 be32_to_cpup(fw_data++));
1585
1586 fw_data = (const __be32 *)rdev->pfp_fw->data;
1587 WREG32(CP_PFP_UCODE_ADDR, 0);
1588 for (i = 0; i < PFP_UCODE_SIZE; i++)
1589 WREG32(CP_PFP_UCODE_DATA,
1590 be32_to_cpup(fw_data++));
1591
1592 WREG32(CP_PFP_UCODE_ADDR, 0);
1593 WREG32(CP_ME_RAM_WADDR, 0);
1594 WREG32(CP_ME_RAM_RADDR, 0);
1595 return 0;
1596}
1597
1598int r600_cp_start(struct radeon_device *rdev)
1599{
1600 int r;
1601 uint32_t cp_me;
1602
1603 r = radeon_ring_lock(rdev, 7);
1604 if (r) {
1605 DRM_ERROR("radeon: cp failed to lock ring (%d).\n", r);
1606 return r;
1607 }
1608 radeon_ring_write(rdev, PACKET3(PACKET3_ME_INITIALIZE, 5));
1609 radeon_ring_write(rdev, 0x1);
Alex Deucherfe251e22010-03-24 13:36:43 -04001610 if (rdev->family >= CHIP_CEDAR) {
1611 radeon_ring_write(rdev, 0x0);
1612 radeon_ring_write(rdev, rdev->config.evergreen.max_hw_contexts - 1);
1613 } else if (rdev->family >= CHIP_RV770) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001614 radeon_ring_write(rdev, 0x0);
1615 radeon_ring_write(rdev, rdev->config.rv770.max_hw_contexts - 1);
Alex Deucherfe251e22010-03-24 13:36:43 -04001616 } else {
1617 radeon_ring_write(rdev, 0x3);
1618 radeon_ring_write(rdev, rdev->config.r600.max_hw_contexts - 1);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001619 }
1620 radeon_ring_write(rdev, PACKET3_ME_INITIALIZE_DEVICE_ID(1));
1621 radeon_ring_write(rdev, 0);
1622 radeon_ring_write(rdev, 0);
1623 radeon_ring_unlock_commit(rdev);
1624
1625 cp_me = 0xff;
1626 WREG32(R_0086D8_CP_ME_CNTL, cp_me);
1627 return 0;
1628}
1629
1630int r600_cp_resume(struct radeon_device *rdev)
1631{
1632 u32 tmp;
1633 u32 rb_bufsz;
1634 int r;
1635
1636 /* Reset cp */
1637 WREG32(GRBM_SOFT_RESET, SOFT_RESET_CP);
1638 RREG32(GRBM_SOFT_RESET);
1639 mdelay(15);
1640 WREG32(GRBM_SOFT_RESET, 0);
1641
1642 /* Set ring buffer size */
1643 rb_bufsz = drm_order(rdev->cp.ring_size / 8);
Alex Deucherd6f28932009-11-02 16:01:27 -05001644 tmp = RB_NO_UPDATE | (drm_order(RADEON_GPU_PAGE_SIZE/8) << 8) | rb_bufsz;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001645#ifdef __BIG_ENDIAN
Alex Deucherd6f28932009-11-02 16:01:27 -05001646 tmp |= BUF_SWAP_32BIT;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001647#endif
Alex Deucherd6f28932009-11-02 16:01:27 -05001648 WREG32(CP_RB_CNTL, tmp);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001649 WREG32(CP_SEM_WAIT_TIMER, 0x4);
1650
1651 /* Set the write pointer delay */
1652 WREG32(CP_RB_WPTR_DELAY, 0);
1653
1654 /* Initialize the ring buffer's read and write pointers */
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001655 WREG32(CP_RB_CNTL, tmp | RB_RPTR_WR_ENA);
1656 WREG32(CP_RB_RPTR_WR, 0);
1657 WREG32(CP_RB_WPTR, 0);
1658 WREG32(CP_RB_RPTR_ADDR, rdev->cp.gpu_addr & 0xFFFFFFFF);
1659 WREG32(CP_RB_RPTR_ADDR_HI, upper_32_bits(rdev->cp.gpu_addr));
1660 mdelay(1);
1661 WREG32(CP_RB_CNTL, tmp);
1662
1663 WREG32(CP_RB_BASE, rdev->cp.gpu_addr >> 8);
1664 WREG32(CP_DEBUG, (1 << 27) | (1 << 28));
1665
1666 rdev->cp.rptr = RREG32(CP_RB_RPTR);
1667 rdev->cp.wptr = RREG32(CP_RB_WPTR);
1668
1669 r600_cp_start(rdev);
1670 rdev->cp.ready = true;
1671 r = radeon_ring_test(rdev);
1672 if (r) {
1673 rdev->cp.ready = false;
1674 return r;
1675 }
1676 return 0;
1677}
1678
1679void r600_cp_commit(struct radeon_device *rdev)
1680{
1681 WREG32(CP_RB_WPTR, rdev->cp.wptr);
1682 (void)RREG32(CP_RB_WPTR);
1683}
1684
1685void r600_ring_init(struct radeon_device *rdev, unsigned ring_size)
1686{
1687 u32 rb_bufsz;
1688
1689 /* Align ring size */
1690 rb_bufsz = drm_order(ring_size / 8);
1691 ring_size = (1 << (rb_bufsz + 1)) * 4;
1692 rdev->cp.ring_size = ring_size;
1693 rdev->cp.align_mask = 16 - 1;
1694}
1695
Jerome Glisse655efd32010-02-02 11:51:45 +01001696void r600_cp_fini(struct radeon_device *rdev)
1697{
1698 r600_cp_stop(rdev);
1699 radeon_ring_fini(rdev);
1700}
1701
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001702
1703/*
1704 * GPU scratch registers helpers function.
1705 */
1706void r600_scratch_init(struct radeon_device *rdev)
1707{
1708 int i;
1709
1710 rdev->scratch.num_reg = 7;
1711 for (i = 0; i < rdev->scratch.num_reg; i++) {
1712 rdev->scratch.free[i] = true;
1713 rdev->scratch.reg[i] = SCRATCH_REG0 + (i * 4);
1714 }
1715}
1716
1717int r600_ring_test(struct radeon_device *rdev)
1718{
1719 uint32_t scratch;
1720 uint32_t tmp = 0;
1721 unsigned i;
1722 int r;
1723
1724 r = radeon_scratch_get(rdev, &scratch);
1725 if (r) {
1726 DRM_ERROR("radeon: cp failed to get scratch reg (%d).\n", r);
1727 return r;
1728 }
1729 WREG32(scratch, 0xCAFEDEAD);
1730 r = radeon_ring_lock(rdev, 3);
1731 if (r) {
1732 DRM_ERROR("radeon: cp failed to lock ring (%d).\n", r);
1733 radeon_scratch_free(rdev, scratch);
1734 return r;
1735 }
1736 radeon_ring_write(rdev, PACKET3(PACKET3_SET_CONFIG_REG, 1));
1737 radeon_ring_write(rdev, ((scratch - PACKET3_SET_CONFIG_REG_OFFSET) >> 2));
1738 radeon_ring_write(rdev, 0xDEADBEEF);
1739 radeon_ring_unlock_commit(rdev);
1740 for (i = 0; i < rdev->usec_timeout; i++) {
1741 tmp = RREG32(scratch);
1742 if (tmp == 0xDEADBEEF)
1743 break;
1744 DRM_UDELAY(1);
1745 }
1746 if (i < rdev->usec_timeout) {
1747 DRM_INFO("ring test succeeded in %d usecs\n", i);
1748 } else {
1749 DRM_ERROR("radeon: ring test failed (scratch(0x%04X)=0x%08X)\n",
1750 scratch, tmp);
1751 r = -EINVAL;
1752 }
1753 radeon_scratch_free(rdev, scratch);
1754 return r;
1755}
1756
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001757void r600_wb_disable(struct radeon_device *rdev)
1758{
Jerome Glisse4c788672009-11-20 14:29:23 +01001759 int r;
1760
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001761 WREG32(SCRATCH_UMSK, 0);
1762 if (rdev->wb.wb_obj) {
Jerome Glisse4c788672009-11-20 14:29:23 +01001763 r = radeon_bo_reserve(rdev->wb.wb_obj, false);
1764 if (unlikely(r != 0))
1765 return;
1766 radeon_bo_kunmap(rdev->wb.wb_obj);
1767 radeon_bo_unpin(rdev->wb.wb_obj);
1768 radeon_bo_unreserve(rdev->wb.wb_obj);
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001769 }
1770}
1771
1772void r600_wb_fini(struct radeon_device *rdev)
1773{
1774 r600_wb_disable(rdev);
1775 if (rdev->wb.wb_obj) {
Jerome Glisse4c788672009-11-20 14:29:23 +01001776 radeon_bo_unref(&rdev->wb.wb_obj);
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001777 rdev->wb.wb = NULL;
1778 rdev->wb.wb_obj = NULL;
1779 }
1780}
1781
1782int r600_wb_enable(struct radeon_device *rdev)
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001783{
1784 int r;
1785
1786 if (rdev->wb.wb_obj == NULL) {
Jerome Glisse4c788672009-11-20 14:29:23 +01001787 r = radeon_bo_create(rdev, NULL, RADEON_GPU_PAGE_SIZE, true,
1788 RADEON_GEM_DOMAIN_GTT, &rdev->wb.wb_obj);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001789 if (r) {
Jerome Glisse4c788672009-11-20 14:29:23 +01001790 dev_warn(rdev->dev, "(%d) create WB bo failed\n", r);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001791 return r;
1792 }
Jerome Glisse4c788672009-11-20 14:29:23 +01001793 r = radeon_bo_reserve(rdev->wb.wb_obj, false);
1794 if (unlikely(r != 0)) {
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001795 r600_wb_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001796 return r;
1797 }
Jerome Glisse4c788672009-11-20 14:29:23 +01001798 r = radeon_bo_pin(rdev->wb.wb_obj, RADEON_GEM_DOMAIN_GTT,
1799 &rdev->wb.gpu_addr);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001800 if (r) {
Jerome Glisse4c788672009-11-20 14:29:23 +01001801 radeon_bo_unreserve(rdev->wb.wb_obj);
1802 dev_warn(rdev->dev, "(%d) pin WB bo failed\n", r);
1803 r600_wb_fini(rdev);
1804 return r;
1805 }
1806 r = radeon_bo_kmap(rdev->wb.wb_obj, (void **)&rdev->wb.wb);
1807 radeon_bo_unreserve(rdev->wb.wb_obj);
1808 if (r) {
1809 dev_warn(rdev->dev, "(%d) map WB bo failed\n", r);
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001810 r600_wb_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001811 return r;
1812 }
1813 }
1814 WREG32(SCRATCH_ADDR, (rdev->wb.gpu_addr >> 8) & 0xFFFFFFFF);
1815 WREG32(CP_RB_RPTR_ADDR, (rdev->wb.gpu_addr + 1024) & 0xFFFFFFFC);
1816 WREG32(CP_RB_RPTR_ADDR_HI, upper_32_bits(rdev->wb.gpu_addr + 1024) & 0xFF);
1817 WREG32(SCRATCH_UMSK, 0xff);
1818 return 0;
1819}
1820
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001821void r600_fence_ring_emit(struct radeon_device *rdev,
1822 struct radeon_fence *fence)
1823{
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001824 /* Also consider EVENT_WRITE_EOP. it handles the interrupts + timestamps + events */
Alex Deucher44224c32010-02-04 11:01:52 -05001825
1826 radeon_ring_write(rdev, PACKET3(PACKET3_EVENT_WRITE, 0));
1827 radeon_ring_write(rdev, CACHE_FLUSH_AND_INV_EVENT);
1828 /* wait for 3D idle clean */
1829 radeon_ring_write(rdev, PACKET3(PACKET3_SET_CONFIG_REG, 1));
1830 radeon_ring_write(rdev, (WAIT_UNTIL - PACKET3_SET_CONFIG_REG_OFFSET) >> 2);
1831 radeon_ring_write(rdev, WAIT_3D_IDLE_bit | WAIT_3D_IDLECLEAN_bit);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001832 /* Emit fence sequence & fire IRQ */
1833 radeon_ring_write(rdev, PACKET3(PACKET3_SET_CONFIG_REG, 1));
1834 radeon_ring_write(rdev, ((rdev->fence_drv.scratch_reg - PACKET3_SET_CONFIG_REG_OFFSET) >> 2));
1835 radeon_ring_write(rdev, fence->seq);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001836 /* CP_INTERRUPT packet 3 no longer exists, use packet 0 */
1837 radeon_ring_write(rdev, PACKET0(CP_INT_STATUS, 0));
1838 radeon_ring_write(rdev, RB_INT_STAT);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001839}
1840
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001841int r600_copy_blit(struct radeon_device *rdev,
1842 uint64_t src_offset, uint64_t dst_offset,
1843 unsigned num_pages, struct radeon_fence *fence)
1844{
Jerome Glisseff82f052010-01-22 15:19:00 +01001845 int r;
1846
1847 mutex_lock(&rdev->r600_blit.mutex);
1848 rdev->r600_blit.vb_ib = NULL;
1849 r = r600_blit_prepare_copy(rdev, num_pages * RADEON_GPU_PAGE_SIZE);
1850 if (r) {
1851 if (rdev->r600_blit.vb_ib)
1852 radeon_ib_free(rdev, &rdev->r600_blit.vb_ib);
1853 mutex_unlock(&rdev->r600_blit.mutex);
1854 return r;
1855 }
Matt Turnera77f1712009-10-14 00:34:41 -04001856 r600_kms_blit_copy(rdev, src_offset, dst_offset, num_pages * RADEON_GPU_PAGE_SIZE);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001857 r600_blit_done_copy(rdev, fence);
Jerome Glisseff82f052010-01-22 15:19:00 +01001858 mutex_unlock(&rdev->r600_blit.mutex);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001859 return 0;
1860}
1861
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001862int r600_set_surface_reg(struct radeon_device *rdev, int reg,
1863 uint32_t tiling_flags, uint32_t pitch,
1864 uint32_t offset, uint32_t obj_size)
1865{
1866 /* FIXME: implement */
1867 return 0;
1868}
1869
1870void r600_clear_surface_reg(struct radeon_device *rdev, int reg)
1871{
1872 /* FIXME: implement */
1873}
1874
1875
1876bool r600_card_posted(struct radeon_device *rdev)
1877{
1878 uint32_t reg;
1879
1880 /* first check CRTCs */
1881 reg = RREG32(D1CRTC_CONTROL) |
1882 RREG32(D2CRTC_CONTROL);
1883 if (reg & CRTC_EN)
1884 return true;
1885
1886 /* then check MEM_SIZE, in case the crtcs are off */
1887 if (RREG32(CONFIG_MEMSIZE))
1888 return true;
1889
1890 return false;
1891}
1892
Dave Airliefc30b8e2009-09-18 15:19:37 +10001893int r600_startup(struct radeon_device *rdev)
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001894{
1895 int r;
1896
Alex Deucher779720a2009-12-09 19:31:44 -05001897 if (!rdev->me_fw || !rdev->pfp_fw || !rdev->rlc_fw) {
1898 r = r600_init_microcode(rdev);
1899 if (r) {
1900 DRM_ERROR("Failed to load firmware!\n");
1901 return r;
1902 }
1903 }
1904
Jerome Glissea3c19452009-10-01 18:02:13 +02001905 r600_mc_program(rdev);
Jerome Glisse1a029b72009-10-06 19:04:30 +02001906 if (rdev->flags & RADEON_IS_AGP) {
1907 r600_agp_enable(rdev);
1908 } else {
1909 r = r600_pcie_gart_enable(rdev);
1910 if (r)
1911 return r;
1912 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001913 r600_gpu_init(rdev);
Jerome Glissec38c7b62010-02-04 17:27:27 +01001914 r = r600_blit_init(rdev);
1915 if (r) {
1916 r600_blit_fini(rdev);
1917 rdev->asic->copy = NULL;
1918 dev_warn(rdev->dev, "failed blitter (%d) falling back to memcpy\n", r);
1919 }
Jerome Glisseff82f052010-01-22 15:19:00 +01001920 /* pin copy shader into vram */
1921 if (rdev->r600_blit.shader_obj) {
1922 r = radeon_bo_reserve(rdev->r600_blit.shader_obj, false);
1923 if (unlikely(r != 0))
1924 return r;
1925 r = radeon_bo_pin(rdev->r600_blit.shader_obj, RADEON_GEM_DOMAIN_VRAM,
1926 &rdev->r600_blit.shader_gpu_addr);
1927 radeon_bo_unreserve(rdev->r600_blit.shader_obj);
Alex Deucher7923c612009-12-15 17:15:07 -05001928 if (r) {
Jerome Glisseff82f052010-01-22 15:19:00 +01001929 dev_err(rdev->dev, "(%d) pin blit object failed\n", r);
Alex Deucher7923c612009-12-15 17:15:07 -05001930 return r;
1931 }
1932 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001933 /* Enable IRQ */
Alex Deucherd8f60cf2009-12-01 13:43:46 -05001934 r = r600_irq_init(rdev);
1935 if (r) {
1936 DRM_ERROR("radeon: IH init failed (%d).\n", r);
1937 radeon_irq_kms_fini(rdev);
1938 return r;
1939 }
1940 r600_irq_set(rdev);
1941
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001942 r = radeon_ring_init(rdev, rdev->cp.ring_size);
1943 if (r)
1944 return r;
1945 r = r600_cp_load_microcode(rdev);
1946 if (r)
1947 return r;
1948 r = r600_cp_resume(rdev);
1949 if (r)
1950 return r;
Jerome Glisse81cc35b2009-10-01 18:02:12 +02001951 /* write back buffer are not vital so don't worry about failure */
1952 r600_wb_enable(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10001953 return 0;
1954}
1955
Dave Airlie28d52042009-09-21 14:33:58 +10001956void r600_vga_set_state(struct radeon_device *rdev, bool state)
1957{
1958 uint32_t temp;
1959
1960 temp = RREG32(CONFIG_CNTL);
1961 if (state == false) {
1962 temp &= ~(1<<0);
1963 temp |= (1<<1);
1964 } else {
1965 temp &= ~(1<<1);
1966 }
1967 WREG32(CONFIG_CNTL, temp);
1968}
1969
Dave Airliefc30b8e2009-09-18 15:19:37 +10001970int r600_resume(struct radeon_device *rdev)
1971{
1972 int r;
1973
Jerome Glisse1a029b72009-10-06 19:04:30 +02001974 /* Do not reset GPU before posting, on r600 hw unlike on r500 hw,
1975 * posting will perform necessary task to bring back GPU into good
1976 * shape.
1977 */
Dave Airliefc30b8e2009-09-18 15:19:37 +10001978 /* post card */
Jerome Glissee7d40b92009-10-01 18:02:15 +02001979 atom_asic_init(rdev->mode_info.atom_context);
Dave Airliefc30b8e2009-09-18 15:19:37 +10001980 /* Initialize clocks */
1981 r = radeon_clocks_init(rdev);
1982 if (r) {
1983 return r;
1984 }
1985
1986 r = r600_startup(rdev);
1987 if (r) {
1988 DRM_ERROR("r600 startup failed on resume\n");
1989 return r;
1990 }
1991
Jerome Glisse62a8ea32009-10-01 18:02:11 +02001992 r = r600_ib_test(rdev);
Dave Airliefc30b8e2009-09-18 15:19:37 +10001993 if (r) {
1994 DRM_ERROR("radeon: failled testing IB (%d).\n", r);
1995 return r;
1996 }
Rafał Miłecki38fd2c62010-01-28 18:16:30 +01001997
1998 r = r600_audio_init(rdev);
1999 if (r) {
2000 DRM_ERROR("radeon: audio resume failed\n");
2001 return r;
2002 }
2003
Dave Airliefc30b8e2009-09-18 15:19:37 +10002004 return r;
2005}
2006
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002007int r600_suspend(struct radeon_device *rdev)
2008{
Jerome Glisse4c788672009-11-20 14:29:23 +01002009 int r;
2010
Rafał Miłecki38fd2c62010-01-28 18:16:30 +01002011 r600_audio_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002012 /* FIXME: we should wait for ring to be empty */
2013 r600_cp_stop(rdev);
Dave Airliebc1a6312009-09-15 11:07:52 +10002014 rdev->cp.ready = false;
Jerome Glisse0c452492010-01-15 14:44:37 +01002015 r600_irq_suspend(rdev);
Jerome Glisse81cc35b2009-10-01 18:02:12 +02002016 r600_wb_disable(rdev);
Jerome Glisse4aac0472009-09-14 18:29:49 +02002017 r600_pcie_gart_disable(rdev);
Dave Airliebc1a6312009-09-15 11:07:52 +10002018 /* unpin shaders bo */
Jerome Glisse30d2d9a2010-01-13 10:29:27 +01002019 if (rdev->r600_blit.shader_obj) {
2020 r = radeon_bo_reserve(rdev->r600_blit.shader_obj, false);
2021 if (!r) {
2022 radeon_bo_unpin(rdev->r600_blit.shader_obj);
2023 radeon_bo_unreserve(rdev->r600_blit.shader_obj);
2024 }
2025 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002026 return 0;
2027}
2028
2029/* Plan is to move initialization in that function and use
2030 * helper function so that radeon_device_init pretty much
2031 * do nothing more than calling asic specific function. This
2032 * should also allow to remove a bunch of callback function
2033 * like vram_info.
2034 */
2035int r600_init(struct radeon_device *rdev)
2036{
2037 int r;
2038
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002039 r = radeon_dummy_page_init(rdev);
2040 if (r)
2041 return r;
2042 if (r600_debugfs_mc_info_init(rdev)) {
2043 DRM_ERROR("Failed to register debugfs file for mc !\n");
2044 }
2045 /* This don't do much */
2046 r = radeon_gem_init(rdev);
2047 if (r)
2048 return r;
2049 /* Read BIOS */
2050 if (!radeon_get_bios(rdev)) {
2051 if (ASIC_IS_AVIVO(rdev))
2052 return -EINVAL;
2053 }
2054 /* Must be an ATOMBIOS */
Jerome Glissee7d40b92009-10-01 18:02:15 +02002055 if (!rdev->is_atom_bios) {
2056 dev_err(rdev->dev, "Expecting atombios for R600 GPU\n");
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002057 return -EINVAL;
Jerome Glissee7d40b92009-10-01 18:02:15 +02002058 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002059 r = radeon_atombios_init(rdev);
2060 if (r)
2061 return r;
2062 /* Post card if necessary */
Dave Airlie72542d72009-12-01 14:06:31 +10002063 if (!r600_card_posted(rdev)) {
2064 if (!rdev->bios) {
2065 dev_err(rdev->dev, "Card not posted and no BIOS - ignoring\n");
2066 return -EINVAL;
2067 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002068 DRM_INFO("GPU not posted. posting now...\n");
2069 atom_asic_init(rdev->mode_info.atom_context);
2070 }
2071 /* Initialize scratch registers */
2072 r600_scratch_init(rdev);
2073 /* Initialize surface registers */
2074 radeon_surface_init(rdev);
Rafał Miłecki74338742009-11-03 00:53:02 +01002075 /* Initialize clocks */
Michel Dänzer5e6dde72009-09-17 09:42:28 +02002076 radeon_get_clock_info(rdev->ddev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002077 r = radeon_clocks_init(rdev);
2078 if (r)
2079 return r;
Rafał Miłecki74338742009-11-03 00:53:02 +01002080 /* Initialize power management */
2081 radeon_pm_init(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002082 /* Fence driver */
2083 r = radeon_fence_driver_init(rdev);
2084 if (r)
2085 return r;
Jerome Glisse700a0cc2010-01-13 15:16:38 +01002086 if (rdev->flags & RADEON_IS_AGP) {
2087 r = radeon_agp_init(rdev);
2088 if (r)
2089 radeon_agp_disable(rdev);
2090 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002091 r = r600_mc_init(rdev);
Jerome Glisseb574f252009-10-06 19:04:29 +02002092 if (r)
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002093 return r;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002094 /* Memory manager */
Jerome Glisse4c788672009-11-20 14:29:23 +01002095 r = radeon_bo_init(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002096 if (r)
2097 return r;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002098
2099 r = radeon_irq_kms_init(rdev);
2100 if (r)
2101 return r;
2102
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002103 rdev->cp.ring_obj = NULL;
2104 r600_ring_init(rdev, 1024 * 1024);
2105
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002106 rdev->ih.ring_obj = NULL;
2107 r600_ih_ring_init(rdev, 64 * 1024);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002108
Jerome Glisse4aac0472009-09-14 18:29:49 +02002109 r = r600_pcie_gart_init(rdev);
2110 if (r)
2111 return r;
2112
Alex Deucher779720a2009-12-09 19:31:44 -05002113 rdev->accel_working = true;
Dave Airliefc30b8e2009-09-18 15:19:37 +10002114 r = r600_startup(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002115 if (r) {
Jerome Glisse655efd32010-02-02 11:51:45 +01002116 dev_err(rdev->dev, "disabling GPU acceleration\n");
2117 r600_cp_fini(rdev);
Jerome Glisse75c81292009-10-01 18:02:14 +02002118 r600_wb_fini(rdev);
Jerome Glisse655efd32010-02-02 11:51:45 +01002119 r600_irq_fini(rdev);
2120 radeon_irq_kms_fini(rdev);
Jerome Glisse75c81292009-10-01 18:02:14 +02002121 r600_pcie_gart_fini(rdev);
Jerome Glisse733289c2009-09-16 15:24:21 +02002122 rdev->accel_working = false;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002123 }
Jerome Glisse733289c2009-09-16 15:24:21 +02002124 if (rdev->accel_working) {
2125 r = radeon_ib_pool_init(rdev);
2126 if (r) {
Jerome Glissedb963802010-01-17 21:21:56 +01002127 dev_err(rdev->dev, "IB initialization failed (%d).\n", r);
Jerome Glisse733289c2009-09-16 15:24:21 +02002128 rdev->accel_working = false;
Jerome Glissedb963802010-01-17 21:21:56 +01002129 } else {
2130 r = r600_ib_test(rdev);
2131 if (r) {
2132 dev_err(rdev->dev, "IB test failed (%d).\n", r);
2133 rdev->accel_working = false;
2134 }
Jerome Glisse733289c2009-09-16 15:24:21 +02002135 }
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002136 }
Christian Koenigdafc3bd2009-10-11 23:49:13 +02002137
2138 r = r600_audio_init(rdev);
2139 if (r)
2140 return r; /* TODO error handling */
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002141 return 0;
2142}
2143
2144void r600_fini(struct radeon_device *rdev)
2145{
Alex Deucher29fb52c2010-03-11 10:01:17 -05002146 radeon_pm_fini(rdev);
Christian Koenigdafc3bd2009-10-11 23:49:13 +02002147 r600_audio_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002148 r600_blit_fini(rdev);
Jerome Glisse655efd32010-02-02 11:51:45 +01002149 r600_cp_fini(rdev);
2150 r600_wb_fini(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002151 r600_irq_fini(rdev);
2152 radeon_irq_kms_fini(rdev);
Jerome Glisse4aac0472009-09-14 18:29:49 +02002153 r600_pcie_gart_fini(rdev);
Jerome Glisse655efd32010-02-02 11:51:45 +01002154 radeon_agp_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002155 radeon_gem_fini(rdev);
2156 radeon_fence_driver_fini(rdev);
2157 radeon_clocks_fini(rdev);
Jerome Glisse4c788672009-11-20 14:29:23 +01002158 radeon_bo_fini(rdev);
Jerome Glissee7d40b92009-10-01 18:02:15 +02002159 radeon_atombios_fini(rdev);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002160 kfree(rdev->bios);
2161 rdev->bios = NULL;
2162 radeon_dummy_page_fini(rdev);
2163}
2164
2165
2166/*
2167 * CS stuff
2168 */
2169void r600_ring_ib_execute(struct radeon_device *rdev, struct radeon_ib *ib)
2170{
2171 /* FIXME: implement */
2172 radeon_ring_write(rdev, PACKET3(PACKET3_INDIRECT_BUFFER, 2));
2173 radeon_ring_write(rdev, ib->gpu_addr & 0xFFFFFFFC);
2174 radeon_ring_write(rdev, upper_32_bits(ib->gpu_addr) & 0xFF);
2175 radeon_ring_write(rdev, ib->length_dw);
2176}
2177
2178int r600_ib_test(struct radeon_device *rdev)
2179{
2180 struct radeon_ib *ib;
2181 uint32_t scratch;
2182 uint32_t tmp = 0;
2183 unsigned i;
2184 int r;
2185
2186 r = radeon_scratch_get(rdev, &scratch);
2187 if (r) {
2188 DRM_ERROR("radeon: failed to get scratch reg (%d).\n", r);
2189 return r;
2190 }
2191 WREG32(scratch, 0xCAFEDEAD);
2192 r = radeon_ib_get(rdev, &ib);
2193 if (r) {
2194 DRM_ERROR("radeon: failed to get ib (%d).\n", r);
2195 return r;
2196 }
2197 ib->ptr[0] = PACKET3(PACKET3_SET_CONFIG_REG, 1);
2198 ib->ptr[1] = ((scratch - PACKET3_SET_CONFIG_REG_OFFSET) >> 2);
2199 ib->ptr[2] = 0xDEADBEEF;
2200 ib->ptr[3] = PACKET2(0);
2201 ib->ptr[4] = PACKET2(0);
2202 ib->ptr[5] = PACKET2(0);
2203 ib->ptr[6] = PACKET2(0);
2204 ib->ptr[7] = PACKET2(0);
2205 ib->ptr[8] = PACKET2(0);
2206 ib->ptr[9] = PACKET2(0);
2207 ib->ptr[10] = PACKET2(0);
2208 ib->ptr[11] = PACKET2(0);
2209 ib->ptr[12] = PACKET2(0);
2210 ib->ptr[13] = PACKET2(0);
2211 ib->ptr[14] = PACKET2(0);
2212 ib->ptr[15] = PACKET2(0);
2213 ib->length_dw = 16;
2214 r = radeon_ib_schedule(rdev, ib);
2215 if (r) {
2216 radeon_scratch_free(rdev, scratch);
2217 radeon_ib_free(rdev, &ib);
2218 DRM_ERROR("radeon: failed to schedule ib (%d).\n", r);
2219 return r;
2220 }
2221 r = radeon_fence_wait(ib->fence, false);
2222 if (r) {
2223 DRM_ERROR("radeon: fence wait failed (%d).\n", r);
2224 return r;
2225 }
2226 for (i = 0; i < rdev->usec_timeout; i++) {
2227 tmp = RREG32(scratch);
2228 if (tmp == 0xDEADBEEF)
2229 break;
2230 DRM_UDELAY(1);
2231 }
2232 if (i < rdev->usec_timeout) {
2233 DRM_INFO("ib test succeeded in %u usecs\n", i);
2234 } else {
2235 DRM_ERROR("radeon: ib test failed (sracth(0x%04X)=0x%08X)\n",
2236 scratch, tmp);
2237 r = -EINVAL;
2238 }
2239 radeon_scratch_free(rdev, scratch);
2240 radeon_ib_free(rdev, &ib);
2241 return r;
2242}
2243
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002244/*
2245 * Interrupts
2246 *
2247 * Interrupts use a ring buffer on r6xx/r7xx hardware. It works pretty
2248 * the same as the CP ring buffer, but in reverse. Rather than the CPU
2249 * writing to the ring and the GPU consuming, the GPU writes to the ring
2250 * and host consumes. As the host irq handler processes interrupts, it
2251 * increments the rptr. When the rptr catches up with the wptr, all the
2252 * current interrupts have been processed.
2253 */
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002254
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002255void r600_ih_ring_init(struct radeon_device *rdev, unsigned ring_size)
2256{
2257 u32 rb_bufsz;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002258
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002259 /* Align ring size */
2260 rb_bufsz = drm_order(ring_size / 4);
2261 ring_size = (1 << rb_bufsz) * 4;
2262 rdev->ih.ring_size = ring_size;
Jerome Glisse0c452492010-01-15 14:44:37 +01002263 rdev->ih.ptr_mask = rdev->ih.ring_size - 1;
2264 rdev->ih.rptr = 0;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002265}
2266
Jerome Glisse0c452492010-01-15 14:44:37 +01002267static int r600_ih_ring_alloc(struct radeon_device *rdev)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002268{
2269 int r;
2270
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002271 /* Allocate ring buffer */
2272 if (rdev->ih.ring_obj == NULL) {
Jerome Glisse4c788672009-11-20 14:29:23 +01002273 r = radeon_bo_create(rdev, NULL, rdev->ih.ring_size,
2274 true,
2275 RADEON_GEM_DOMAIN_GTT,
2276 &rdev->ih.ring_obj);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002277 if (r) {
2278 DRM_ERROR("radeon: failed to create ih ring buffer (%d).\n", r);
2279 return r;
2280 }
Jerome Glisse4c788672009-11-20 14:29:23 +01002281 r = radeon_bo_reserve(rdev->ih.ring_obj, false);
2282 if (unlikely(r != 0))
2283 return r;
2284 r = radeon_bo_pin(rdev->ih.ring_obj,
2285 RADEON_GEM_DOMAIN_GTT,
2286 &rdev->ih.gpu_addr);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002287 if (r) {
Jerome Glisse4c788672009-11-20 14:29:23 +01002288 radeon_bo_unreserve(rdev->ih.ring_obj);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002289 DRM_ERROR("radeon: failed to pin ih ring buffer (%d).\n", r);
2290 return r;
2291 }
Jerome Glisse4c788672009-11-20 14:29:23 +01002292 r = radeon_bo_kmap(rdev->ih.ring_obj,
2293 (void **)&rdev->ih.ring);
2294 radeon_bo_unreserve(rdev->ih.ring_obj);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002295 if (r) {
2296 DRM_ERROR("radeon: failed to map ih ring buffer (%d).\n", r);
2297 return r;
2298 }
2299 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002300 return 0;
2301}
2302
2303static void r600_ih_ring_fini(struct radeon_device *rdev)
2304{
Jerome Glisse4c788672009-11-20 14:29:23 +01002305 int r;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002306 if (rdev->ih.ring_obj) {
Jerome Glisse4c788672009-11-20 14:29:23 +01002307 r = radeon_bo_reserve(rdev->ih.ring_obj, false);
2308 if (likely(r == 0)) {
2309 radeon_bo_kunmap(rdev->ih.ring_obj);
2310 radeon_bo_unpin(rdev->ih.ring_obj);
2311 radeon_bo_unreserve(rdev->ih.ring_obj);
2312 }
2313 radeon_bo_unref(&rdev->ih.ring_obj);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002314 rdev->ih.ring = NULL;
2315 rdev->ih.ring_obj = NULL;
2316 }
2317}
2318
Alex Deucher45f9a392010-03-24 13:55:51 -04002319void r600_rlc_stop(struct radeon_device *rdev)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002320{
2321
Alex Deucher45f9a392010-03-24 13:55:51 -04002322 if ((rdev->family >= CHIP_RV770) &&
2323 (rdev->family <= CHIP_RV740)) {
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002324 /* r7xx asics need to soft reset RLC before halting */
2325 WREG32(SRBM_SOFT_RESET, SOFT_RESET_RLC);
2326 RREG32(SRBM_SOFT_RESET);
2327 udelay(15000);
2328 WREG32(SRBM_SOFT_RESET, 0);
2329 RREG32(SRBM_SOFT_RESET);
2330 }
2331
2332 WREG32(RLC_CNTL, 0);
2333}
2334
2335static void r600_rlc_start(struct radeon_device *rdev)
2336{
2337 WREG32(RLC_CNTL, RLC_ENABLE);
2338}
2339
2340static int r600_rlc_init(struct radeon_device *rdev)
2341{
2342 u32 i;
2343 const __be32 *fw_data;
2344
2345 if (!rdev->rlc_fw)
2346 return -EINVAL;
2347
2348 r600_rlc_stop(rdev);
2349
2350 WREG32(RLC_HB_BASE, 0);
2351 WREG32(RLC_HB_CNTL, 0);
2352 WREG32(RLC_HB_RPTR, 0);
2353 WREG32(RLC_HB_WPTR, 0);
2354 WREG32(RLC_HB_WPTR_LSB_ADDR, 0);
2355 WREG32(RLC_HB_WPTR_MSB_ADDR, 0);
2356 WREG32(RLC_MC_CNTL, 0);
2357 WREG32(RLC_UCODE_CNTL, 0);
2358
2359 fw_data = (const __be32 *)rdev->rlc_fw->data;
Alex Deucher45f9a392010-03-24 13:55:51 -04002360 if (rdev->family >= CHIP_CEDAR) {
2361 for (i = 0; i < EVERGREEN_RLC_UCODE_SIZE; i++) {
2362 WREG32(RLC_UCODE_ADDR, i);
2363 WREG32(RLC_UCODE_DATA, be32_to_cpup(fw_data++));
2364 }
2365 } else if (rdev->family >= CHIP_RV770) {
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002366 for (i = 0; i < R700_RLC_UCODE_SIZE; i++) {
2367 WREG32(RLC_UCODE_ADDR, i);
2368 WREG32(RLC_UCODE_DATA, be32_to_cpup(fw_data++));
2369 }
2370 } else {
2371 for (i = 0; i < RLC_UCODE_SIZE; i++) {
2372 WREG32(RLC_UCODE_ADDR, i);
2373 WREG32(RLC_UCODE_DATA, be32_to_cpup(fw_data++));
2374 }
2375 }
2376 WREG32(RLC_UCODE_ADDR, 0);
2377
2378 r600_rlc_start(rdev);
2379
2380 return 0;
2381}
2382
2383static void r600_enable_interrupts(struct radeon_device *rdev)
2384{
2385 u32 ih_cntl = RREG32(IH_CNTL);
2386 u32 ih_rb_cntl = RREG32(IH_RB_CNTL);
2387
2388 ih_cntl |= ENABLE_INTR;
2389 ih_rb_cntl |= IH_RB_ENABLE;
2390 WREG32(IH_CNTL, ih_cntl);
2391 WREG32(IH_RB_CNTL, ih_rb_cntl);
2392 rdev->ih.enabled = true;
2393}
2394
Alex Deucher45f9a392010-03-24 13:55:51 -04002395void r600_disable_interrupts(struct radeon_device *rdev)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002396{
2397 u32 ih_rb_cntl = RREG32(IH_RB_CNTL);
2398 u32 ih_cntl = RREG32(IH_CNTL);
2399
2400 ih_rb_cntl &= ~IH_RB_ENABLE;
2401 ih_cntl &= ~ENABLE_INTR;
2402 WREG32(IH_RB_CNTL, ih_rb_cntl);
2403 WREG32(IH_CNTL, ih_cntl);
2404 /* set rptr, wptr to 0 */
2405 WREG32(IH_RB_RPTR, 0);
2406 WREG32(IH_RB_WPTR, 0);
2407 rdev->ih.enabled = false;
2408 rdev->ih.wptr = 0;
2409 rdev->ih.rptr = 0;
2410}
2411
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002412static void r600_disable_interrupt_state(struct radeon_device *rdev)
2413{
2414 u32 tmp;
2415
2416 WREG32(CP_INT_CNTL, 0);
2417 WREG32(GRBM_INT_CNTL, 0);
2418 WREG32(DxMODE_INT_MASK, 0);
2419 if (ASIC_IS_DCE3(rdev)) {
2420 WREG32(DCE3_DACA_AUTODETECT_INT_CONTROL, 0);
2421 WREG32(DCE3_DACB_AUTODETECT_INT_CONTROL, 0);
2422 tmp = RREG32(DC_HPD1_INT_CONTROL) & DC_HPDx_INT_POLARITY;
2423 WREG32(DC_HPD1_INT_CONTROL, tmp);
2424 tmp = RREG32(DC_HPD2_INT_CONTROL) & DC_HPDx_INT_POLARITY;
2425 WREG32(DC_HPD2_INT_CONTROL, tmp);
2426 tmp = RREG32(DC_HPD3_INT_CONTROL) & DC_HPDx_INT_POLARITY;
2427 WREG32(DC_HPD3_INT_CONTROL, tmp);
2428 tmp = RREG32(DC_HPD4_INT_CONTROL) & DC_HPDx_INT_POLARITY;
2429 WREG32(DC_HPD4_INT_CONTROL, tmp);
2430 if (ASIC_IS_DCE32(rdev)) {
2431 tmp = RREG32(DC_HPD5_INT_CONTROL) & DC_HPDx_INT_POLARITY;
Alex Deucher5898b1f2010-03-24 13:57:29 -04002432 WREG32(DC_HPD5_INT_CONTROL, tmp);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002433 tmp = RREG32(DC_HPD6_INT_CONTROL) & DC_HPDx_INT_POLARITY;
Alex Deucher5898b1f2010-03-24 13:57:29 -04002434 WREG32(DC_HPD6_INT_CONTROL, tmp);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002435 }
2436 } else {
2437 WREG32(DACA_AUTODETECT_INT_CONTROL, 0);
2438 WREG32(DACB_AUTODETECT_INT_CONTROL, 0);
2439 tmp = RREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL) & DC_HOT_PLUG_DETECTx_INT_POLARITY;
Alex Deucher5898b1f2010-03-24 13:57:29 -04002440 WREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL, tmp);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002441 tmp = RREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL) & DC_HOT_PLUG_DETECTx_INT_POLARITY;
Alex Deucher5898b1f2010-03-24 13:57:29 -04002442 WREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL, tmp);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002443 tmp = RREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL) & DC_HOT_PLUG_DETECTx_INT_POLARITY;
Alex Deucher5898b1f2010-03-24 13:57:29 -04002444 WREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL, tmp);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002445 }
2446}
2447
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002448int r600_irq_init(struct radeon_device *rdev)
2449{
2450 int ret = 0;
2451 int rb_bufsz;
2452 u32 interrupt_cntl, ih_cntl, ih_rb_cntl;
2453
2454 /* allocate ring */
Jerome Glisse0c452492010-01-15 14:44:37 +01002455 ret = r600_ih_ring_alloc(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002456 if (ret)
2457 return ret;
2458
2459 /* disable irqs */
2460 r600_disable_interrupts(rdev);
2461
2462 /* init rlc */
2463 ret = r600_rlc_init(rdev);
2464 if (ret) {
2465 r600_ih_ring_fini(rdev);
2466 return ret;
2467 }
2468
2469 /* setup interrupt control */
2470 /* set dummy read address to ring address */
2471 WREG32(INTERRUPT_CNTL2, rdev->ih.gpu_addr >> 8);
2472 interrupt_cntl = RREG32(INTERRUPT_CNTL);
2473 /* IH_DUMMY_RD_OVERRIDE=0 - dummy read disabled with msi, enabled without msi
2474 * IH_DUMMY_RD_OVERRIDE=1 - dummy read controlled by IH_DUMMY_RD_EN
2475 */
2476 interrupt_cntl &= ~IH_DUMMY_RD_OVERRIDE;
2477 /* IH_REQ_NONSNOOP_EN=1 if ring is in non-cacheable memory, e.g., vram */
2478 interrupt_cntl &= ~IH_REQ_NONSNOOP_EN;
2479 WREG32(INTERRUPT_CNTL, interrupt_cntl);
2480
2481 WREG32(IH_RB_BASE, rdev->ih.gpu_addr >> 8);
2482 rb_bufsz = drm_order(rdev->ih.ring_size / 4);
2483
2484 ih_rb_cntl = (IH_WPTR_OVERFLOW_ENABLE |
2485 IH_WPTR_OVERFLOW_CLEAR |
2486 (rb_bufsz << 1));
2487 /* WPTR writeback, not yet */
2488 /*ih_rb_cntl |= IH_WPTR_WRITEBACK_ENABLE;*/
2489 WREG32(IH_RB_WPTR_ADDR_LO, 0);
2490 WREG32(IH_RB_WPTR_ADDR_HI, 0);
2491
2492 WREG32(IH_RB_CNTL, ih_rb_cntl);
2493
2494 /* set rptr, wptr to 0 */
2495 WREG32(IH_RB_RPTR, 0);
2496 WREG32(IH_RB_WPTR, 0);
2497
2498 /* Default settings for IH_CNTL (disabled at first) */
2499 ih_cntl = MC_WRREQ_CREDIT(0x10) | MC_WR_CLEAN_CNT(0x10);
2500 /* RPTR_REARM only works if msi's are enabled */
2501 if (rdev->msi_enabled)
2502 ih_cntl |= RPTR_REARM;
2503
2504#ifdef __BIG_ENDIAN
2505 ih_cntl |= IH_MC_SWAP(IH_MC_SWAP_32BIT);
2506#endif
2507 WREG32(IH_CNTL, ih_cntl);
2508
2509 /* force the active interrupt state to all disabled */
Alex Deucher45f9a392010-03-24 13:55:51 -04002510 if (rdev->family >= CHIP_CEDAR)
2511 evergreen_disable_interrupt_state(rdev);
2512 else
2513 r600_disable_interrupt_state(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002514
2515 /* enable irqs */
2516 r600_enable_interrupts(rdev);
2517
2518 return ret;
2519}
2520
Jerome Glisse0c452492010-01-15 14:44:37 +01002521void r600_irq_suspend(struct radeon_device *rdev)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002522{
Alex Deucher45f9a392010-03-24 13:55:51 -04002523 r600_irq_disable(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002524 r600_rlc_stop(rdev);
Jerome Glisse0c452492010-01-15 14:44:37 +01002525}
2526
2527void r600_irq_fini(struct radeon_device *rdev)
2528{
2529 r600_irq_suspend(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002530 r600_ih_ring_fini(rdev);
2531}
2532
2533int r600_irq_set(struct radeon_device *rdev)
2534{
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002535 u32 cp_int_cntl = CNTX_BUSY_INT_ENABLE | CNTX_EMPTY_INT_ENABLE;
2536 u32 mode_int = 0;
2537 u32 hpd1, hpd2, hpd3, hpd4 = 0, hpd5 = 0, hpd6 = 0;
Christian Koenigf2594932010-04-10 03:13:16 +02002538 u32 hdmi1, hdmi2;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002539
Jerome Glisse003e69f2010-01-07 15:39:14 +01002540 if (!rdev->irq.installed) {
2541 WARN(1, "Can't enable IRQ/MSI because no handler is installed.\n");
2542 return -EINVAL;
2543 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002544 /* don't enable anything if the ih is disabled */
Jerome Glisse79c2bbc2010-01-15 14:44:38 +01002545 if (!rdev->ih.enabled) {
2546 r600_disable_interrupts(rdev);
2547 /* force the active interrupt state to all disabled */
2548 r600_disable_interrupt_state(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002549 return 0;
Jerome Glisse79c2bbc2010-01-15 14:44:38 +01002550 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002551
Christian Koenigf2594932010-04-10 03:13:16 +02002552 hdmi1 = RREG32(R600_HDMI_BLOCK1 + R600_HDMI_CNTL) & ~R600_HDMI_INT_EN;
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002553 if (ASIC_IS_DCE3(rdev)) {
Christian Koenigf2594932010-04-10 03:13:16 +02002554 hdmi2 = RREG32(R600_HDMI_BLOCK3 + R600_HDMI_CNTL) & ~R600_HDMI_INT_EN;
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002555 hpd1 = RREG32(DC_HPD1_INT_CONTROL) & ~DC_HPDx_INT_EN;
2556 hpd2 = RREG32(DC_HPD2_INT_CONTROL) & ~DC_HPDx_INT_EN;
2557 hpd3 = RREG32(DC_HPD3_INT_CONTROL) & ~DC_HPDx_INT_EN;
2558 hpd4 = RREG32(DC_HPD4_INT_CONTROL) & ~DC_HPDx_INT_EN;
2559 if (ASIC_IS_DCE32(rdev)) {
2560 hpd5 = RREG32(DC_HPD5_INT_CONTROL) & ~DC_HPDx_INT_EN;
2561 hpd6 = RREG32(DC_HPD6_INT_CONTROL) & ~DC_HPDx_INT_EN;
2562 }
2563 } else {
Christian Koenigf2594932010-04-10 03:13:16 +02002564 hdmi2 = RREG32(R600_HDMI_BLOCK2 + R600_HDMI_CNTL) & ~R600_HDMI_INT_EN;
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002565 hpd1 = RREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL) & ~DC_HPDx_INT_EN;
2566 hpd2 = RREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL) & ~DC_HPDx_INT_EN;
2567 hpd3 = RREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL) & ~DC_HPDx_INT_EN;
2568 }
2569
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002570 if (rdev->irq.sw_int) {
2571 DRM_DEBUG("r600_irq_set: sw int\n");
2572 cp_int_cntl |= RB_INT_ENABLE;
2573 }
2574 if (rdev->irq.crtc_vblank_int[0]) {
2575 DRM_DEBUG("r600_irq_set: vblank 0\n");
2576 mode_int |= D1MODE_VBLANK_INT_MASK;
2577 }
2578 if (rdev->irq.crtc_vblank_int[1]) {
2579 DRM_DEBUG("r600_irq_set: vblank 1\n");
2580 mode_int |= D2MODE_VBLANK_INT_MASK;
2581 }
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002582 if (rdev->irq.hpd[0]) {
2583 DRM_DEBUG("r600_irq_set: hpd 1\n");
2584 hpd1 |= DC_HPDx_INT_EN;
2585 }
2586 if (rdev->irq.hpd[1]) {
2587 DRM_DEBUG("r600_irq_set: hpd 2\n");
2588 hpd2 |= DC_HPDx_INT_EN;
2589 }
2590 if (rdev->irq.hpd[2]) {
2591 DRM_DEBUG("r600_irq_set: hpd 3\n");
2592 hpd3 |= DC_HPDx_INT_EN;
2593 }
2594 if (rdev->irq.hpd[3]) {
2595 DRM_DEBUG("r600_irq_set: hpd 4\n");
2596 hpd4 |= DC_HPDx_INT_EN;
2597 }
2598 if (rdev->irq.hpd[4]) {
2599 DRM_DEBUG("r600_irq_set: hpd 5\n");
2600 hpd5 |= DC_HPDx_INT_EN;
2601 }
2602 if (rdev->irq.hpd[5]) {
2603 DRM_DEBUG("r600_irq_set: hpd 6\n");
2604 hpd6 |= DC_HPDx_INT_EN;
2605 }
Christian Koenigf2594932010-04-10 03:13:16 +02002606 if (rdev->irq.hdmi[0]) {
2607 DRM_DEBUG("r600_irq_set: hdmi 1\n");
2608 hdmi1 |= R600_HDMI_INT_EN;
2609 }
2610 if (rdev->irq.hdmi[1]) {
2611 DRM_DEBUG("r600_irq_set: hdmi 2\n");
2612 hdmi2 |= R600_HDMI_INT_EN;
2613 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002614
2615 WREG32(CP_INT_CNTL, cp_int_cntl);
2616 WREG32(DxMODE_INT_MASK, mode_int);
Christian Koenigf2594932010-04-10 03:13:16 +02002617 WREG32(R600_HDMI_BLOCK1 + R600_HDMI_CNTL, hdmi1);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002618 if (ASIC_IS_DCE3(rdev)) {
Christian Koenigf2594932010-04-10 03:13:16 +02002619 WREG32(R600_HDMI_BLOCK3 + R600_HDMI_CNTL, hdmi2);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002620 WREG32(DC_HPD1_INT_CONTROL, hpd1);
2621 WREG32(DC_HPD2_INT_CONTROL, hpd2);
2622 WREG32(DC_HPD3_INT_CONTROL, hpd3);
2623 WREG32(DC_HPD4_INT_CONTROL, hpd4);
2624 if (ASIC_IS_DCE32(rdev)) {
2625 WREG32(DC_HPD5_INT_CONTROL, hpd5);
2626 WREG32(DC_HPD6_INT_CONTROL, hpd6);
2627 }
2628 } else {
Christian Koenigf2594932010-04-10 03:13:16 +02002629 WREG32(R600_HDMI_BLOCK2 + R600_HDMI_CNTL, hdmi2);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002630 WREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL, hpd1);
2631 WREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL, hpd2);
2632 WREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL, hpd3);
2633 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002634
2635 return 0;
2636}
2637
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002638static inline void r600_irq_ack(struct radeon_device *rdev,
2639 u32 *disp_int,
2640 u32 *disp_int_cont,
2641 u32 *disp_int_cont2)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002642{
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002643 u32 tmp;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002644
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002645 if (ASIC_IS_DCE3(rdev)) {
2646 *disp_int = RREG32(DCE3_DISP_INTERRUPT_STATUS);
2647 *disp_int_cont = RREG32(DCE3_DISP_INTERRUPT_STATUS_CONTINUE);
2648 *disp_int_cont2 = RREG32(DCE3_DISP_INTERRUPT_STATUS_CONTINUE2);
2649 } else {
2650 *disp_int = RREG32(DISP_INTERRUPT_STATUS);
2651 *disp_int_cont = RREG32(DISP_INTERRUPT_STATUS_CONTINUE);
2652 *disp_int_cont2 = 0;
2653 }
2654
2655 if (*disp_int & LB_D1_VBLANK_INTERRUPT)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002656 WREG32(D1MODE_VBLANK_STATUS, DxMODE_VBLANK_ACK);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002657 if (*disp_int & LB_D1_VLINE_INTERRUPT)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002658 WREG32(D1MODE_VLINE_STATUS, DxMODE_VLINE_ACK);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002659 if (*disp_int & LB_D2_VBLANK_INTERRUPT)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002660 WREG32(D2MODE_VBLANK_STATUS, DxMODE_VBLANK_ACK);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002661 if (*disp_int & LB_D2_VLINE_INTERRUPT)
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002662 WREG32(D2MODE_VLINE_STATUS, DxMODE_VLINE_ACK);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002663 if (*disp_int & DC_HPD1_INTERRUPT) {
2664 if (ASIC_IS_DCE3(rdev)) {
2665 tmp = RREG32(DC_HPD1_INT_CONTROL);
2666 tmp |= DC_HPDx_INT_ACK;
2667 WREG32(DC_HPD1_INT_CONTROL, tmp);
2668 } else {
2669 tmp = RREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL);
2670 tmp |= DC_HPDx_INT_ACK;
2671 WREG32(DC_HOT_PLUG_DETECT1_INT_CONTROL, tmp);
2672 }
2673 }
2674 if (*disp_int & DC_HPD2_INTERRUPT) {
2675 if (ASIC_IS_DCE3(rdev)) {
2676 tmp = RREG32(DC_HPD2_INT_CONTROL);
2677 tmp |= DC_HPDx_INT_ACK;
2678 WREG32(DC_HPD2_INT_CONTROL, tmp);
2679 } else {
2680 tmp = RREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL);
2681 tmp |= DC_HPDx_INT_ACK;
2682 WREG32(DC_HOT_PLUG_DETECT2_INT_CONTROL, tmp);
2683 }
2684 }
2685 if (*disp_int_cont & DC_HPD3_INTERRUPT) {
2686 if (ASIC_IS_DCE3(rdev)) {
2687 tmp = RREG32(DC_HPD3_INT_CONTROL);
2688 tmp |= DC_HPDx_INT_ACK;
2689 WREG32(DC_HPD3_INT_CONTROL, tmp);
2690 } else {
2691 tmp = RREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL);
2692 tmp |= DC_HPDx_INT_ACK;
2693 WREG32(DC_HOT_PLUG_DETECT3_INT_CONTROL, tmp);
2694 }
2695 }
2696 if (*disp_int_cont & DC_HPD4_INTERRUPT) {
2697 tmp = RREG32(DC_HPD4_INT_CONTROL);
2698 tmp |= DC_HPDx_INT_ACK;
2699 WREG32(DC_HPD4_INT_CONTROL, tmp);
2700 }
2701 if (ASIC_IS_DCE32(rdev)) {
2702 if (*disp_int_cont2 & DC_HPD5_INTERRUPT) {
2703 tmp = RREG32(DC_HPD5_INT_CONTROL);
2704 tmp |= DC_HPDx_INT_ACK;
2705 WREG32(DC_HPD5_INT_CONTROL, tmp);
2706 }
2707 if (*disp_int_cont2 & DC_HPD6_INTERRUPT) {
2708 tmp = RREG32(DC_HPD5_INT_CONTROL);
2709 tmp |= DC_HPDx_INT_ACK;
2710 WREG32(DC_HPD6_INT_CONTROL, tmp);
2711 }
2712 }
Christian Koenigf2594932010-04-10 03:13:16 +02002713 if (RREG32(R600_HDMI_BLOCK1 + R600_HDMI_STATUS) & R600_HDMI_INT_PENDING) {
2714 WREG32_P(R600_HDMI_BLOCK1 + R600_HDMI_CNTL, R600_HDMI_INT_ACK, ~R600_HDMI_INT_ACK);
2715 }
2716 if (ASIC_IS_DCE3(rdev)) {
2717 if (RREG32(R600_HDMI_BLOCK3 + R600_HDMI_STATUS) & R600_HDMI_INT_PENDING) {
2718 WREG32_P(R600_HDMI_BLOCK3 + R600_HDMI_CNTL, R600_HDMI_INT_ACK, ~R600_HDMI_INT_ACK);
2719 }
2720 } else {
2721 if (RREG32(R600_HDMI_BLOCK2 + R600_HDMI_STATUS) & R600_HDMI_INT_PENDING) {
2722 WREG32_P(R600_HDMI_BLOCK2 + R600_HDMI_CNTL, R600_HDMI_INT_ACK, ~R600_HDMI_INT_ACK);
2723 }
2724 }
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002725}
2726
2727void r600_irq_disable(struct radeon_device *rdev)
2728{
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002729 u32 disp_int, disp_int_cont, disp_int_cont2;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002730
2731 r600_disable_interrupts(rdev);
2732 /* Wait and acknowledge irq */
2733 mdelay(1);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002734 r600_irq_ack(rdev, &disp_int, &disp_int_cont, &disp_int_cont2);
2735 r600_disable_interrupt_state(rdev);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002736}
2737
2738static inline u32 r600_get_ih_wptr(struct radeon_device *rdev)
2739{
2740 u32 wptr, tmp;
2741
2742 /* XXX use writeback */
2743 wptr = RREG32(IH_RB_WPTR);
2744
2745 if (wptr & RB_OVERFLOW) {
Jerome Glisse7924e5e2010-01-15 14:44:39 +01002746 /* When a ring buffer overflow happen start parsing interrupt
2747 * from the last not overwritten vector (wptr + 16). Hopefully
2748 * this should allow us to catchup.
2749 */
2750 dev_warn(rdev->dev, "IH ring buffer overflow (0x%08X, %d, %d)\n",
2751 wptr, rdev->ih.rptr, (wptr + 16) + rdev->ih.ptr_mask);
2752 rdev->ih.rptr = (wptr + 16) & rdev->ih.ptr_mask;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002753 tmp = RREG32(IH_RB_CNTL);
2754 tmp |= IH_WPTR_OVERFLOW_CLEAR;
2755 WREG32(IH_RB_CNTL, tmp);
2756 }
Jerome Glisse0c452492010-01-15 14:44:37 +01002757 return (wptr & rdev->ih.ptr_mask);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002758}
2759
2760/* r600 IV Ring
2761 * Each IV ring entry is 128 bits:
2762 * [7:0] - interrupt source id
2763 * [31:8] - reserved
2764 * [59:32] - interrupt source data
2765 * [127:60] - reserved
2766 *
2767 * The basic interrupt vector entries
2768 * are decoded as follows:
2769 * src_id src_data description
2770 * 1 0 D1 Vblank
2771 * 1 1 D1 Vline
2772 * 5 0 D2 Vblank
2773 * 5 1 D2 Vline
2774 * 19 0 FP Hot plug detection A
2775 * 19 1 FP Hot plug detection B
2776 * 19 2 DAC A auto-detection
2777 * 19 3 DAC B auto-detection
Christian Koenigf2594932010-04-10 03:13:16 +02002778 * 21 4 HDMI block A
2779 * 21 5 HDMI block B
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002780 * 176 - CP_INT RB
2781 * 177 - CP_INT IB1
2782 * 178 - CP_INT IB2
2783 * 181 - EOP Interrupt
2784 * 233 - GUI Idle
2785 *
2786 * Note, these are based on r600 and may need to be
2787 * adjusted or added to on newer asics
2788 */
2789
2790int r600_irq_process(struct radeon_device *rdev)
2791{
2792 u32 wptr = r600_get_ih_wptr(rdev);
2793 u32 rptr = rdev->ih.rptr;
2794 u32 src_id, src_data;
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002795 u32 ring_index, disp_int, disp_int_cont, disp_int_cont2;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002796 unsigned long flags;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002797 bool queue_hotplug = false;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002798
2799 DRM_DEBUG("r600_irq_process start: rptr %d, wptr %d\n", rptr, wptr);
Jerome Glisse79c2bbc2010-01-15 14:44:38 +01002800 if (!rdev->ih.enabled)
2801 return IRQ_NONE;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002802
2803 spin_lock_irqsave(&rdev->ih.lock, flags);
2804
2805 if (rptr == wptr) {
2806 spin_unlock_irqrestore(&rdev->ih.lock, flags);
2807 return IRQ_NONE;
2808 }
2809 if (rdev->shutdown) {
2810 spin_unlock_irqrestore(&rdev->ih.lock, flags);
2811 return IRQ_NONE;
2812 }
2813
2814restart_ih:
2815 /* display interrupts */
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002816 r600_irq_ack(rdev, &disp_int, &disp_int_cont, &disp_int_cont2);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002817
2818 rdev->ih.wptr = wptr;
2819 while (rptr != wptr) {
2820 /* wptr/rptr are in bytes! */
2821 ring_index = rptr / 4;
2822 src_id = rdev->ih.ring[ring_index] & 0xff;
2823 src_data = rdev->ih.ring[ring_index + 1] & 0xfffffff;
2824
2825 switch (src_id) {
2826 case 1: /* D1 vblank/vline */
2827 switch (src_data) {
2828 case 0: /* D1 vblank */
2829 if (disp_int & LB_D1_VBLANK_INTERRUPT) {
2830 drm_handle_vblank(rdev->ddev, 0);
Rafał Miłecki839461d2010-03-02 22:06:51 +01002831 rdev->pm.vblank_sync = true;
Rafał Miłecki73a6d3f2010-01-08 00:22:47 +01002832 wake_up(&rdev->irq.vblank_queue);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002833 disp_int &= ~LB_D1_VBLANK_INTERRUPT;
2834 DRM_DEBUG("IH: D1 vblank\n");
2835 }
2836 break;
2837 case 1: /* D1 vline */
2838 if (disp_int & LB_D1_VLINE_INTERRUPT) {
2839 disp_int &= ~LB_D1_VLINE_INTERRUPT;
2840 DRM_DEBUG("IH: D1 vline\n");
2841 }
2842 break;
2843 default:
Alex Deucherb0425892010-01-11 19:47:38 -05002844 DRM_DEBUG("Unhandled interrupt: %d %d\n", src_id, src_data);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002845 break;
2846 }
2847 break;
2848 case 5: /* D2 vblank/vline */
2849 switch (src_data) {
2850 case 0: /* D2 vblank */
2851 if (disp_int & LB_D2_VBLANK_INTERRUPT) {
2852 drm_handle_vblank(rdev->ddev, 1);
Rafał Miłecki839461d2010-03-02 22:06:51 +01002853 rdev->pm.vblank_sync = true;
Rafał Miłecki73a6d3f2010-01-08 00:22:47 +01002854 wake_up(&rdev->irq.vblank_queue);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002855 disp_int &= ~LB_D2_VBLANK_INTERRUPT;
2856 DRM_DEBUG("IH: D2 vblank\n");
2857 }
2858 break;
2859 case 1: /* D1 vline */
2860 if (disp_int & LB_D2_VLINE_INTERRUPT) {
2861 disp_int &= ~LB_D2_VLINE_INTERRUPT;
2862 DRM_DEBUG("IH: D2 vline\n");
2863 }
2864 break;
2865 default:
Alex Deucherb0425892010-01-11 19:47:38 -05002866 DRM_DEBUG("Unhandled interrupt: %d %d\n", src_id, src_data);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002867 break;
2868 }
2869 break;
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002870 case 19: /* HPD/DAC hotplug */
2871 switch (src_data) {
2872 case 0:
2873 if (disp_int & DC_HPD1_INTERRUPT) {
2874 disp_int &= ~DC_HPD1_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002875 queue_hotplug = true;
2876 DRM_DEBUG("IH: HPD1\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002877 }
2878 break;
2879 case 1:
2880 if (disp_int & DC_HPD2_INTERRUPT) {
2881 disp_int &= ~DC_HPD2_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002882 queue_hotplug = true;
2883 DRM_DEBUG("IH: HPD2\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002884 }
2885 break;
2886 case 4:
2887 if (disp_int_cont & DC_HPD3_INTERRUPT) {
2888 disp_int_cont &= ~DC_HPD3_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002889 queue_hotplug = true;
2890 DRM_DEBUG("IH: HPD3\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002891 }
2892 break;
2893 case 5:
2894 if (disp_int_cont & DC_HPD4_INTERRUPT) {
2895 disp_int_cont &= ~DC_HPD4_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002896 queue_hotplug = true;
2897 DRM_DEBUG("IH: HPD4\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002898 }
2899 break;
2900 case 10:
2901 if (disp_int_cont2 & DC_HPD5_INTERRUPT) {
Alex Deucher5898b1f2010-03-24 13:57:29 -04002902 disp_int_cont2 &= ~DC_HPD5_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002903 queue_hotplug = true;
2904 DRM_DEBUG("IH: HPD5\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002905 }
2906 break;
2907 case 12:
2908 if (disp_int_cont2 & DC_HPD6_INTERRUPT) {
Alex Deucher5898b1f2010-03-24 13:57:29 -04002909 disp_int_cont2 &= ~DC_HPD6_INTERRUPT;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002910 queue_hotplug = true;
2911 DRM_DEBUG("IH: HPD6\n");
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002912 }
2913 break;
2914 default:
Alex Deucherb0425892010-01-11 19:47:38 -05002915 DRM_DEBUG("Unhandled interrupt: %d %d\n", src_id, src_data);
Alex Deuchere0df1ac2009-12-04 15:12:21 -05002916 break;
2917 }
2918 break;
Christian Koenigf2594932010-04-10 03:13:16 +02002919 case 21: /* HDMI */
2920 DRM_DEBUG("IH: HDMI: 0x%x\n", src_data);
2921 r600_audio_schedule_polling(rdev);
2922 break;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002923 case 176: /* CP_INT in ring buffer */
2924 case 177: /* CP_INT in IB1 */
2925 case 178: /* CP_INT in IB2 */
2926 DRM_DEBUG("IH: CP int: 0x%08x\n", src_data);
2927 radeon_fence_process(rdev);
2928 break;
2929 case 181: /* CP EOP event */
2930 DRM_DEBUG("IH: CP EOP\n");
2931 break;
2932 default:
Alex Deucherb0425892010-01-11 19:47:38 -05002933 DRM_DEBUG("Unhandled interrupt: %d %d\n", src_id, src_data);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002934 break;
2935 }
2936
2937 /* wptr/rptr are in bytes! */
Jerome Glisse0c452492010-01-15 14:44:37 +01002938 rptr += 16;
2939 rptr &= rdev->ih.ptr_mask;
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002940 }
2941 /* make sure wptr hasn't changed while processing */
2942 wptr = r600_get_ih_wptr(rdev);
2943 if (wptr != rdev->ih.wptr)
2944 goto restart_ih;
Alex Deucherd4877cf2009-12-04 16:56:37 -05002945 if (queue_hotplug)
2946 queue_work(rdev->wq, &rdev->hotplug_work);
Alex Deucherd8f60cf2009-12-01 13:43:46 -05002947 rdev->ih.rptr = rptr;
2948 WREG32(IH_RB_RPTR, rdev->ih.rptr);
2949 spin_unlock_irqrestore(&rdev->ih.lock, flags);
2950 return IRQ_HANDLED;
2951}
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002952
2953/*
2954 * Debugfs info
2955 */
2956#if defined(CONFIG_DEBUG_FS)
2957
2958static int r600_debugfs_cp_ring_info(struct seq_file *m, void *data)
2959{
2960 struct drm_info_node *node = (struct drm_info_node *) m->private;
2961 struct drm_device *dev = node->minor->dev;
2962 struct radeon_device *rdev = dev->dev_private;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002963 unsigned count, i, j;
2964
2965 radeon_ring_free_size(rdev);
Rafał Miłeckid6840762009-11-10 22:26:21 +01002966 count = (rdev->cp.ring_size / 4) - rdev->cp.ring_free_dw;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002967 seq_printf(m, "CP_STAT 0x%08x\n", RREG32(CP_STAT));
Rafał Miłeckid6840762009-11-10 22:26:21 +01002968 seq_printf(m, "CP_RB_WPTR 0x%08x\n", RREG32(CP_RB_WPTR));
2969 seq_printf(m, "CP_RB_RPTR 0x%08x\n", RREG32(CP_RB_RPTR));
2970 seq_printf(m, "driver's copy of the CP_RB_WPTR 0x%08x\n", rdev->cp.wptr);
2971 seq_printf(m, "driver's copy of the CP_RB_RPTR 0x%08x\n", rdev->cp.rptr);
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002972 seq_printf(m, "%u free dwords in ring\n", rdev->cp.ring_free_dw);
2973 seq_printf(m, "%u dwords in ring\n", count);
Rafał Miłeckid6840762009-11-10 22:26:21 +01002974 i = rdev->cp.rptr;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002975 for (j = 0; j <= count; j++) {
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002976 seq_printf(m, "r[%04d]=0x%08x\n", i, rdev->cp.ring[i]);
Rafał Miłeckid6840762009-11-10 22:26:21 +01002977 i = (i + 1) & rdev->cp.ptr_mask;
Jerome Glisse3ce0a232009-09-08 10:10:24 +10002978 }
2979 return 0;
2980}
2981
2982static int r600_debugfs_mc_info(struct seq_file *m, void *data)
2983{
2984 struct drm_info_node *node = (struct drm_info_node *) m->private;
2985 struct drm_device *dev = node->minor->dev;
2986 struct radeon_device *rdev = dev->dev_private;
2987
2988 DREG32_SYS(m, rdev, R_000E50_SRBM_STATUS);
2989 DREG32_SYS(m, rdev, VM_L2_STATUS);
2990 return 0;
2991}
2992
2993static struct drm_info_list r600_mc_info_list[] = {
2994 {"r600_mc_info", r600_debugfs_mc_info, 0, NULL},
2995 {"r600_ring_info", r600_debugfs_cp_ring_info, 0, NULL},
2996};
2997#endif
2998
2999int r600_debugfs_mc_info_init(struct radeon_device *rdev)
3000{
3001#if defined(CONFIG_DEBUG_FS)
3002 return radeon_debugfs_add_files(rdev, r600_mc_info_list, ARRAY_SIZE(r600_mc_info_list));
3003#else
3004 return 0;
3005#endif
Jerome Glisse771fe6b2009-06-05 14:42:42 +02003006}
Jerome Glisse062b3892010-02-04 20:36:39 +01003007
3008/**
3009 * r600_ioctl_wait_idle - flush host path cache on wait idle ioctl
3010 * rdev: radeon device structure
3011 * bo: buffer object struct which userspace is waiting for idle
3012 *
3013 * Some R6XX/R7XX doesn't seems to take into account HDP flush performed
3014 * through ring buffer, this leads to corruption in rendering, see
3015 * http://bugzilla.kernel.org/show_bug.cgi?id=15186 to avoid this we
3016 * directly perform HDP flush by writing register through MMIO.
3017 */
3018void r600_ioctl_wait_idle(struct radeon_device *rdev, struct radeon_bo *bo)
3019{
3020 WREG32(R_005480_HDP_MEM_COHERENCY_FLUSH_CNTL, 0x1);
3021}