xref: /openbmc/linux/drivers/gpu/drm/msm/adreno/a3xx_gpu.c (revision 7b73a9c8)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (C) 2013 Red Hat
4  * Author: Rob Clark <robdclark@gmail.com>
5  *
6  * Copyright (c) 2014 The Linux Foundation. All rights reserved.
7  */
8 
9 #include "a3xx_gpu.h"
10 
11 #define A3XX_INT0_MASK \
12 	(A3XX_INT0_RBBM_AHB_ERROR |        \
13 	 A3XX_INT0_RBBM_ATB_BUS_OVERFLOW | \
14 	 A3XX_INT0_CP_T0_PACKET_IN_IB |    \
15 	 A3XX_INT0_CP_OPCODE_ERROR |       \
16 	 A3XX_INT0_CP_RESERVED_BIT_ERROR | \
17 	 A3XX_INT0_CP_HW_FAULT |           \
18 	 A3XX_INT0_CP_IB1_INT |            \
19 	 A3XX_INT0_CP_IB2_INT |            \
20 	 A3XX_INT0_CP_RB_INT |             \
21 	 A3XX_INT0_CP_REG_PROTECT_FAULT |  \
22 	 A3XX_INT0_CP_AHB_ERROR_HALT |     \
23 	 A3XX_INT0_CACHE_FLUSH_TS |        \
24 	 A3XX_INT0_UCHE_OOB_ACCESS)
25 
26 extern bool hang_debug;
27 
28 static void a3xx_dump(struct msm_gpu *gpu);
29 static bool a3xx_idle(struct msm_gpu *gpu);
30 
31 static bool a3xx_me_init(struct msm_gpu *gpu)
32 {
33 	struct msm_ringbuffer *ring = gpu->rb[0];
34 
35 	OUT_PKT3(ring, CP_ME_INIT, 17);
36 	OUT_RING(ring, 0x000003f7);
37 	OUT_RING(ring, 0x00000000);
38 	OUT_RING(ring, 0x00000000);
39 	OUT_RING(ring, 0x00000000);
40 	OUT_RING(ring, 0x00000080);
41 	OUT_RING(ring, 0x00000100);
42 	OUT_RING(ring, 0x00000180);
43 	OUT_RING(ring, 0x00006600);
44 	OUT_RING(ring, 0x00000150);
45 	OUT_RING(ring, 0x0000014e);
46 	OUT_RING(ring, 0x00000154);
47 	OUT_RING(ring, 0x00000001);
48 	OUT_RING(ring, 0x00000000);
49 	OUT_RING(ring, 0x00000000);
50 	OUT_RING(ring, 0x00000000);
51 	OUT_RING(ring, 0x00000000);
52 	OUT_RING(ring, 0x00000000);
53 
54 	gpu->funcs->flush(gpu, ring);
55 	return a3xx_idle(gpu);
56 }
57 
58 static int a3xx_hw_init(struct msm_gpu *gpu)
59 {
60 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
61 	struct a3xx_gpu *a3xx_gpu = to_a3xx_gpu(adreno_gpu);
62 	uint32_t *ptr, len;
63 	int i, ret;
64 
65 	DBG("%s", gpu->name);
66 
67 	if (adreno_is_a305(adreno_gpu)) {
68 		/* Set up 16 deep read/write request queues: */
69 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF0, 0x10101010);
70 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF1, 0x10101010);
71 		gpu_write(gpu, REG_A3XX_VBIF_OUT_RD_LIM_CONF0, 0x10101010);
72 		gpu_write(gpu, REG_A3XX_VBIF_OUT_WR_LIM_CONF0, 0x10101010);
73 		gpu_write(gpu, REG_A3XX_VBIF_DDR_OUT_MAX_BURST, 0x0000303);
74 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF0, 0x10101010);
75 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF1, 0x10101010);
76 		/* Enable WR-REQ: */
77 		gpu_write(gpu, REG_A3XX_VBIF_GATE_OFF_WRREQ_EN, 0x0000ff);
78 		/* Set up round robin arbitration between both AXI ports: */
79 		gpu_write(gpu, REG_A3XX_VBIF_ARB_CTL, 0x00000030);
80 		/* Set up AOOO: */
81 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO_EN, 0x0000003c);
82 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO, 0x003c003c);
83 	} else if (adreno_is_a306(adreno_gpu)) {
84 		gpu_write(gpu, REG_A3XX_VBIF_ROUND_ROBIN_QOS_ARB, 0x0003);
85 		gpu_write(gpu, REG_A3XX_VBIF_OUT_RD_LIM_CONF0, 0x0000000a);
86 		gpu_write(gpu, REG_A3XX_VBIF_OUT_WR_LIM_CONF0, 0x0000000a);
87 	} else if (adreno_is_a320(adreno_gpu)) {
88 		/* Set up 16 deep read/write request queues: */
89 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF0, 0x10101010);
90 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF1, 0x10101010);
91 		gpu_write(gpu, REG_A3XX_VBIF_OUT_RD_LIM_CONF0, 0x10101010);
92 		gpu_write(gpu, REG_A3XX_VBIF_OUT_WR_LIM_CONF0, 0x10101010);
93 		gpu_write(gpu, REG_A3XX_VBIF_DDR_OUT_MAX_BURST, 0x0000303);
94 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF0, 0x10101010);
95 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF1, 0x10101010);
96 		/* Enable WR-REQ: */
97 		gpu_write(gpu, REG_A3XX_VBIF_GATE_OFF_WRREQ_EN, 0x0000ff);
98 		/* Set up round robin arbitration between both AXI ports: */
99 		gpu_write(gpu, REG_A3XX_VBIF_ARB_CTL, 0x00000030);
100 		/* Set up AOOO: */
101 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO_EN, 0x0000003c);
102 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO, 0x003c003c);
103 		/* Enable 1K sort: */
104 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT, 0x000000ff);
105 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT_CONF, 0x000000a4);
106 
107 	} else if (adreno_is_a330v2(adreno_gpu)) {
108 		/*
109 		 * Most of the VBIF registers on 8974v2 have the correct
110 		 * values at power on, so we won't modify those if we don't
111 		 * need to
112 		 */
113 		/* Enable 1k sort: */
114 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT, 0x0001003f);
115 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT_CONF, 0x000000a4);
116 		/* Enable WR-REQ: */
117 		gpu_write(gpu, REG_A3XX_VBIF_GATE_OFF_WRREQ_EN, 0x00003f);
118 		gpu_write(gpu, REG_A3XX_VBIF_DDR_OUT_MAX_BURST, 0x0000303);
119 		/* Set up VBIF_ROUND_ROBIN_QOS_ARB: */
120 		gpu_write(gpu, REG_A3XX_VBIF_ROUND_ROBIN_QOS_ARB, 0x0003);
121 
122 	} else if (adreno_is_a330(adreno_gpu)) {
123 		/* Set up 16 deep read/write request queues: */
124 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF0, 0x18181818);
125 		gpu_write(gpu, REG_A3XX_VBIF_IN_RD_LIM_CONF1, 0x18181818);
126 		gpu_write(gpu, REG_A3XX_VBIF_OUT_RD_LIM_CONF0, 0x18181818);
127 		gpu_write(gpu, REG_A3XX_VBIF_OUT_WR_LIM_CONF0, 0x18181818);
128 		gpu_write(gpu, REG_A3XX_VBIF_DDR_OUT_MAX_BURST, 0x0000303);
129 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF0, 0x18181818);
130 		gpu_write(gpu, REG_A3XX_VBIF_IN_WR_LIM_CONF1, 0x18181818);
131 		/* Enable WR-REQ: */
132 		gpu_write(gpu, REG_A3XX_VBIF_GATE_OFF_WRREQ_EN, 0x00003f);
133 		/* Set up round robin arbitration between both AXI ports: */
134 		gpu_write(gpu, REG_A3XX_VBIF_ARB_CTL, 0x00000030);
135 		/* Set up VBIF_ROUND_ROBIN_QOS_ARB: */
136 		gpu_write(gpu, REG_A3XX_VBIF_ROUND_ROBIN_QOS_ARB, 0x0001);
137 		/* Set up AOOO: */
138 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO_EN, 0x0000003f);
139 		gpu_write(gpu, REG_A3XX_VBIF_OUT_AXI_AOOO, 0x003f003f);
140 		/* Enable 1K sort: */
141 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT, 0x0001003f);
142 		gpu_write(gpu, REG_A3XX_VBIF_ABIT_SORT_CONF, 0x000000a4);
143 		/* Disable VBIF clock gating. This is to enable AXI running
144 		 * higher frequency than GPU:
145 		 */
146 		gpu_write(gpu, REG_A3XX_VBIF_CLKON, 0x00000001);
147 
148 	} else {
149 		BUG();
150 	}
151 
152 	/* Make all blocks contribute to the GPU BUSY perf counter: */
153 	gpu_write(gpu, REG_A3XX_RBBM_GPU_BUSY_MASKED, 0xffffffff);
154 
155 	/* Tune the hystersis counters for SP and CP idle detection: */
156 	gpu_write(gpu, REG_A3XX_RBBM_SP_HYST_CNT, 0x10);
157 	gpu_write(gpu, REG_A3XX_RBBM_WAIT_IDLE_CLOCKS_CTL, 0x10);
158 
159 	/* Enable the RBBM error reporting bits.  This lets us get
160 	 * useful information on failure:
161 	 */
162 	gpu_write(gpu, REG_A3XX_RBBM_AHB_CTL0, 0x00000001);
163 
164 	/* Enable AHB error reporting: */
165 	gpu_write(gpu, REG_A3XX_RBBM_AHB_CTL1, 0xa6ffffff);
166 
167 	/* Turn on the power counters: */
168 	gpu_write(gpu, REG_A3XX_RBBM_RBBM_CTL, 0x00030000);
169 
170 	/* Turn on hang detection - this spews a lot of useful information
171 	 * into the RBBM registers on a hang:
172 	 */
173 	gpu_write(gpu, REG_A3XX_RBBM_INTERFACE_HANG_INT_CTL, 0x00010fff);
174 
175 	/* Enable 64-byte cacheline size. HW Default is 32-byte (0x000000E0): */
176 	gpu_write(gpu, REG_A3XX_UCHE_CACHE_MODE_CONTROL_REG, 0x00000001);
177 
178 	/* Enable Clock gating: */
179 	if (adreno_is_a306(adreno_gpu))
180 		gpu_write(gpu, REG_A3XX_RBBM_CLOCK_CTL, 0xaaaaaaaa);
181 	else if (adreno_is_a320(adreno_gpu))
182 		gpu_write(gpu, REG_A3XX_RBBM_CLOCK_CTL, 0xbfffffff);
183 	else if (adreno_is_a330v2(adreno_gpu))
184 		gpu_write(gpu, REG_A3XX_RBBM_CLOCK_CTL, 0xaaaaaaaa);
185 	else if (adreno_is_a330(adreno_gpu))
186 		gpu_write(gpu, REG_A3XX_RBBM_CLOCK_CTL, 0xbffcffff);
187 
188 	if (adreno_is_a330v2(adreno_gpu))
189 		gpu_write(gpu, REG_A3XX_RBBM_GPR0_CTL, 0x05515455);
190 	else if (adreno_is_a330(adreno_gpu))
191 		gpu_write(gpu, REG_A3XX_RBBM_GPR0_CTL, 0x00000000);
192 
193 	/* Set the OCMEM base address for A330, etc */
194 	if (a3xx_gpu->ocmem.hdl) {
195 		gpu_write(gpu, REG_A3XX_RB_GMEM_BASE_ADDR,
196 			(unsigned int)(a3xx_gpu->ocmem.base >> 14));
197 	}
198 
199 	/* Turn on performance counters: */
200 	gpu_write(gpu, REG_A3XX_RBBM_PERFCTR_CTL, 0x01);
201 
202 	/* Enable the perfcntrs that we use.. */
203 	for (i = 0; i < gpu->num_perfcntrs; i++) {
204 		const struct msm_gpu_perfcntr *perfcntr = &gpu->perfcntrs[i];
205 		gpu_write(gpu, perfcntr->select_reg, perfcntr->select_val);
206 	}
207 
208 	gpu_write(gpu, REG_A3XX_RBBM_INT_0_MASK, A3XX_INT0_MASK);
209 
210 	ret = adreno_hw_init(gpu);
211 	if (ret)
212 		return ret;
213 
214 	/* setup access protection: */
215 	gpu_write(gpu, REG_A3XX_CP_PROTECT_CTRL, 0x00000007);
216 
217 	/* RBBM registers */
218 	gpu_write(gpu, REG_A3XX_CP_PROTECT(0), 0x63000040);
219 	gpu_write(gpu, REG_A3XX_CP_PROTECT(1), 0x62000080);
220 	gpu_write(gpu, REG_A3XX_CP_PROTECT(2), 0x600000cc);
221 	gpu_write(gpu, REG_A3XX_CP_PROTECT(3), 0x60000108);
222 	gpu_write(gpu, REG_A3XX_CP_PROTECT(4), 0x64000140);
223 	gpu_write(gpu, REG_A3XX_CP_PROTECT(5), 0x66000400);
224 
225 	/* CP registers */
226 	gpu_write(gpu, REG_A3XX_CP_PROTECT(6), 0x65000700);
227 	gpu_write(gpu, REG_A3XX_CP_PROTECT(7), 0x610007d8);
228 	gpu_write(gpu, REG_A3XX_CP_PROTECT(8), 0x620007e0);
229 	gpu_write(gpu, REG_A3XX_CP_PROTECT(9), 0x61001178);
230 	gpu_write(gpu, REG_A3XX_CP_PROTECT(10), 0x64001180);
231 
232 	/* RB registers */
233 	gpu_write(gpu, REG_A3XX_CP_PROTECT(11), 0x60003300);
234 
235 	/* VBIF registers */
236 	gpu_write(gpu, REG_A3XX_CP_PROTECT(12), 0x6b00c000);
237 
238 	/* NOTE: PM4/micro-engine firmware registers look to be the same
239 	 * for a2xx and a3xx.. we could possibly push that part down to
240 	 * adreno_gpu base class.  Or push both PM4 and PFP but
241 	 * parameterize the pfp ucode addr/data registers..
242 	 */
243 
244 	/* Load PM4: */
245 	ptr = (uint32_t *)(adreno_gpu->fw[ADRENO_FW_PM4]->data);
246 	len = adreno_gpu->fw[ADRENO_FW_PM4]->size / 4;
247 	DBG("loading PM4 ucode version: %x", ptr[1]);
248 
249 	gpu_write(gpu, REG_AXXX_CP_DEBUG,
250 			AXXX_CP_DEBUG_DYNAMIC_CLK_DISABLE |
251 			AXXX_CP_DEBUG_MIU_128BIT_WRITE_ENABLE);
252 	gpu_write(gpu, REG_AXXX_CP_ME_RAM_WADDR, 0);
253 	for (i = 1; i < len; i++)
254 		gpu_write(gpu, REG_AXXX_CP_ME_RAM_DATA, ptr[i]);
255 
256 	/* Load PFP: */
257 	ptr = (uint32_t *)(adreno_gpu->fw[ADRENO_FW_PFP]->data);
258 	len = adreno_gpu->fw[ADRENO_FW_PFP]->size / 4;
259 	DBG("loading PFP ucode version: %x", ptr[5]);
260 
261 	gpu_write(gpu, REG_A3XX_CP_PFP_UCODE_ADDR, 0);
262 	for (i = 1; i < len; i++)
263 		gpu_write(gpu, REG_A3XX_CP_PFP_UCODE_DATA, ptr[i]);
264 
265 	/* CP ROQ queue sizes (bytes) - RB:16, ST:16, IB1:32, IB2:64 */
266 	if (adreno_is_a305(adreno_gpu) || adreno_is_a306(adreno_gpu) ||
267 			adreno_is_a320(adreno_gpu)) {
268 		gpu_write(gpu, REG_AXXX_CP_QUEUE_THRESHOLDS,
269 				AXXX_CP_QUEUE_THRESHOLDS_CSQ_IB1_START(2) |
270 				AXXX_CP_QUEUE_THRESHOLDS_CSQ_IB2_START(6) |
271 				AXXX_CP_QUEUE_THRESHOLDS_CSQ_ST_START(14));
272 	} else if (adreno_is_a330(adreno_gpu)) {
273 		/* NOTE: this (value take from downstream android driver)
274 		 * includes some bits outside of the known bitfields.  But
275 		 * A330 has this "MERCIU queue" thing too, which might
276 		 * explain a new bitfield or reshuffling:
277 		 */
278 		gpu_write(gpu, REG_AXXX_CP_QUEUE_THRESHOLDS, 0x003e2008);
279 	}
280 
281 	/* clear ME_HALT to start micro engine */
282 	gpu_write(gpu, REG_AXXX_CP_ME_CNTL, 0);
283 
284 	return a3xx_me_init(gpu) ? 0 : -EINVAL;
285 }
286 
287 static void a3xx_recover(struct msm_gpu *gpu)
288 {
289 	int i;
290 
291 	adreno_dump_info(gpu);
292 
293 	for (i = 0; i < 8; i++) {
294 		printk("CP_SCRATCH_REG%d: %u\n", i,
295 			gpu_read(gpu, REG_AXXX_CP_SCRATCH_REG0 + i));
296 	}
297 
298 	/* dump registers before resetting gpu, if enabled: */
299 	if (hang_debug)
300 		a3xx_dump(gpu);
301 
302 	gpu_write(gpu, REG_A3XX_RBBM_SW_RESET_CMD, 1);
303 	gpu_read(gpu, REG_A3XX_RBBM_SW_RESET_CMD);
304 	gpu_write(gpu, REG_A3XX_RBBM_SW_RESET_CMD, 0);
305 	adreno_recover(gpu);
306 }
307 
308 static void a3xx_destroy(struct msm_gpu *gpu)
309 {
310 	struct adreno_gpu *adreno_gpu = to_adreno_gpu(gpu);
311 	struct a3xx_gpu *a3xx_gpu = to_a3xx_gpu(adreno_gpu);
312 
313 	DBG("%s", gpu->name);
314 
315 	adreno_gpu_cleanup(adreno_gpu);
316 
317 	adreno_gpu_ocmem_cleanup(&a3xx_gpu->ocmem);
318 
319 	kfree(a3xx_gpu);
320 }
321 
322 static bool a3xx_idle(struct msm_gpu *gpu)
323 {
324 	/* wait for ringbuffer to drain: */
325 	if (!adreno_idle(gpu, gpu->rb[0]))
326 		return false;
327 
328 	/* then wait for GPU to finish: */
329 	if (spin_until(!(gpu_read(gpu, REG_A3XX_RBBM_STATUS) &
330 			A3XX_RBBM_STATUS_GPU_BUSY))) {
331 		DRM_ERROR("%s: timeout waiting for GPU to idle!\n", gpu->name);
332 
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 a3xx_irq(struct msm_gpu *gpu)
341 {
342 	uint32_t status;
343 
344 	status = gpu_read(gpu, REG_A3XX_RBBM_INT_0_STATUS);
345 	DBG("%s: %08x", gpu->name, status);
346 
347 	// TODO
348 
349 	gpu_write(gpu, REG_A3XX_RBBM_INT_CLEAR_CMD, status);
350 
351 	msm_gpu_retire(gpu);
352 
353 	return IRQ_HANDLED;
354 }
355 
356 static const unsigned int a3xx_registers[] = {
357 	0x0000, 0x0002, 0x0010, 0x0012, 0x0018, 0x0018, 0x0020, 0x0027,
358 	0x0029, 0x002b, 0x002e, 0x0033, 0x0040, 0x0042, 0x0050, 0x005c,
359 	0x0060, 0x006c, 0x0080, 0x0082, 0x0084, 0x0088, 0x0090, 0x00e5,
360 	0x00ea, 0x00ed, 0x0100, 0x0100, 0x0110, 0x0123, 0x01c0, 0x01c1,
361 	0x01c3, 0x01c5, 0x01c7, 0x01c7, 0x01d5, 0x01d9, 0x01dc, 0x01dd,
362 	0x01ea, 0x01ea, 0x01ee, 0x01f1, 0x01f5, 0x01f5, 0x01fc, 0x01ff,
363 	0x0440, 0x0440, 0x0443, 0x0443, 0x0445, 0x0445, 0x044d, 0x044f,
364 	0x0452, 0x0452, 0x0454, 0x046f, 0x047c, 0x047c, 0x047f, 0x047f,
365 	0x0578, 0x057f, 0x0600, 0x0602, 0x0605, 0x0607, 0x060a, 0x060e,
366 	0x0612, 0x0614, 0x0c01, 0x0c02, 0x0c06, 0x0c1d, 0x0c3d, 0x0c3f,
367 	0x0c48, 0x0c4b, 0x0c80, 0x0c80, 0x0c88, 0x0c8b, 0x0ca0, 0x0cb7,
368 	0x0cc0, 0x0cc1, 0x0cc6, 0x0cc7, 0x0ce4, 0x0ce5, 0x0e00, 0x0e05,
369 	0x0e0c, 0x0e0c, 0x0e22, 0x0e23, 0x0e41, 0x0e45, 0x0e64, 0x0e65,
370 	0x0e80, 0x0e82, 0x0e84, 0x0e89, 0x0ea0, 0x0ea1, 0x0ea4, 0x0ea7,
371 	0x0ec4, 0x0ecb, 0x0ee0, 0x0ee0, 0x0f00, 0x0f01, 0x0f03, 0x0f09,
372 	0x2040, 0x2040, 0x2044, 0x2044, 0x2048, 0x204d, 0x2068, 0x2069,
373 	0x206c, 0x206d, 0x2070, 0x2070, 0x2072, 0x2072, 0x2074, 0x2075,
374 	0x2079, 0x207a, 0x20c0, 0x20d3, 0x20e4, 0x20ef, 0x2100, 0x2109,
375 	0x210c, 0x210c, 0x210e, 0x210e, 0x2110, 0x2111, 0x2114, 0x2115,
376 	0x21e4, 0x21e4, 0x21ea, 0x21ea, 0x21ec, 0x21ed, 0x21f0, 0x21f0,
377 	0x2200, 0x2212, 0x2214, 0x2217, 0x221a, 0x221a, 0x2240, 0x227e,
378 	0x2280, 0x228b, 0x22c0, 0x22c0, 0x22c4, 0x22ce, 0x22d0, 0x22d8,
379 	0x22df, 0x22e6, 0x22e8, 0x22e9, 0x22ec, 0x22ec, 0x22f0, 0x22f7,
380 	0x22ff, 0x22ff, 0x2340, 0x2343, 0x2440, 0x2440, 0x2444, 0x2444,
381 	0x2448, 0x244d, 0x2468, 0x2469, 0x246c, 0x246d, 0x2470, 0x2470,
382 	0x2472, 0x2472, 0x2474, 0x2475, 0x2479, 0x247a, 0x24c0, 0x24d3,
383 	0x24e4, 0x24ef, 0x2500, 0x2509, 0x250c, 0x250c, 0x250e, 0x250e,
384 	0x2510, 0x2511, 0x2514, 0x2515, 0x25e4, 0x25e4, 0x25ea, 0x25ea,
385 	0x25ec, 0x25ed, 0x25f0, 0x25f0, 0x2600, 0x2612, 0x2614, 0x2617,
386 	0x261a, 0x261a, 0x2640, 0x267e, 0x2680, 0x268b, 0x26c0, 0x26c0,
387 	0x26c4, 0x26ce, 0x26d0, 0x26d8, 0x26df, 0x26e6, 0x26e8, 0x26e9,
388 	0x26ec, 0x26ec, 0x26f0, 0x26f7, 0x26ff, 0x26ff, 0x2740, 0x2743,
389 	0x300c, 0x300e, 0x301c, 0x301d, 0x302a, 0x302a, 0x302c, 0x302d,
390 	0x3030, 0x3031, 0x3034, 0x3036, 0x303c, 0x303c, 0x305e, 0x305f,
391 	~0   /* sentinel */
392 };
393 
394 /* would be nice to not have to duplicate the _show() stuff with printk(): */
395 static void a3xx_dump(struct msm_gpu *gpu)
396 {
397 	printk("status:   %08x\n",
398 			gpu_read(gpu, REG_A3XX_RBBM_STATUS));
399 	adreno_dump(gpu);
400 }
401 
402 static struct msm_gpu_state *a3xx_gpu_state_get(struct msm_gpu *gpu)
403 {
404 	struct msm_gpu_state *state = kzalloc(sizeof(*state), GFP_KERNEL);
405 
406 	if (!state)
407 		return ERR_PTR(-ENOMEM);
408 
409 	adreno_gpu_state_get(gpu, state);
410 
411 	state->rbbm_status = gpu_read(gpu, REG_A3XX_RBBM_STATUS);
412 
413 	return state;
414 }
415 
416 /* Register offset defines for A3XX */
417 static const unsigned int a3xx_register_offsets[REG_ADRENO_REGISTER_MAX] = {
418 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_BASE, REG_AXXX_CP_RB_BASE),
419 	REG_ADRENO_SKIP(REG_ADRENO_CP_RB_BASE_HI),
420 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_RPTR_ADDR, REG_AXXX_CP_RB_RPTR_ADDR),
421 	REG_ADRENO_SKIP(REG_ADRENO_CP_RB_RPTR_ADDR_HI),
422 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_RPTR, REG_AXXX_CP_RB_RPTR),
423 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_WPTR, REG_AXXX_CP_RB_WPTR),
424 	REG_ADRENO_DEFINE(REG_ADRENO_CP_RB_CNTL, REG_AXXX_CP_RB_CNTL),
425 };
426 
427 static const struct adreno_gpu_funcs funcs = {
428 	.base = {
429 		.get_param = adreno_get_param,
430 		.hw_init = a3xx_hw_init,
431 		.pm_suspend = msm_gpu_pm_suspend,
432 		.pm_resume = msm_gpu_pm_resume,
433 		.recover = a3xx_recover,
434 		.submit = adreno_submit,
435 		.flush = adreno_flush,
436 		.active_ring = adreno_active_ring,
437 		.irq = a3xx_irq,
438 		.destroy = a3xx_destroy,
439 #if defined(CONFIG_DEBUG_FS) || defined(CONFIG_DEV_COREDUMP)
440 		.show = adreno_show,
441 #endif
442 		.gpu_state_get = a3xx_gpu_state_get,
443 		.gpu_state_put = adreno_gpu_state_put,
444 	},
445 };
446 
447 static const struct msm_gpu_perfcntr perfcntrs[] = {
448 	{ REG_A3XX_SP_PERFCOUNTER6_SELECT, REG_A3XX_RBBM_PERFCTR_SP_6_LO,
449 			SP_ALU_ACTIVE_CYCLES, "ALUACTIVE" },
450 	{ REG_A3XX_SP_PERFCOUNTER7_SELECT, REG_A3XX_RBBM_PERFCTR_SP_7_LO,
451 			SP_FS_FULL_ALU_INSTRUCTIONS, "ALUFULL" },
452 };
453 
454 struct msm_gpu *a3xx_gpu_init(struct drm_device *dev)
455 {
456 	struct a3xx_gpu *a3xx_gpu = NULL;
457 	struct adreno_gpu *adreno_gpu;
458 	struct msm_gpu *gpu;
459 	struct msm_drm_private *priv = dev->dev_private;
460 	struct platform_device *pdev = priv->gpu_pdev;
461 	int ret;
462 
463 	if (!pdev) {
464 		DRM_DEV_ERROR(dev->dev, "no a3xx device\n");
465 		ret = -ENXIO;
466 		goto fail;
467 	}
468 
469 	a3xx_gpu = kzalloc(sizeof(*a3xx_gpu), GFP_KERNEL);
470 	if (!a3xx_gpu) {
471 		ret = -ENOMEM;
472 		goto fail;
473 	}
474 
475 	adreno_gpu = &a3xx_gpu->base;
476 	gpu = &adreno_gpu->base;
477 
478 	gpu->perfcntrs = perfcntrs;
479 	gpu->num_perfcntrs = ARRAY_SIZE(perfcntrs);
480 
481 	adreno_gpu->registers = a3xx_registers;
482 	adreno_gpu->reg_offsets = a3xx_register_offsets;
483 
484 	ret = adreno_gpu_init(dev, pdev, adreno_gpu, &funcs, 1);
485 	if (ret)
486 		goto fail;
487 
488 	/* if needed, allocate gmem: */
489 	if (adreno_is_a330(adreno_gpu)) {
490 		ret = adreno_gpu_ocmem_init(&adreno_gpu->base.pdev->dev,
491 					    adreno_gpu, &a3xx_gpu->ocmem);
492 		if (ret)
493 			goto fail;
494 	}
495 
496 	if (!gpu->aspace) {
497 		/* TODO we think it is possible to configure the GPU to
498 		 * restrict access to VRAM carveout.  But the required
499 		 * registers are unknown.  For now just bail out and
500 		 * limp along with just modesetting.  If it turns out
501 		 * to not be possible to restrict access, then we must
502 		 * implement a cmdstream validator.
503 		 */
504 		DRM_DEV_ERROR(dev->dev, "No memory protection without IOMMU\n");
505 		ret = -ENXIO;
506 		goto fail;
507 	}
508 
509 	return gpu;
510 
511 fail:
512 	if (a3xx_gpu)
513 		a3xx_destroy(&a3xx_gpu->base.base);
514 
515 	return ERR_PTR(ret);
516 }
517