xref: /openbmc/linux/drivers/gpu/drm/msm/adreno/a4xx_gpu.c (revision 7b73a9c8)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /* Copyright (c) 2014 The Linux Foundation. All rights reserved.
3  */
4 #include "a4xx_gpu.h"
5 
6 #define A4XX_INT0_MASK \
7 	(A4XX_INT0_RBBM_AHB_ERROR |        \
8 	 A4XX_INT0_RBBM_ATB_BUS_OVERFLOW | \
9 	 A4XX_INT0_CP_T0_PACKET_IN_IB |    \
10 	 A4XX_INT0_CP_OPCODE_ERROR |       \
11 	 A4XX_INT0_CP_RESERVED_BIT_ERROR | \
12 	 A4XX_INT0_CP_HW_FAULT |           \
13 	 A4XX_INT0_CP_IB1_INT |            \
14 	 A4XX_INT0_CP_IB2_INT |            \
15 	 A4XX_INT0_CP_RB_INT |             \
16 	 A4XX_INT0_CP_REG_PROTECT_FAULT |  \
17 	 A4XX_INT0_CP_AHB_ERROR_HALT |     \
18 	 A4XX_INT0_CACHE_FLUSH_TS |        \
19 	 A4XX_INT0_UCHE_OOB_ACCESS)
20 
21 extern bool hang_debug;
22 static void a4xx_dump(struct msm_gpu *gpu);
23 static bool a4xx_idle(struct msm_gpu *gpu);
24 
25 /*
26  * a4xx_enable_hwcg() - Program the clock control registers
27  * @device: The adreno device pointer
28  */
29 static void a4xx_enable_hwcg(struct msm_gpu *gpu)
30 {
31 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
32 	unsigned int i;
33 	for (i = 0; i < 4; i++)
34 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_TP(i), 0x02222202);
35 	for (i = 0; i < 4; i++)
36 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2_TP(i), 0x00002222);
37 	for (i = 0; i < 4; i++)
38 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_TP(i), 0x0E739CE7);
39 	for (i = 0; i < 4; i++)
40 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_TP(i), 0x00111111);
41 	for (i = 0; i < 4; i++)
42 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_SP(i), 0x22222222);
43 	for (i = 0; i < 4; i++)
44 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2_SP(i), 0x00222222);
45 	for (i = 0; i < 4; i++)
46 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_SP(i), 0x00000104);
47 	for (i = 0; i < 4; i++)
48 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_SP(i), 0x00000081);
49 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_UCHE, 0x22222222);
50 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2_UCHE, 0x02222222);
51 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL3_UCHE, 0x00000000);
52 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL4_UCHE, 0x00000000);
53 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_UCHE, 0x00004444);
54 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_UCHE, 0x00001112);
55 	for (i = 0; i < 4; i++)
56 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_RB(i), 0x22222222);
57 
58 	/* Disable L1 clocking in A420 due to CCU issues with it */
59 	for (i = 0; i < 4; i++) {
60 		if (adreno_is_a420(adreno_gpu)) {
61 			gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2_RB(i),
62 					0x00002020);
63 		} else {
64 			gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2_RB(i),
65 					0x00022020);
66 		}
67 	}
68 
69 	for (i = 0; i < 4; i++) {
70 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_MARB_CCU(i),
71 				0x00000922);
72 	}
73 
74 	for (i = 0; i < 4; i++) {
75 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_RB_MARB_CCU(i),
76 				0x00000000);
77 	}
78 
79 	for (i = 0; i < 4; i++) {
80 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_RB_MARB_CCU_L1(i),
81 				0x00000001);
82 	}
83 
84 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_MODE_GPC, 0x02222222);
85 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_GPC, 0x04100104);
86 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_GPC, 0x00022222);
87 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_COM_DCOM, 0x00000022);
88 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_COM_DCOM, 0x0000010F);
89 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_COM_DCOM, 0x00000022);
90 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_TSE_RAS_RBBM, 0x00222222);
91 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_TSE_RAS_RBBM, 0x00004104);
92 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_TSE_RAS_RBBM, 0x00000222);
93 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL_HLSQ , 0x00000000);
94 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_HYST_HLSQ, 0x00000000);
95 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_HLSQ, 0x00220000);
96 	/* Early A430's have a timing issue with SP/TP power collapse;
97 	   disabling HW clock gating prevents it. */
98 	if (adreno_is_a430(adreno_gpu) && adreno_gpu->rev.patchid < 2)
99 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL, 0);
100 	else
101 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL, 0xAAAAAAAA);
102 	gpu_write(gpu, REG_A4XX_RBBM_CLOCK_CTL2, 0);
103 }
104 
105 
106 static bool a4xx_me_init(struct msm_gpu *gpu)
107 {
108 	struct msm_ringbuffer *ring = gpu->rb[0];
109 
110 	OUT_PKT3(ring, CP_ME_INIT, 17);
111 	OUT_RING(ring, 0x000003f7);
112 	OUT_RING(ring, 0x00000000);
113 	OUT_RING(ring, 0x00000000);
114 	OUT_RING(ring, 0x00000000);
115 	OUT_RING(ring, 0x00000080);
116 	OUT_RING(ring, 0x00000100);
117 	OUT_RING(ring, 0x00000180);
118 	OUT_RING(ring, 0x00006600);
119 	OUT_RING(ring, 0x00000150);
120 	OUT_RING(ring, 0x0000014e);
121 	OUT_RING(ring, 0x00000154);
122 	OUT_RING(ring, 0x00000001);
123 	OUT_RING(ring, 0x00000000);
124 	OUT_RING(ring, 0x00000000);
125 	OUT_RING(ring, 0x00000000);
126 	OUT_RING(ring, 0x00000000);
127 	OUT_RING(ring, 0x00000000);
128 
129 	gpu->funcs->flush(gpu, ring);
130 	return a4xx_idle(gpu);
131 }
132 
133 static int a4xx_hw_init(struct msm_gpu *gpu)
134 {
135 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
136 	struct a4xx_gpu *a4xx_gpu = to_a4xx_gpu(adreno_gpu);
137 	uint32_t *ptr, len;
138 	int i, ret;
139 
140 	if (adreno_is_a420(adreno_gpu)) {
141 		gpu_write(gpu, REG_A4XX_VBIF_ABIT_SORT, 0x0001001F);
142 		gpu_write(gpu, REG_A4XX_VBIF_ABIT_SORT_CONF, 0x000000A4);
143 		gpu_write(gpu, REG_A4XX_VBIF_GATE_OFF_WRREQ_EN, 0x00000001);
144 		gpu_write(gpu, REG_A4XX_VBIF_IN_RD_LIM_CONF0, 0x18181818);
145 		gpu_write(gpu, REG_A4XX_VBIF_IN_RD_LIM_CONF1, 0x00000018);
146 		gpu_write(gpu, REG_A4XX_VBIF_IN_WR_LIM_CONF0, 0x18181818);
147 		gpu_write(gpu, REG_A4XX_VBIF_IN_WR_LIM_CONF1, 0x00000018);
148 		gpu_write(gpu, REG_A4XX_VBIF_ROUND_ROBIN_QOS_ARB, 0x00000003);
149 	} else if (adreno_is_a430(adreno_gpu)) {
150 		gpu_write(gpu, REG_A4XX_VBIF_GATE_OFF_WRREQ_EN, 0x00000001);
151 		gpu_write(gpu, REG_A4XX_VBIF_IN_RD_LIM_CONF0, 0x18181818);
152 		gpu_write(gpu, REG_A4XX_VBIF_IN_RD_LIM_CONF1, 0x00000018);
153 		gpu_write(gpu, REG_A4XX_VBIF_IN_WR_LIM_CONF0, 0x18181818);
154 		gpu_write(gpu, REG_A4XX_VBIF_IN_WR_LIM_CONF1, 0x00000018);
155 		gpu_write(gpu, REG_A4XX_VBIF_ROUND_ROBIN_QOS_ARB, 0x00000003);
156 	} else {
157 		BUG();
158 	}
159 
160 	/* Make all blocks contribute to the GPU BUSY perf counter */
161 	gpu_write(gpu, REG_A4XX_RBBM_GPU_BUSY_MASKED, 0xffffffff);
162 
163 	/* Tune the hystersis counters for SP and CP idle detection */
164 	gpu_write(gpu, REG_A4XX_RBBM_SP_HYST_CNT, 0x10);
165 	gpu_write(gpu, REG_A4XX_RBBM_WAIT_IDLE_CLOCKS_CTL, 0x10);
166 
167 	if (adreno_is_a430(adreno_gpu)) {
168 		gpu_write(gpu, REG_A4XX_RBBM_WAIT_IDLE_CLOCKS_CTL2, 0x30);
169 	}
170 
171 	 /* Enable the RBBM error reporting bits */
172 	gpu_write(gpu, REG_A4XX_RBBM_AHB_CTL0, 0x00000001);
173 
174 	/* Enable AHB error reporting*/
175 	gpu_write(gpu, REG_A4XX_RBBM_AHB_CTL1, 0xa6ffffff);
176 
177 	/* Enable power counters*/
178 	gpu_write(gpu, REG_A4XX_RBBM_RBBM_CTL, 0x00000030);
179 
180 	/*
181 	 * Turn on hang detection - this spews a lot of useful information
182 	 * into the RBBM registers on a hang:
183 	 */
184 	gpu_write(gpu, REG_A4XX_RBBM_INTERFACE_HANG_INT_CTL,
185 			(1 << 30) | 0xFFFF);
186 
187 	gpu_write(gpu, REG_A4XX_RB_GMEM_BASE_ADDR,
188 			(unsigned int)(a4xx_gpu->ocmem.base >> 14));
189 
190 	/* Turn on performance counters: */
191 	gpu_write(gpu, REG_A4XX_RBBM_PERFCTR_CTL, 0x01);
192 
193 	/* use the first CP counter for timestamp queries.. userspace may set
194 	 * this as well but it selects the same counter/countable:
195 	 */
196 	gpu_write(gpu, REG_A4XX_CP_PERFCTR_CP_SEL_0, CP_ALWAYS_COUNT);
197 
198 	if (adreno_is_a430(adreno_gpu))
199 		gpu_write(gpu, REG_A4XX_UCHE_CACHE_WAYS_VFD, 0x07);
200 
201 	/* Disable L2 bypass to avoid UCHE out of bounds errors */
202 	gpu_write(gpu, REG_A4XX_UCHE_TRAP_BASE_LO, 0xffff0000);
203 	gpu_write(gpu, REG_A4XX_UCHE_TRAP_BASE_HI, 0xffff0000);
204 
205 	gpu_write(gpu, REG_A4XX_CP_DEBUG, (1 << 25) |
206 			(adreno_is_a420(adreno_gpu) ? (1 << 29) : 0));
207 
208 	/* On A430 enable SP regfile sleep for power savings */
209 	/* TODO downstream does this for !420, so maybe applies for 405 too? */
210 	if (!adreno_is_a420(adreno_gpu)) {
211 		gpu_write(gpu, REG_A4XX_RBBM_SP_REGFILE_SLEEP_CNTL_0,
212 			0x00000441);
213 		gpu_write(gpu, REG_A4XX_RBBM_SP_REGFILE_SLEEP_CNTL_1,
214 			0x00000441);
215 	}
216 
217 	a4xx_enable_hwcg(gpu);
218 
219 	/*
220 	 * For A420 set RBBM_CLOCK_DELAY_HLSQ.CGC_HLSQ_TP_EARLY_CYC >= 2
221 	 * due to timing issue with HLSQ_TP_CLK_EN
222 	 */
223 	if (adreno_is_a420(adreno_gpu)) {
224 		unsigned int val;
225 		val = gpu_read(gpu, REG_A4XX_RBBM_CLOCK_DELAY_HLSQ);
226 		val &= ~A4XX_CGC_HLSQ_EARLY_CYC__MASK;
227 		val |= 2 << A4XX_CGC_HLSQ_EARLY_CYC__SHIFT;
228 		gpu_write(gpu, REG_A4XX_RBBM_CLOCK_DELAY_HLSQ, val);
229 	}
230 
231 	/* setup access protection: */
232 	gpu_write(gpu, REG_A4XX_CP_PROTECT_CTRL, 0x00000007);
233 
234 	/* RBBM registers */
235 	gpu_write(gpu, REG_A4XX_CP_PROTECT(0), 0x62000010);
236 	gpu_write(gpu, REG_A4XX_CP_PROTECT(1), 0x63000020);
237 	gpu_write(gpu, REG_A4XX_CP_PROTECT(2), 0x64000040);
238 	gpu_write(gpu, REG_A4XX_CP_PROTECT(3), 0x65000080);
239 	gpu_write(gpu, REG_A4XX_CP_PROTECT(4), 0x66000100);
240 	gpu_write(gpu, REG_A4XX_CP_PROTECT(5), 0x64000200);
241 
242 	/* CP registers */
243 	gpu_write(gpu, REG_A4XX_CP_PROTECT(6), 0x67000800);
244 	gpu_write(gpu, REG_A4XX_CP_PROTECT(7), 0x64001600);
245 
246 
247 	/* RB registers */
248 	gpu_write(gpu, REG_A4XX_CP_PROTECT(8), 0x60003300);
249 
250 	/* HLSQ registers */
251 	gpu_write(gpu, REG_A4XX_CP_PROTECT(9), 0x60003800);
252 
253 	/* VPC registers */
254 	gpu_write(gpu, REG_A4XX_CP_PROTECT(10), 0x61003980);
255 
256 	/* SMMU registers */
257 	gpu_write(gpu, REG_A4XX_CP_PROTECT(11), 0x6e010000);
258 
259 	gpu_write(gpu, REG_A4XX_RBBM_INT_0_MASK, A4XX_INT0_MASK);
260 
261 	ret = adreno_hw_init(gpu);
262 	if (ret)
263 		return ret;
264 
265 	/* Load PM4: */
266 	ptr = (uint32_t *)(adreno_gpu->fw[ADRENO_FW_PM4]->data);
267 	len = adreno_gpu->fw[ADRENO_FW_PM4]->size / 4;
268 	DBG("loading PM4 ucode version: %u", ptr[0]);
269 	gpu_write(gpu, REG_A4XX_CP_ME_RAM_WADDR, 0);
270 	for (i = 1; i < len; i++)
271 		gpu_write(gpu, REG_A4XX_CP_ME_RAM_DATA, ptr[i]);
272 
273 	/* Load PFP: */
274 	ptr = (uint32_t *)(adreno_gpu->fw[ADRENO_FW_PFP]->data);
275 	len = adreno_gpu->fw[ADRENO_FW_PFP]->size / 4;
276 	DBG("loading PFP ucode version: %u", ptr[0]);
277 
278 	gpu_write(gpu, REG_A4XX_CP_PFP_UCODE_ADDR, 0);
279 	for (i = 1; i < len; i++)
280 		gpu_write(gpu, REG_A4XX_CP_PFP_UCODE_DATA, ptr[i]);
281 
282 	/* clear ME_HALT to start micro engine */
283 	gpu_write(gpu, REG_A4XX_CP_ME_CNTL, 0);
284 
285 	return a4xx_me_init(gpu) ? 0 : -EINVAL;
286 }
287 
288 static void a4xx_recover(struct msm_gpu *gpu)
289 {
290 	int i;
291 
292 	adreno_dump_info(gpu);
293 
294 	for (i = 0; i < 8; i++) {
295 		printk("CP_SCRATCH_REG%d: %u\n", i,
296 			gpu_read(gpu, REG_AXXX_CP_SCRATCH_REG0 + i));
297 	}
298 
299 	/* dump registers before resetting gpu, if enabled: */
300 	if (hang_debug)
301 		a4xx_dump(gpu);
302 
303 	gpu_write(gpu, REG_A4XX_RBBM_SW_RESET_CMD, 1);
304 	gpu_read(gpu, REG_A4XX_RBBM_SW_RESET_CMD);
305 	gpu_write(gpu, REG_A4XX_RBBM_SW_RESET_CMD, 0);
306 	adreno_recover(gpu);
307 }
308 
309 static void a4xx_destroy(struct msm_gpu *gpu)
310 {
311 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
312 	struct a4xx_gpu *a4xx_gpu = to_a4xx_gpu(adreno_gpu);
313 
314 	DBG("%s", gpu->name);
315 
316 	adreno_gpu_cleanup(adreno_gpu);
317 
318 	adreno_gpu_ocmem_cleanup(&a4xx_gpu->ocmem);
319 
320 	kfree(a4xx_gpu);
321 }
322 
323 static bool a4xx_idle(struct msm_gpu *gpu)
324 {
325 	/* wait for ringbuffer to drain: */
326 	if (!adreno_idle(gpu, gpu->rb[0]))
327 		return false;
328 
329 	/* then wait for GPU to finish: */
330 	if (spin_until(!(gpu_read(gpu, REG_A4XX_RBBM_STATUS) &
331 					A4XX_RBBM_STATUS_GPU_BUSY))) {
332 		DRM_ERROR("%s: timeout waiting for GPU to idle!\n", gpu->name);
333 		/* TODO maybe we need to reset GPU here to recover from hang? */
334 		return false;
335 	}
336 
337 	return true;
338 }
339 
340 static irqreturn_t a4xx_irq(struct msm_gpu *gpu)
341 {
342 	uint32_t status;
343 
344 	status = gpu_read(gpu, REG_A4XX_RBBM_INT_0_STATUS);
345 	DBG("%s: Int status %08x", gpu->name, status);
346 
347 	if (status & A4XX_INT0_CP_REG_PROTECT_FAULT) {
348 		uint32_t reg = gpu_read(gpu, REG_A4XX_CP_PROTECT_STATUS);
349 		printk("CP | Protected mode error| %s | addr=%x\n",
350 			reg & (1 << 24) ? "WRITE" : "READ",
351 			(reg & 0xFFFFF) >> 2);
352 	}
353 
354 	gpu_write(gpu, REG_A4XX_RBBM_INT_CLEAR_CMD, status);
355 
356 	msm_gpu_retire(gpu);
357 
358 	return IRQ_HANDLED;
359 }
360 
361 static const unsigned int a4xx_registers[] = {
362 	/* RBBM */
363 	0x0000, 0x0002, 0x0004, 0x0021, 0x0023, 0x0024, 0x0026, 0x0026,
364 	0x0028, 0x002B, 0x002E, 0x0034, 0x0037, 0x0044, 0x0047, 0x0066,
365 	0x0068, 0x0095, 0x009C, 0x0170, 0x0174, 0x01AF,
366 	/* CP */
367 	0x0200, 0x0233, 0x0240, 0x0250, 0x04C0, 0x04DD, 0x0500, 0x050B,
368 	0x0578, 0x058F,
369 	/* VSC */
370 	0x0C00, 0x0C03, 0x0C08, 0x0C41, 0x0C50, 0x0C51,
371 	/* GRAS */
372 	0x0C80, 0x0C81, 0x0C88, 0x0C8F,
373 	/* RB */
374 	0x0CC0, 0x0CC0, 0x0CC4, 0x0CD2,
375 	/* PC */
376 	0x0D00, 0x0D0C, 0x0D10, 0x0D17, 0x0D20, 0x0D23,
377 	/* VFD */
378 	0x0E40, 0x0E4A,
379 	/* VPC */
380 	0x0E60, 0x0E61, 0x0E63, 0x0E68,
381 	/* UCHE */
382 	0x0E80, 0x0E84, 0x0E88, 0x0E95,
383 	/* VMIDMT */
384 	0x1000, 0x1000, 0x1002, 0x1002, 0x1004, 0x1004, 0x1008, 0x100A,
385 	0x100C, 0x100D, 0x100F, 0x1010, 0x1012, 0x1016, 0x1024, 0x1024,
386 	0x1027, 0x1027, 0x1100, 0x1100, 0x1102, 0x1102, 0x1104, 0x1104,
387 	0x1110, 0x1110, 0x1112, 0x1116, 0x1124, 0x1124, 0x1300, 0x1300,
388 	0x1380, 0x1380,
389 	/* GRAS CTX 0 */
390 	0x2000, 0x2004, 0x2008, 0x2067, 0x2070, 0x2078, 0x207B, 0x216E,
391 	/* PC CTX 0 */
392 	0x21C0, 0x21C6, 0x21D0, 0x21D0, 0x21D9, 0x21D9, 0x21E5, 0x21E7,
393 	/* VFD CTX 0 */
394 	0x2200, 0x2204, 0x2208, 0x22A9,
395 	/* GRAS CTX 1 */
396 	0x2400, 0x2404, 0x2408, 0x2467, 0x2470, 0x2478, 0x247B, 0x256E,
397 	/* PC CTX 1 */
398 	0x25C0, 0x25C6, 0x25D0, 0x25D0, 0x25D9, 0x25D9, 0x25E5, 0x25E7,
399 	/* VFD CTX 1 */
400 	0x2600, 0x2604, 0x2608, 0x26A9,
401 	/* XPU */
402 	0x2C00, 0x2C01, 0x2C10, 0x2C10, 0x2C12, 0x2C16, 0x2C1D, 0x2C20,
403 	0x2C28, 0x2C28, 0x2C30, 0x2C30, 0x2C32, 0x2C36, 0x2C40, 0x2C40,
404 	0x2C50, 0x2C50, 0x2C52, 0x2C56, 0x2C80, 0x2C80, 0x2C94, 0x2C95,
405 	/* VBIF */
406 	0x3000, 0x3007, 0x300C, 0x3014, 0x3018, 0x301D, 0x3020, 0x3022,
407 	0x3024, 0x3026, 0x3028, 0x302A, 0x302C, 0x302D, 0x3030, 0x3031,
408 	0x3034, 0x3036, 0x3038, 0x3038, 0x303C, 0x303D, 0x3040, 0x3040,
409 	0x3049, 0x3049, 0x3058, 0x3058, 0x305B, 0x3061, 0x3064, 0x3068,
410 	0x306C, 0x306D, 0x3080, 0x3088, 0x308B, 0x308C, 0x3090, 0x3094,
411 	0x3098, 0x3098, 0x309C, 0x309C, 0x30C0, 0x30C0, 0x30C8, 0x30C8,
412 	0x30D0, 0x30D0, 0x30D8, 0x30D8, 0x30E0, 0x30E0, 0x3100, 0x3100,
413 	0x3108, 0x3108, 0x3110, 0x3110, 0x3118, 0x3118, 0x3120, 0x3120,
414 	0x3124, 0x3125, 0x3129, 0x3129, 0x3131, 0x3131, 0x330C, 0x330C,
415 	0x3310, 0x3310, 0x3400, 0x3401, 0x3410, 0x3410, 0x3412, 0x3416,
416 	0x341D, 0x3420, 0x3428, 0x3428, 0x3430, 0x3430, 0x3432, 0x3436,
417 	0x3440, 0x3440, 0x3450, 0x3450, 0x3452, 0x3456, 0x3480, 0x3480,
418 	0x3494, 0x3495, 0x4000, 0x4000, 0x4002, 0x4002, 0x4004, 0x4004,
419 	0x4008, 0x400A, 0x400C, 0x400D, 0x400F, 0x4012, 0x4014, 0x4016,
420 	0x401D, 0x401D, 0x4020, 0x4027, 0x4060, 0x4062, 0x4200, 0x4200,
421 	0x4300, 0x4300, 0x4400, 0x4400, 0x4500, 0x4500, 0x4800, 0x4802,
422 	0x480F, 0x480F, 0x4811, 0x4811, 0x4813, 0x4813, 0x4815, 0x4816,
423 	0x482B, 0x482B, 0x4857, 0x4857, 0x4883, 0x4883, 0x48AF, 0x48AF,
424 	0x48C5, 0x48C5, 0x48E5, 0x48E5, 0x4905, 0x4905, 0x4925, 0x4925,
425 	0x4945, 0x4945, 0x4950, 0x4950, 0x495B, 0x495B, 0x4980, 0x498E,
426 	0x4B00, 0x4B00, 0x4C00, 0x4C00, 0x4D00, 0x4D00, 0x4E00, 0x4E00,
427 	0x4E80, 0x4E80, 0x4F00, 0x4F00, 0x4F08, 0x4F08, 0x4F10, 0x4F10,
428 	0x4F18, 0x4F18, 0x4F20, 0x4F20, 0x4F30, 0x4F30, 0x4F60, 0x4F60,
429 	0x4F80, 0x4F81, 0x4F88, 0x4F89, 0x4FEE, 0x4FEE, 0x4FF3, 0x4FF3,
430 	0x6000, 0x6001, 0x6008, 0x600F, 0x6014, 0x6016, 0x6018, 0x601B,
431 	0x61FD, 0x61FD, 0x623C, 0x623C, 0x6380, 0x6380, 0x63A0, 0x63A0,
432 	0x63C0, 0x63C1, 0x63C8, 0x63C9, 0x63D0, 0x63D4, 0x63D6, 0x63D6,
433 	0x63EE, 0x63EE, 0x6400, 0x6401, 0x6408, 0x640F, 0x6414, 0x6416,
434 	0x6418, 0x641B, 0x65FD, 0x65FD, 0x663C, 0x663C, 0x6780, 0x6780,
435 	0x67A0, 0x67A0, 0x67C0, 0x67C1, 0x67C8, 0x67C9, 0x67D0, 0x67D4,
436 	0x67D6, 0x67D6, 0x67EE, 0x67EE, 0x6800, 0x6801, 0x6808, 0x680F,
437 	0x6814, 0x6816, 0x6818, 0x681B, 0x69FD, 0x69FD, 0x6A3C, 0x6A3C,
438 	0x6B80, 0x6B80, 0x6BA0, 0x6BA0, 0x6BC0, 0x6BC1, 0x6BC8, 0x6BC9,
439 	0x6BD0, 0x6BD4, 0x6BD6, 0x6BD6, 0x6BEE, 0x6BEE,
440 	~0 /* sentinel */
441 };
442 
443 static struct msm_gpu_state *a4xx_gpu_state_get(struct msm_gpu *gpu)
444 {
445 	struct msm_gpu_state *state = kzalloc(sizeof(*state), GFP_KERNEL);
446 
447 	if (!state)
448 		return ERR_PTR(-ENOMEM);
449 
450 	adreno_gpu_state_get(gpu, state);
451 
452 	state->rbbm_status = gpu_read(gpu, REG_A4XX_RBBM_STATUS);
453 
454 	return state;
455 }
456 
457 /* Register offset defines for A4XX, in order of enum adreno_regs */
458 static const unsigned int a4xx_register_offsets[REG_ADRENO_REGISTER_MAX] = {
459 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_BASE, REG_A4XX_CP_RB_BASE),
460 	REG_ADRENO_SKIP(REG_ADRENO_CP_RB_BASE_HI),
461 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_RPTR_ADDR, REG_A4XX_CP_RB_RPTR_ADDR),
462 	REG_ADRENO_SKIP(REG_ADRENO_CP_RB_RPTR_ADDR_HI),
463 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_RPTR, REG_A4XX_CP_RB_RPTR),
464 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_WPTR, REG_A4XX_CP_RB_WPTR),
465 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_CNTL, REG_A4XX_CP_RB_CNTL),
466 };
467 
468 static void a4xx_dump(struct msm_gpu *gpu)
469 {
470 	printk("status:   %08x\n",
471 			gpu_read(gpu, REG_A4XX_RBBM_STATUS));
472 	adreno_dump(gpu);
473 }
474 
475 static int a4xx_pm_resume(struct msm_gpu *gpu) {
476 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
477 	int ret;
478 
479 	ret = msm_gpu_pm_resume(gpu);
480 	if (ret)
481 		return ret;
482 
483 	if (adreno_is_a430(adreno_gpu)) {
484 		unsigned int reg;
485 		/* Set the default register values; set SW_COLLAPSE to 0 */
486 		gpu_write(gpu, REG_A4XX_RBBM_POWER_CNTL_IP, 0x778000);
487 		do {
488 			udelay(5);
489 			reg = gpu_read(gpu, REG_A4XX_RBBM_POWER_STATUS);
490 		} while (!(reg & A4XX_RBBM_POWER_CNTL_IP_SP_TP_PWR_ON));
491 	}
492 	return 0;
493 }
494 
495 static int a4xx_pm_suspend(struct msm_gpu *gpu) {
496 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
497 	int ret;
498 
499 	ret = msm_gpu_pm_suspend(gpu);
500 	if (ret)
501 		return ret;
502 
503 	if (adreno_is_a430(adreno_gpu)) {
504 		/* Set the default register values; set SW_COLLAPSE to 1 */
505 		gpu_write(gpu, REG_A4XX_RBBM_POWER_CNTL_IP, 0x778001);
506 	}
507 	return 0;
508 }
509 
510 static int a4xx_get_timestamp(struct msm_gpu *gpu, uint64_t *value)
511 {
512 	*value = gpu_read64(gpu, REG_A4XX_RBBM_PERFCTR_CP_0_LO,
513 		REG_A4XX_RBBM_PERFCTR_CP_0_HI);
514 
515 	return 0;
516 }
517 
518 static const struct adreno_gpu_funcs funcs = {
519 	.base = {
520 		.get_param = adreno_get_param,
521 		.hw_init = a4xx_hw_init,
522 		.pm_suspend = a4xx_pm_suspend,
523 		.pm_resume = a4xx_pm_resume,
524 		.recover = a4xx_recover,
525 		.submit = adreno_submit,
526 		.flush = adreno_flush,
527 		.active_ring = adreno_active_ring,
528 		.irq = a4xx_irq,
529 		.destroy = a4xx_destroy,
530 #if defined(CONFIG_DEBUG_FS) || defined(CONFIG_DEV_COREDUMP)
531 		.show = adreno_show,
532 #endif
533 		.gpu_state_get = a4xx_gpu_state_get,
534 		.gpu_state_put = adreno_gpu_state_put,
535 	},
536 	.get_timestamp = a4xx_get_timestamp,
537 };
538 
539 struct msm_gpu *a4xx_gpu_init(struct drm_device *dev)
540 {
541 	struct a4xx_gpu *a4xx_gpu = NULL;
542 	struct adreno_gpu *adreno_gpu;
543 	struct msm_gpu *gpu;
544 	struct msm_drm_private *priv = dev->dev_private;
545 	struct platform_device *pdev = priv->gpu_pdev;
546 	int ret;
547 
548 	if (!pdev) {
549 		DRM_DEV_ERROR(dev->dev, "no a4xx device\n");
550 		ret = -ENXIO;
551 		goto fail;
552 	}
553 
554 	a4xx_gpu = kzalloc(sizeof(*a4xx_gpu), GFP_KERNEL);
555 	if (!a4xx_gpu) {
556 		ret = -ENOMEM;
557 		goto fail;
558 	}
559 
560 	adreno_gpu = &a4xx_gpu->base;
561 	gpu = &adreno_gpu->base;
562 
563 	gpu->perfcntrs = NULL;
564 	gpu->num_perfcntrs = 0;
565 
566 	adreno_gpu->registers = a4xx_registers;
567 	adreno_gpu->reg_offsets = a4xx_register_offsets;
568 
569 	ret = adreno_gpu_init(dev, pdev, adreno_gpu, &funcs, 1);
570 	if (ret)
571 		goto fail;
572 
573 	/* if needed, allocate gmem: */
574 	if (adreno_is_a4xx(adreno_gpu)) {
575 		ret = adreno_gpu_ocmem_init(dev->dev, adreno_gpu,
576 					    &a4xx_gpu->ocmem);
577 		if (ret)
578 			goto fail;
579 	}
580 
581 	if (!gpu->aspace) {
582 		/* TODO we think it is possible to configure the GPU to
583 		 * restrict access to VRAM carveout.  But the required
584 		 * registers are unknown.  For now just bail out and
585 		 * limp along with just modesetting.  If it turns out
586 		 * to not be possible to restrict access, then we must
587 		 * implement a cmdstream validator.
588 		 */
589 		DRM_DEV_ERROR(dev->dev, "No memory protection without IOMMU\n");
590 		ret = -ENXIO;
591 		goto fail;
592 	}
593 
594 	return gpu;
595 
596 fail:
597 	if (a4xx_gpu)
598 		a4xx_destroy(&a4xx_gpu->base.base);
599 
600 	return ERR_PTR(ret);
601 }
602