blob: 9a72d959084c8a10a37cf60e8cdd4900a5ba2741 [file] [log] [blame]
Dave Airlie282a1672005-08-07 15:43:54 +10001/* savage_state.c -- State and drawing support for Savage
2 *
3 * Copyright 2004 Felix Kuehling
4 * All Rights Reserved.
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, sub license,
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 (including the
14 * next paragraph) shall be included in all copies or substantial portions
15 * of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 * NON-INFRINGEMENT. IN NO EVENT SHALL FELIX KUEHLING BE LIABLE FOR
21 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
22 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 */
25#include "drmP.h"
26#include "savage_drm.h"
27#include "savage_drv.h"
28
Dave Airlieb5e89ed2005-09-25 14:28:13 +100029void savage_emit_clip_rect_s3d(drm_savage_private_t * dev_priv,
Dave Airlieeddca552007-07-11 16:09:54 +100030 const struct drm_clip_rect * pbox)
Dave Airlie282a1672005-08-07 15:43:54 +100031{
32 uint32_t scstart = dev_priv->state.s3d.new_scstart;
Dave Airlieb5e89ed2005-09-25 14:28:13 +100033 uint32_t scend = dev_priv->state.s3d.new_scend;
Dave Airlie282a1672005-08-07 15:43:54 +100034 scstart = (scstart & ~SAVAGE_SCISSOR_MASK_S3D) |
Dave Airlieb5e89ed2005-09-25 14:28:13 +100035 ((uint32_t) pbox->x1 & 0x000007ff) |
36 (((uint32_t) pbox->y1 << 16) & 0x07ff0000);
37 scend = (scend & ~SAVAGE_SCISSOR_MASK_S3D) |
38 (((uint32_t) pbox->x2 - 1) & 0x000007ff) |
39 ((((uint32_t) pbox->y2 - 1) << 16) & 0x07ff0000);
Dave Airlie282a1672005-08-07 15:43:54 +100040 if (scstart != dev_priv->state.s3d.scstart ||
Dave Airlieb5e89ed2005-09-25 14:28:13 +100041 scend != dev_priv->state.s3d.scend) {
Dave Airlie282a1672005-08-07 15:43:54 +100042 DMA_LOCALS;
43 BEGIN_DMA(4);
Dave Airlieb5e89ed2005-09-25 14:28:13 +100044 DMA_WRITE(BCI_CMD_WAIT | BCI_CMD_WAIT_3D);
Dave Airlie282a1672005-08-07 15:43:54 +100045 DMA_SET_REGISTERS(SAVAGE_SCSTART_S3D, 2);
46 DMA_WRITE(scstart);
47 DMA_WRITE(scend);
48 dev_priv->state.s3d.scstart = scstart;
Dave Airlieb5e89ed2005-09-25 14:28:13 +100049 dev_priv->state.s3d.scend = scend;
Dave Airlie282a1672005-08-07 15:43:54 +100050 dev_priv->waiting = 1;
51 DMA_COMMIT();
52 }
53}
54
Dave Airlieb5e89ed2005-09-25 14:28:13 +100055void savage_emit_clip_rect_s4(drm_savage_private_t * dev_priv,
Dave Airlieeddca552007-07-11 16:09:54 +100056 const struct drm_clip_rect * pbox)
Dave Airlie282a1672005-08-07 15:43:54 +100057{
58 uint32_t drawctrl0 = dev_priv->state.s4.new_drawctrl0;
59 uint32_t drawctrl1 = dev_priv->state.s4.new_drawctrl1;
60 drawctrl0 = (drawctrl0 & ~SAVAGE_SCISSOR_MASK_S4) |
Dave Airlieb5e89ed2005-09-25 14:28:13 +100061 ((uint32_t) pbox->x1 & 0x000007ff) |
62 (((uint32_t) pbox->y1 << 12) & 0x00fff000);
Dave Airlie282a1672005-08-07 15:43:54 +100063 drawctrl1 = (drawctrl1 & ~SAVAGE_SCISSOR_MASK_S4) |
Dave Airlieb5e89ed2005-09-25 14:28:13 +100064 (((uint32_t) pbox->x2 - 1) & 0x000007ff) |
65 ((((uint32_t) pbox->y2 - 1) << 12) & 0x00fff000);
Dave Airlie282a1672005-08-07 15:43:54 +100066 if (drawctrl0 != dev_priv->state.s4.drawctrl0 ||
67 drawctrl1 != dev_priv->state.s4.drawctrl1) {
68 DMA_LOCALS;
69 BEGIN_DMA(4);
Dave Airlieb5e89ed2005-09-25 14:28:13 +100070 DMA_WRITE(BCI_CMD_WAIT | BCI_CMD_WAIT_3D);
Dave Airlie282a1672005-08-07 15:43:54 +100071 DMA_SET_REGISTERS(SAVAGE_DRAWCTRL0_S4, 2);
72 DMA_WRITE(drawctrl0);
73 DMA_WRITE(drawctrl1);
74 dev_priv->state.s4.drawctrl0 = drawctrl0;
75 dev_priv->state.s4.drawctrl1 = drawctrl1;
76 dev_priv->waiting = 1;
77 DMA_COMMIT();
78 }
79}
80
Dave Airlieb5e89ed2005-09-25 14:28:13 +100081static int savage_verify_texaddr(drm_savage_private_t * dev_priv, int unit,
Dave Airlie282a1672005-08-07 15:43:54 +100082 uint32_t addr)
83{
Dave Airlieb5e89ed2005-09-25 14:28:13 +100084 if ((addr & 6) != 2) { /* reserved bits */
Dave Airlie282a1672005-08-07 15:43:54 +100085 DRM_ERROR("bad texAddr%d %08x (reserved bits)\n", unit, addr);
Eric Anholt20caafa2007-08-25 19:22:43 +100086 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +100087 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +100088 if (!(addr & 1)) { /* local */
Dave Airlie282a1672005-08-07 15:43:54 +100089 addr &= ~7;
Dave Airlieb5e89ed2005-09-25 14:28:13 +100090 if (addr < dev_priv->texture_offset ||
91 addr >= dev_priv->texture_offset + dev_priv->texture_size) {
92 DRM_ERROR
93 ("bad texAddr%d %08x (local addr out of range)\n",
94 unit, addr);
Eric Anholt20caafa2007-08-25 19:22:43 +100095 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +100096 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +100097 } else { /* AGP */
Dave Airlie282a1672005-08-07 15:43:54 +100098 if (!dev_priv->agp_textures) {
99 DRM_ERROR("bad texAddr%d %08x (AGP not available)\n",
100 unit, addr);
Eric Anholt20caafa2007-08-25 19:22:43 +1000101 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000102 }
103 addr &= ~7;
104 if (addr < dev_priv->agp_textures->offset ||
105 addr >= (dev_priv->agp_textures->offset +
106 dev_priv->agp_textures->size)) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000107 DRM_ERROR
108 ("bad texAddr%d %08x (AGP addr out of range)\n",
109 unit, addr);
Eric Anholt20caafa2007-08-25 19:22:43 +1000110 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000111 }
112 }
113 return 0;
114}
115
116#define SAVE_STATE(reg,where) \
117 if(start <= reg && start+count > reg) \
Dave Airlie3528af12006-01-02 16:11:44 +1100118 dev_priv->state.where = regs[reg - start]
Dave Airlie282a1672005-08-07 15:43:54 +1000119#define SAVE_STATE_MASK(reg,where,mask) do { \
120 if(start <= reg && start+count > reg) { \
121 uint32_t tmp; \
Dave Airlie3528af12006-01-02 16:11:44 +1100122 tmp = regs[reg - start]; \
Dave Airlie282a1672005-08-07 15:43:54 +1000123 dev_priv->state.where = (tmp & (mask)) | \
124 (dev_priv->state.where & ~(mask)); \
125 } \
126} while (0)
Dave Airlie3528af12006-01-02 16:11:44 +1100127
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000128static int savage_verify_state_s3d(drm_savage_private_t * dev_priv,
Dave Airlie282a1672005-08-07 15:43:54 +1000129 unsigned int start, unsigned int count,
Dave Airlie3528af12006-01-02 16:11:44 +1100130 const uint32_t *regs)
Dave Airlie282a1672005-08-07 15:43:54 +1000131{
132 if (start < SAVAGE_TEXPALADDR_S3D ||
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000133 start + count - 1 > SAVAGE_DESTTEXRWWATERMARK_S3D) {
Dave Airlie282a1672005-08-07 15:43:54 +1000134 DRM_ERROR("invalid register range (0x%04x-0x%04x)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000135 start, start + count - 1);
Eric Anholt20caafa2007-08-25 19:22:43 +1000136 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000137 }
138
139 SAVE_STATE_MASK(SAVAGE_SCSTART_S3D, s3d.new_scstart,
140 ~SAVAGE_SCISSOR_MASK_S3D);
141 SAVE_STATE_MASK(SAVAGE_SCEND_S3D, s3d.new_scend,
142 ~SAVAGE_SCISSOR_MASK_S3D);
143
144 /* if any texture regs were changed ... */
145 if (start <= SAVAGE_TEXCTRL_S3D &&
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000146 start + count > SAVAGE_TEXPALADDR_S3D) {
Dave Airlie282a1672005-08-07 15:43:54 +1000147 /* ... check texture state */
148 SAVE_STATE(SAVAGE_TEXCTRL_S3D, s3d.texctrl);
149 SAVE_STATE(SAVAGE_TEXADDR_S3D, s3d.texaddr);
150 if (dev_priv->state.s3d.texctrl & SAVAGE_TEXCTRL_TEXEN_MASK)
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000151 return savage_verify_texaddr(dev_priv, 0,
Dave Airlie3528af12006-01-02 16:11:44 +1100152 dev_priv->state.s3d.texaddr);
Dave Airlie282a1672005-08-07 15:43:54 +1000153 }
154
155 return 0;
156}
157
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000158static int savage_verify_state_s4(drm_savage_private_t * dev_priv,
Dave Airlie282a1672005-08-07 15:43:54 +1000159 unsigned int start, unsigned int count,
Dave Airlie3528af12006-01-02 16:11:44 +1100160 const uint32_t *regs)
Dave Airlie282a1672005-08-07 15:43:54 +1000161{
162 int ret = 0;
163
164 if (start < SAVAGE_DRAWLOCALCTRL_S4 ||
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000165 start + count - 1 > SAVAGE_TEXBLENDCOLOR_S4) {
Dave Airlie282a1672005-08-07 15:43:54 +1000166 DRM_ERROR("invalid register range (0x%04x-0x%04x)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000167 start, start + count - 1);
Eric Anholt20caafa2007-08-25 19:22:43 +1000168 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000169 }
170
171 SAVE_STATE_MASK(SAVAGE_DRAWCTRL0_S4, s4.new_drawctrl0,
172 ~SAVAGE_SCISSOR_MASK_S4);
173 SAVE_STATE_MASK(SAVAGE_DRAWCTRL1_S4, s4.new_drawctrl1,
174 ~SAVAGE_SCISSOR_MASK_S4);
175
176 /* if any texture regs were changed ... */
Dave Airlie3528af12006-01-02 16:11:44 +1100177 if (start <= SAVAGE_TEXDESCR_S4 &&
178 start + count > SAVAGE_TEXPALADDR_S4) {
Dave Airlie282a1672005-08-07 15:43:54 +1000179 /* ... check texture state */
180 SAVE_STATE(SAVAGE_TEXDESCR_S4, s4.texdescr);
181 SAVE_STATE(SAVAGE_TEXADDR0_S4, s4.texaddr0);
182 SAVE_STATE(SAVAGE_TEXADDR1_S4, s4.texaddr1);
183 if (dev_priv->state.s4.texdescr & SAVAGE_TEXDESCR_TEX0EN_MASK)
Dave Airlie3528af12006-01-02 16:11:44 +1100184 ret |= savage_verify_texaddr(dev_priv, 0,
185 dev_priv->state.s4.texaddr0);
Dave Airlie282a1672005-08-07 15:43:54 +1000186 if (dev_priv->state.s4.texdescr & SAVAGE_TEXDESCR_TEX1EN_MASK)
Dave Airlie3528af12006-01-02 16:11:44 +1100187 ret |= savage_verify_texaddr(dev_priv, 1,
188 dev_priv->state.s4.texaddr1);
Dave Airlie282a1672005-08-07 15:43:54 +1000189 }
190
191 return ret;
192}
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000193
Dave Airlie282a1672005-08-07 15:43:54 +1000194#undef SAVE_STATE
195#undef SAVE_STATE_MASK
196
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000197static int savage_dispatch_state(drm_savage_private_t * dev_priv,
198 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +1100199 const uint32_t *regs)
Dave Airlie282a1672005-08-07 15:43:54 +1000200{
201 unsigned int count = cmd_header->state.count;
202 unsigned int start = cmd_header->state.start;
203 unsigned int count2 = 0;
204 unsigned int bci_size;
205 int ret;
206 DMA_LOCALS;
207
208 if (!count)
209 return 0;
210
Dave Airlie282a1672005-08-07 15:43:54 +1000211 if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
212 ret = savage_verify_state_s3d(dev_priv, start, count, regs);
213 if (ret != 0)
214 return ret;
215 /* scissor regs are emitted in savage_dispatch_draw */
216 if (start < SAVAGE_SCSTART_S3D) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000217 if (start + count > SAVAGE_SCEND_S3D + 1)
218 count2 = count - (SAVAGE_SCEND_S3D + 1 - start);
219 if (start + count > SAVAGE_SCSTART_S3D)
Dave Airlie282a1672005-08-07 15:43:54 +1000220 count = SAVAGE_SCSTART_S3D - start;
221 } else if (start <= SAVAGE_SCEND_S3D) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000222 if (start + count > SAVAGE_SCEND_S3D + 1) {
223 count -= SAVAGE_SCEND_S3D + 1 - start;
224 start = SAVAGE_SCEND_S3D + 1;
Dave Airlie282a1672005-08-07 15:43:54 +1000225 } else
226 return 0;
227 }
228 } else {
229 ret = savage_verify_state_s4(dev_priv, start, count, regs);
230 if (ret != 0)
231 return ret;
232 /* scissor regs are emitted in savage_dispatch_draw */
233 if (start < SAVAGE_DRAWCTRL0_S4) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000234 if (start + count > SAVAGE_DRAWCTRL1_S4 + 1)
Dave Airlie3528af12006-01-02 16:11:44 +1100235 count2 = count -
236 (SAVAGE_DRAWCTRL1_S4 + 1 - start);
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000237 if (start + count > SAVAGE_DRAWCTRL0_S4)
Dave Airlie282a1672005-08-07 15:43:54 +1000238 count = SAVAGE_DRAWCTRL0_S4 - start;
239 } else if (start <= SAVAGE_DRAWCTRL1_S4) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000240 if (start + count > SAVAGE_DRAWCTRL1_S4 + 1) {
241 count -= SAVAGE_DRAWCTRL1_S4 + 1 - start;
242 start = SAVAGE_DRAWCTRL1_S4 + 1;
Dave Airlie282a1672005-08-07 15:43:54 +1000243 } else
244 return 0;
245 }
246 }
247
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000248 bci_size = count + (count + 254) / 255 + count2 + (count2 + 254) / 255;
Dave Airlie282a1672005-08-07 15:43:54 +1000249
250 if (cmd_header->state.global) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000251 BEGIN_DMA(bci_size + 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000252 DMA_WRITE(BCI_CMD_WAIT | BCI_CMD_WAIT_3D);
253 dev_priv->waiting = 1;
254 } else {
255 BEGIN_DMA(bci_size);
256 }
257
258 do {
259 while (count > 0) {
260 unsigned int n = count < 255 ? count : 255;
261 DMA_SET_REGISTERS(start, n);
Dave Airlie3528af12006-01-02 16:11:44 +1100262 DMA_COPY(regs, n);
Dave Airlie282a1672005-08-07 15:43:54 +1000263 count -= n;
264 start += n;
265 regs += n;
266 }
267 start += 2;
268 regs += 2;
269 count = count2;
270 count2 = 0;
271 } while (count);
272
273 DMA_COMMIT();
274
275 return 0;
276}
277
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000278static int savage_dispatch_dma_prim(drm_savage_private_t * dev_priv,
279 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie056219e2007-07-11 16:17:42 +1000280 const struct drm_buf * dmabuf)
Dave Airlie282a1672005-08-07 15:43:54 +1000281{
282 unsigned char reorder = 0;
283 unsigned int prim = cmd_header->prim.prim;
284 unsigned int skip = cmd_header->prim.skip;
285 unsigned int n = cmd_header->prim.count;
286 unsigned int start = cmd_header->prim.start;
287 unsigned int i;
288 BCI_LOCALS;
289
290 if (!dmabuf) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000291 DRM_ERROR("called without dma buffers!\n");
Eric Anholt20caafa2007-08-25 19:22:43 +1000292 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000293 }
294
295 if (!n)
296 return 0;
297
298 switch (prim) {
299 case SAVAGE_PRIM_TRILIST_201:
300 reorder = 1;
301 prim = SAVAGE_PRIM_TRILIST;
302 case SAVAGE_PRIM_TRILIST:
303 if (n % 3 != 0) {
304 DRM_ERROR("wrong number of vertices %u in TRILIST\n",
305 n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000306 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000307 }
308 break;
309 case SAVAGE_PRIM_TRISTRIP:
310 case SAVAGE_PRIM_TRIFAN:
311 if (n < 3) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000312 DRM_ERROR
313 ("wrong number of vertices %u in TRIFAN/STRIP\n",
314 n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000315 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000316 }
317 break;
318 default:
319 DRM_ERROR("invalid primitive type %u\n", prim);
Eric Anholt20caafa2007-08-25 19:22:43 +1000320 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000321 }
322
323 if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
324 if (skip != 0) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000325 DRM_ERROR("invalid skip flags 0x%04x for DMA\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000326 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000327 }
328 } else {
329 unsigned int size = 10 - (skip & 1) - (skip >> 1 & 1) -
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000330 (skip >> 2 & 1) - (skip >> 3 & 1) - (skip >> 4 & 1) -
331 (skip >> 5 & 1) - (skip >> 6 & 1) - (skip >> 7 & 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000332 if (skip > SAVAGE_SKIP_ALL_S4 || size != 8) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000333 DRM_ERROR("invalid skip flags 0x%04x for DMA\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000334 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000335 }
336 if (reorder) {
337 DRM_ERROR("TRILIST_201 used on Savage4 hardware\n");
Eric Anholt20caafa2007-08-25 19:22:43 +1000338 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000339 }
340 }
341
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000342 if (start + n > dmabuf->total / 32) {
Dave Airlie282a1672005-08-07 15:43:54 +1000343 DRM_ERROR("vertex indices (%u-%u) out of range (0-%u)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000344 start, start + n - 1, dmabuf->total / 32);
Eric Anholt20caafa2007-08-25 19:22:43 +1000345 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000346 }
347
348 /* Vertex DMA doesn't work with command DMA at the same time,
349 * so we use BCI_... to submit commands here. Flush buffered
350 * faked DMA first. */
351 DMA_FLUSH();
352
353 if (dmabuf->bus_address != dev_priv->state.common.vbaddr) {
354 BEGIN_BCI(2);
355 BCI_SET_REGISTERS(SAVAGE_VERTBUFADDR, 1);
356 BCI_WRITE(dmabuf->bus_address | dev_priv->dma_type);
357 dev_priv->state.common.vbaddr = dmabuf->bus_address;
358 }
359 if (S3_SAVAGE3D_SERIES(dev_priv->chipset) && dev_priv->waiting) {
360 /* Workaround for what looks like a hardware bug. If a
361 * WAIT_3D_IDLE was emitted some time before the
362 * indexed drawing command then the engine will lock
363 * up. There are two known workarounds:
364 * WAIT_IDLE_EMPTY or emit at least 63 NOPs. */
365 BEGIN_BCI(63);
366 for (i = 0; i < 63; ++i)
367 BCI_WRITE(BCI_CMD_WAIT);
368 dev_priv->waiting = 0;
369 }
370
371 prim <<= 25;
372 while (n != 0) {
373 /* Can emit up to 255 indices (85 triangles) at once. */
374 unsigned int count = n > 255 ? 255 : n;
375 if (reorder) {
376 /* Need to reorder indices for correct flat
377 * shading while preserving the clock sense
378 * for correct culling. Only on Savage3D. */
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000379 int reorder[3] = { -1, -1, -1 };
380 reorder[start % 3] = 2;
Dave Airlie282a1672005-08-07 15:43:54 +1000381
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000382 BEGIN_BCI((count + 1 + 1) / 2);
383 BCI_DRAW_INDICES_S3D(count, prim, start + 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000384
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000385 for (i = start + 1; i + 1 < start + count; i += 2)
Dave Airlie282a1672005-08-07 15:43:54 +1000386 BCI_WRITE((i + reorder[i % 3]) |
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000387 ((i + 1 +
388 reorder[(i + 1) % 3]) << 16));
389 if (i < start + count)
390 BCI_WRITE(i + reorder[i % 3]);
Dave Airlie282a1672005-08-07 15:43:54 +1000391 } else if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000392 BEGIN_BCI((count + 1 + 1) / 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000393 BCI_DRAW_INDICES_S3D(count, prim, start);
394
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000395 for (i = start + 1; i + 1 < start + count; i += 2)
396 BCI_WRITE(i | ((i + 1) << 16));
397 if (i < start + count)
Dave Airlie282a1672005-08-07 15:43:54 +1000398 BCI_WRITE(i);
399 } else {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000400 BEGIN_BCI((count + 2 + 1) / 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000401 BCI_DRAW_INDICES_S4(count, prim, skip);
402
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000403 for (i = start; i + 1 < start + count; i += 2)
404 BCI_WRITE(i | ((i + 1) << 16));
405 if (i < start + count)
Dave Airlie282a1672005-08-07 15:43:54 +1000406 BCI_WRITE(i);
407 }
408
409 start += count;
410 n -= count;
411
412 prim |= BCI_CMD_DRAW_CONT;
413 }
414
415 return 0;
416}
417
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000418static int savage_dispatch_vb_prim(drm_savage_private_t * dev_priv,
419 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +1100420 const uint32_t *vtxbuf, unsigned int vb_size,
421 unsigned int vb_stride)
Dave Airlie282a1672005-08-07 15:43:54 +1000422{
423 unsigned char reorder = 0;
424 unsigned int prim = cmd_header->prim.prim;
425 unsigned int skip = cmd_header->prim.skip;
426 unsigned int n = cmd_header->prim.count;
427 unsigned int start = cmd_header->prim.start;
428 unsigned int vtx_size;
429 unsigned int i;
430 DMA_LOCALS;
431
432 if (!n)
433 return 0;
434
435 switch (prim) {
436 case SAVAGE_PRIM_TRILIST_201:
437 reorder = 1;
438 prim = SAVAGE_PRIM_TRILIST;
439 case SAVAGE_PRIM_TRILIST:
440 if (n % 3 != 0) {
441 DRM_ERROR("wrong number of vertices %u in TRILIST\n",
442 n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000443 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000444 }
445 break;
446 case SAVAGE_PRIM_TRISTRIP:
447 case SAVAGE_PRIM_TRIFAN:
448 if (n < 3) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000449 DRM_ERROR
450 ("wrong number of vertices %u in TRIFAN/STRIP\n",
451 n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000452 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000453 }
454 break;
455 default:
456 DRM_ERROR("invalid primitive type %u\n", prim);
Eric Anholt20caafa2007-08-25 19:22:43 +1000457 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000458 }
459
460 if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
461 if (skip > SAVAGE_SKIP_ALL_S3D) {
462 DRM_ERROR("invalid skip flags 0x%04x\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000463 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000464 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000465 vtx_size = 8; /* full vertex */
Dave Airlie282a1672005-08-07 15:43:54 +1000466 } else {
467 if (skip > SAVAGE_SKIP_ALL_S4) {
468 DRM_ERROR("invalid skip flags 0x%04x\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000469 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000470 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000471 vtx_size = 10; /* full vertex */
Dave Airlie282a1672005-08-07 15:43:54 +1000472 }
473
474 vtx_size -= (skip & 1) + (skip >> 1 & 1) +
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000475 (skip >> 2 & 1) + (skip >> 3 & 1) + (skip >> 4 & 1) +
476 (skip >> 5 & 1) + (skip >> 6 & 1) + (skip >> 7 & 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000477
478 if (vtx_size > vb_stride) {
479 DRM_ERROR("vertex size greater than vb stride (%u > %u)\n",
480 vtx_size, vb_stride);
Eric Anholt20caafa2007-08-25 19:22:43 +1000481 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000482 }
483
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000484 if (start + n > vb_size / (vb_stride * 4)) {
Dave Airlie282a1672005-08-07 15:43:54 +1000485 DRM_ERROR("vertex indices (%u-%u) out of range (0-%u)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000486 start, start + n - 1, vb_size / (vb_stride * 4));
Eric Anholt20caafa2007-08-25 19:22:43 +1000487 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000488 }
489
490 prim <<= 25;
491 while (n != 0) {
492 /* Can emit up to 255 vertices (85 triangles) at once. */
493 unsigned int count = n > 255 ? 255 : n;
494 if (reorder) {
495 /* Need to reorder vertices for correct flat
496 * shading while preserving the clock sense
497 * for correct culling. Only on Savage3D. */
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000498 int reorder[3] = { -1, -1, -1 };
499 reorder[start % 3] = 2;
Dave Airlie282a1672005-08-07 15:43:54 +1000500
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000501 BEGIN_DMA(count * vtx_size + 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000502 DMA_DRAW_PRIMITIVE(count, prim, skip);
503
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000504 for (i = start; i < start + count; ++i) {
Dave Airlie282a1672005-08-07 15:43:54 +1000505 unsigned int j = i + reorder[i % 3];
Dave Airlie3528af12006-01-02 16:11:44 +1100506 DMA_COPY(&vtxbuf[vb_stride * j], vtx_size);
Dave Airlie282a1672005-08-07 15:43:54 +1000507 }
508
509 DMA_COMMIT();
510 } else {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000511 BEGIN_DMA(count * vtx_size + 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000512 DMA_DRAW_PRIMITIVE(count, prim, skip);
513
514 if (vb_stride == vtx_size) {
Dave Airlie3528af12006-01-02 16:11:44 +1100515 DMA_COPY(&vtxbuf[vb_stride * start],
516 vtx_size * count);
Dave Airlie282a1672005-08-07 15:43:54 +1000517 } else {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000518 for (i = start; i < start + count; ++i) {
Dave Airlie3528af12006-01-02 16:11:44 +1100519 DMA_COPY(&vtxbuf [vb_stride * i],
520 vtx_size);
Dave Airlie282a1672005-08-07 15:43:54 +1000521 }
522 }
523
524 DMA_COMMIT();
525 }
526
527 start += count;
528 n -= count;
529
530 prim |= BCI_CMD_DRAW_CONT;
531 }
532
533 return 0;
534}
535
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000536static int savage_dispatch_dma_idx(drm_savage_private_t * dev_priv,
537 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +1100538 const uint16_t *idx,
Dave Airlie056219e2007-07-11 16:17:42 +1000539 const struct drm_buf * dmabuf)
Dave Airlie282a1672005-08-07 15:43:54 +1000540{
541 unsigned char reorder = 0;
542 unsigned int prim = cmd_header->idx.prim;
543 unsigned int skip = cmd_header->idx.skip;
544 unsigned int n = cmd_header->idx.count;
545 unsigned int i;
546 BCI_LOCALS;
547
548 if (!dmabuf) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000549 DRM_ERROR("called without dma buffers!\n");
Eric Anholt20caafa2007-08-25 19:22:43 +1000550 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000551 }
552
553 if (!n)
554 return 0;
555
556 switch (prim) {
557 case SAVAGE_PRIM_TRILIST_201:
558 reorder = 1;
559 prim = SAVAGE_PRIM_TRILIST;
560 case SAVAGE_PRIM_TRILIST:
561 if (n % 3 != 0) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000562 DRM_ERROR("wrong number of indices %u in TRILIST\n", n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000563 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000564 }
565 break;
566 case SAVAGE_PRIM_TRISTRIP:
567 case SAVAGE_PRIM_TRIFAN:
568 if (n < 3) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000569 DRM_ERROR
570 ("wrong number of indices %u in TRIFAN/STRIP\n", n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000571 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000572 }
573 break;
574 default:
575 DRM_ERROR("invalid primitive type %u\n", prim);
Eric Anholt20caafa2007-08-25 19:22:43 +1000576 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000577 }
578
579 if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
580 if (skip != 0) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000581 DRM_ERROR("invalid skip flags 0x%04x for DMA\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000582 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000583 }
584 } else {
585 unsigned int size = 10 - (skip & 1) - (skip >> 1 & 1) -
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000586 (skip >> 2 & 1) - (skip >> 3 & 1) - (skip >> 4 & 1) -
587 (skip >> 5 & 1) - (skip >> 6 & 1) - (skip >> 7 & 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000588 if (skip > SAVAGE_SKIP_ALL_S4 || size != 8) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000589 DRM_ERROR("invalid skip flags 0x%04x for DMA\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000590 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000591 }
592 if (reorder) {
593 DRM_ERROR("TRILIST_201 used on Savage4 hardware\n");
Eric Anholt20caafa2007-08-25 19:22:43 +1000594 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000595 }
596 }
597
598 /* Vertex DMA doesn't work with command DMA at the same time,
599 * so we use BCI_... to submit commands here. Flush buffered
600 * faked DMA first. */
601 DMA_FLUSH();
602
603 if (dmabuf->bus_address != dev_priv->state.common.vbaddr) {
604 BEGIN_BCI(2);
605 BCI_SET_REGISTERS(SAVAGE_VERTBUFADDR, 1);
606 BCI_WRITE(dmabuf->bus_address | dev_priv->dma_type);
607 dev_priv->state.common.vbaddr = dmabuf->bus_address;
608 }
609 if (S3_SAVAGE3D_SERIES(dev_priv->chipset) && dev_priv->waiting) {
610 /* Workaround for what looks like a hardware bug. If a
611 * WAIT_3D_IDLE was emitted some time before the
612 * indexed drawing command then the engine will lock
613 * up. There are two known workarounds:
614 * WAIT_IDLE_EMPTY or emit at least 63 NOPs. */
615 BEGIN_BCI(63);
616 for (i = 0; i < 63; ++i)
617 BCI_WRITE(BCI_CMD_WAIT);
618 dev_priv->waiting = 0;
619 }
620
621 prim <<= 25;
622 while (n != 0) {
623 /* Can emit up to 255 indices (85 triangles) at once. */
624 unsigned int count = n > 255 ? 255 : n;
Dave Airlie282a1672005-08-07 15:43:54 +1000625
Dave Airlie3528af12006-01-02 16:11:44 +1100626 /* check indices */
Dave Airlie282a1672005-08-07 15:43:54 +1000627 for (i = 0; i < count; ++i) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000628 if (idx[i] > dmabuf->total / 32) {
Dave Airlie282a1672005-08-07 15:43:54 +1000629 DRM_ERROR("idx[%u]=%u out of range (0-%u)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000630 i, idx[i], dmabuf->total / 32);
Eric Anholt20caafa2007-08-25 19:22:43 +1000631 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000632 }
633 }
634
635 if (reorder) {
636 /* Need to reorder indices for correct flat
637 * shading while preserving the clock sense
638 * for correct culling. Only on Savage3D. */
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000639 int reorder[3] = { 2, -1, -1 };
Dave Airlie282a1672005-08-07 15:43:54 +1000640
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000641 BEGIN_BCI((count + 1 + 1) / 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000642 BCI_DRAW_INDICES_S3D(count, prim, idx[2]);
643
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000644 for (i = 1; i + 1 < count; i += 2)
Dave Airlie282a1672005-08-07 15:43:54 +1000645 BCI_WRITE(idx[i + reorder[i % 3]] |
Dave Airlie3528af12006-01-02 16:11:44 +1100646 (idx[i + 1 +
647 reorder[(i + 1) % 3]] << 16));
Dave Airlie282a1672005-08-07 15:43:54 +1000648 if (i < count)
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000649 BCI_WRITE(idx[i + reorder[i % 3]]);
Dave Airlie282a1672005-08-07 15:43:54 +1000650 } else if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000651 BEGIN_BCI((count + 1 + 1) / 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000652 BCI_DRAW_INDICES_S3D(count, prim, idx[0]);
653
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000654 for (i = 1; i + 1 < count; i += 2)
655 BCI_WRITE(idx[i] | (idx[i + 1] << 16));
Dave Airlie282a1672005-08-07 15:43:54 +1000656 if (i < count)
657 BCI_WRITE(idx[i]);
658 } else {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000659 BEGIN_BCI((count + 2 + 1) / 2);
Dave Airlie282a1672005-08-07 15:43:54 +1000660 BCI_DRAW_INDICES_S4(count, prim, skip);
661
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000662 for (i = 0; i + 1 < count; i += 2)
663 BCI_WRITE(idx[i] | (idx[i + 1] << 16));
Dave Airlie282a1672005-08-07 15:43:54 +1000664 if (i < count)
665 BCI_WRITE(idx[i]);
666 }
667
Dave Airlie3528af12006-01-02 16:11:44 +1100668 idx += count;
Dave Airlie282a1672005-08-07 15:43:54 +1000669 n -= count;
670
671 prim |= BCI_CMD_DRAW_CONT;
672 }
673
674 return 0;
675}
676
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000677static int savage_dispatch_vb_idx(drm_savage_private_t * dev_priv,
678 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +1100679 const uint16_t *idx,
680 const uint32_t *vtxbuf,
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000681 unsigned int vb_size, unsigned int vb_stride)
Dave Airlie282a1672005-08-07 15:43:54 +1000682{
683 unsigned char reorder = 0;
684 unsigned int prim = cmd_header->idx.prim;
685 unsigned int skip = cmd_header->idx.skip;
686 unsigned int n = cmd_header->idx.count;
687 unsigned int vtx_size;
688 unsigned int i;
689 DMA_LOCALS;
690
691 if (!n)
692 return 0;
693
694 switch (prim) {
695 case SAVAGE_PRIM_TRILIST_201:
696 reorder = 1;
697 prim = SAVAGE_PRIM_TRILIST;
698 case SAVAGE_PRIM_TRILIST:
699 if (n % 3 != 0) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000700 DRM_ERROR("wrong number of indices %u in TRILIST\n", n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000701 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000702 }
703 break;
704 case SAVAGE_PRIM_TRISTRIP:
705 case SAVAGE_PRIM_TRIFAN:
706 if (n < 3) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000707 DRM_ERROR
708 ("wrong number of indices %u in TRIFAN/STRIP\n", n);
Eric Anholt20caafa2007-08-25 19:22:43 +1000709 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000710 }
711 break;
712 default:
713 DRM_ERROR("invalid primitive type %u\n", prim);
Eric Anholt20caafa2007-08-25 19:22:43 +1000714 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000715 }
716
717 if (S3_SAVAGE3D_SERIES(dev_priv->chipset)) {
718 if (skip > SAVAGE_SKIP_ALL_S3D) {
719 DRM_ERROR("invalid skip flags 0x%04x\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000720 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000721 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000722 vtx_size = 8; /* full vertex */
Dave Airlie282a1672005-08-07 15:43:54 +1000723 } else {
724 if (skip > SAVAGE_SKIP_ALL_S4) {
725 DRM_ERROR("invalid skip flags 0x%04x\n", skip);
Eric Anholt20caafa2007-08-25 19:22:43 +1000726 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000727 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000728 vtx_size = 10; /* full vertex */
Dave Airlie282a1672005-08-07 15:43:54 +1000729 }
730
731 vtx_size -= (skip & 1) + (skip >> 1 & 1) +
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000732 (skip >> 2 & 1) + (skip >> 3 & 1) + (skip >> 4 & 1) +
733 (skip >> 5 & 1) + (skip >> 6 & 1) + (skip >> 7 & 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000734
735 if (vtx_size > vb_stride) {
736 DRM_ERROR("vertex size greater than vb stride (%u > %u)\n",
737 vtx_size, vb_stride);
Eric Anholt20caafa2007-08-25 19:22:43 +1000738 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000739 }
740
741 prim <<= 25;
742 while (n != 0) {
743 /* Can emit up to 255 vertices (85 triangles) at once. */
744 unsigned int count = n > 255 ? 255 : n;
Dave Airlie3528af12006-01-02 16:11:44 +1100745
746 /* Check indices */
Dave Airlie282a1672005-08-07 15:43:54 +1000747 for (i = 0; i < count; ++i) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000748 if (idx[i] > vb_size / (vb_stride * 4)) {
Dave Airlie282a1672005-08-07 15:43:54 +1000749 DRM_ERROR("idx[%u]=%u out of range (0-%u)\n",
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000750 i, idx[i], vb_size / (vb_stride * 4));
Eric Anholt20caafa2007-08-25 19:22:43 +1000751 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000752 }
753 }
754
755 if (reorder) {
756 /* Need to reorder vertices for correct flat
757 * shading while preserving the clock sense
758 * for correct culling. Only on Savage3D. */
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000759 int reorder[3] = { 2, -1, -1 };
Dave Airlie282a1672005-08-07 15:43:54 +1000760
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000761 BEGIN_DMA(count * vtx_size + 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000762 DMA_DRAW_PRIMITIVE(count, prim, skip);
763
764 for (i = 0; i < count; ++i) {
765 unsigned int j = idx[i + reorder[i % 3]];
Dave Airlie3528af12006-01-02 16:11:44 +1100766 DMA_COPY(&vtxbuf[vb_stride * j], vtx_size);
Dave Airlie282a1672005-08-07 15:43:54 +1000767 }
768
769 DMA_COMMIT();
770 } else {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000771 BEGIN_DMA(count * vtx_size + 1);
Dave Airlie282a1672005-08-07 15:43:54 +1000772 DMA_DRAW_PRIMITIVE(count, prim, skip);
773
774 for (i = 0; i < count; ++i) {
775 unsigned int j = idx[i];
Dave Airlie3528af12006-01-02 16:11:44 +1100776 DMA_COPY(&vtxbuf[vb_stride * j], vtx_size);
Dave Airlie282a1672005-08-07 15:43:54 +1000777 }
778
779 DMA_COMMIT();
780 }
781
Dave Airlie3528af12006-01-02 16:11:44 +1100782 idx += count;
Dave Airlie282a1672005-08-07 15:43:54 +1000783 n -= count;
784
785 prim |= BCI_CMD_DRAW_CONT;
786 }
787
788 return 0;
789}
790
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000791static int savage_dispatch_clear(drm_savage_private_t * dev_priv,
792 const drm_savage_cmd_header_t * cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +1100793 const drm_savage_cmd_header_t *data,
Dave Airlie282a1672005-08-07 15:43:54 +1000794 unsigned int nbox,
Dave Airlieeddca552007-07-11 16:09:54 +1000795 const struct drm_clip_rect *boxes)
Dave Airlie282a1672005-08-07 15:43:54 +1000796{
Dave Airlie3528af12006-01-02 16:11:44 +1100797 unsigned int flags = cmd_header->clear0.flags;
Dave Airlie282a1672005-08-07 15:43:54 +1000798 unsigned int clear_cmd;
799 unsigned int i, nbufs;
800 DMA_LOCALS;
801
802 if (nbox == 0)
803 return 0;
804
Dave Airlie282a1672005-08-07 15:43:54 +1000805 clear_cmd = BCI_CMD_RECT | BCI_CMD_RECT_XP | BCI_CMD_RECT_YP |
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000806 BCI_CMD_SEND_COLOR | BCI_CMD_DEST_PBD_NEW;
807 BCI_CMD_SET_ROP(clear_cmd, 0xCC);
Dave Airlie282a1672005-08-07 15:43:54 +1000808
809 nbufs = ((flags & SAVAGE_FRONT) ? 1 : 0) +
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000810 ((flags & SAVAGE_BACK) ? 1 : 0) + ((flags & SAVAGE_DEPTH) ? 1 : 0);
Dave Airlie282a1672005-08-07 15:43:54 +1000811 if (nbufs == 0)
812 return 0;
813
Dave Airlie3528af12006-01-02 16:11:44 +1100814 if (data->clear1.mask != 0xffffffff) {
Dave Airlie282a1672005-08-07 15:43:54 +1000815 /* set mask */
816 BEGIN_DMA(2);
817 DMA_SET_REGISTERS(SAVAGE_BITPLANEWTMASK, 1);
Dave Airlie3528af12006-01-02 16:11:44 +1100818 DMA_WRITE(data->clear1.mask);
Dave Airlie282a1672005-08-07 15:43:54 +1000819 DMA_COMMIT();
820 }
821 for (i = 0; i < nbox; ++i) {
Dave Airlie282a1672005-08-07 15:43:54 +1000822 unsigned int x, y, w, h;
823 unsigned int buf;
Dave Airlie3528af12006-01-02 16:11:44 +1100824 x = boxes[i].x1, y = boxes[i].y1;
825 w = boxes[i].x2 - boxes[i].x1;
826 h = boxes[i].y2 - boxes[i].y1;
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000827 BEGIN_DMA(nbufs * 6);
Dave Airlie282a1672005-08-07 15:43:54 +1000828 for (buf = SAVAGE_FRONT; buf <= SAVAGE_DEPTH; buf <<= 1) {
829 if (!(flags & buf))
830 continue;
831 DMA_WRITE(clear_cmd);
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000832 switch (buf) {
Dave Airlie282a1672005-08-07 15:43:54 +1000833 case SAVAGE_FRONT:
834 DMA_WRITE(dev_priv->front_offset);
835 DMA_WRITE(dev_priv->front_bd);
836 break;
837 case SAVAGE_BACK:
838 DMA_WRITE(dev_priv->back_offset);
839 DMA_WRITE(dev_priv->back_bd);
840 break;
841 case SAVAGE_DEPTH:
842 DMA_WRITE(dev_priv->depth_offset);
843 DMA_WRITE(dev_priv->depth_bd);
844 break;
845 }
Dave Airlie3528af12006-01-02 16:11:44 +1100846 DMA_WRITE(data->clear1.value);
Dave Airlie282a1672005-08-07 15:43:54 +1000847 DMA_WRITE(BCI_X_Y(x, y));
848 DMA_WRITE(BCI_W_H(w, h));
849 }
850 DMA_COMMIT();
851 }
Dave Airlie3528af12006-01-02 16:11:44 +1100852 if (data->clear1.mask != 0xffffffff) {
Dave Airlie282a1672005-08-07 15:43:54 +1000853 /* reset mask */
854 BEGIN_DMA(2);
855 DMA_SET_REGISTERS(SAVAGE_BITPLANEWTMASK, 1);
856 DMA_WRITE(0xffffffff);
857 DMA_COMMIT();
858 }
859
860 return 0;
861}
862
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000863static int savage_dispatch_swap(drm_savage_private_t * dev_priv,
Dave Airlieeddca552007-07-11 16:09:54 +1000864 unsigned int nbox, const struct drm_clip_rect *boxes)
Dave Airlie282a1672005-08-07 15:43:54 +1000865{
866 unsigned int swap_cmd;
867 unsigned int i;
868 DMA_LOCALS;
869
870 if (nbox == 0)
871 return 0;
872
873 swap_cmd = BCI_CMD_RECT | BCI_CMD_RECT_XP | BCI_CMD_RECT_YP |
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000874 BCI_CMD_SRC_PBD_COLOR_NEW | BCI_CMD_DEST_GBD;
875 BCI_CMD_SET_ROP(swap_cmd, 0xCC);
Dave Airlie282a1672005-08-07 15:43:54 +1000876
877 for (i = 0; i < nbox; ++i) {
Dave Airlie282a1672005-08-07 15:43:54 +1000878 BEGIN_DMA(6);
879 DMA_WRITE(swap_cmd);
880 DMA_WRITE(dev_priv->back_offset);
881 DMA_WRITE(dev_priv->back_bd);
Dave Airlie3528af12006-01-02 16:11:44 +1100882 DMA_WRITE(BCI_X_Y(boxes[i].x1, boxes[i].y1));
883 DMA_WRITE(BCI_X_Y(boxes[i].x1, boxes[i].y1));
884 DMA_WRITE(BCI_W_H(boxes[i].x2 - boxes[i].x1,
885 boxes[i].y2 - boxes[i].y1));
Dave Airlie282a1672005-08-07 15:43:54 +1000886 DMA_COMMIT();
887 }
888
889 return 0;
890}
891
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000892static int savage_dispatch_draw(drm_savage_private_t * dev_priv,
Dave Airlie3528af12006-01-02 16:11:44 +1100893 const drm_savage_cmd_header_t *start,
894 const drm_savage_cmd_header_t *end,
Dave Airlie056219e2007-07-11 16:17:42 +1000895 const struct drm_buf * dmabuf,
Dave Airlie3528af12006-01-02 16:11:44 +1100896 const unsigned int *vtxbuf,
Dave Airlie282a1672005-08-07 15:43:54 +1000897 unsigned int vb_size, unsigned int vb_stride,
898 unsigned int nbox,
Dave Airlieeddca552007-07-11 16:09:54 +1000899 const struct drm_clip_rect *boxes)
Dave Airlie282a1672005-08-07 15:43:54 +1000900{
901 unsigned int i, j;
902 int ret;
903
904 for (i = 0; i < nbox; ++i) {
Dave Airlie3528af12006-01-02 16:11:44 +1100905 const drm_savage_cmd_header_t *cmdbuf;
906 dev_priv->emit_clip_rect(dev_priv, &boxes[i]);
Dave Airlie282a1672005-08-07 15:43:54 +1000907
Dave Airlie3528af12006-01-02 16:11:44 +1100908 cmdbuf = start;
909 while (cmdbuf < end) {
Dave Airlie282a1672005-08-07 15:43:54 +1000910 drm_savage_cmd_header_t cmd_header;
Dave Airlie3528af12006-01-02 16:11:44 +1100911 cmd_header = *cmdbuf;
912 cmdbuf++;
Dave Airlie282a1672005-08-07 15:43:54 +1000913 switch (cmd_header.cmd.cmd) {
914 case SAVAGE_CMD_DMA_PRIM:
Dave Airlie3528af12006-01-02 16:11:44 +1100915 ret = savage_dispatch_dma_prim(
916 dev_priv, &cmd_header, dmabuf);
Dave Airlie282a1672005-08-07 15:43:54 +1000917 break;
918 case SAVAGE_CMD_VB_PRIM:
Dave Airlie3528af12006-01-02 16:11:44 +1100919 ret = savage_dispatch_vb_prim(
920 dev_priv, &cmd_header,
921 vtxbuf, vb_size, vb_stride);
Dave Airlie282a1672005-08-07 15:43:54 +1000922 break;
923 case SAVAGE_CMD_DMA_IDX:
924 j = (cmd_header.idx.count + 3) / 4;
925 /* j was check in savage_bci_cmdbuf */
Dave Airlie3528af12006-01-02 16:11:44 +1100926 ret = savage_dispatch_dma_idx(dev_priv,
927 &cmd_header, (const uint16_t *)cmdbuf,
928 dmabuf);
929 cmdbuf += j;
Dave Airlie282a1672005-08-07 15:43:54 +1000930 break;
931 case SAVAGE_CMD_VB_IDX:
932 j = (cmd_header.idx.count + 3) / 4;
933 /* j was check in savage_bci_cmdbuf */
Dave Airlie3528af12006-01-02 16:11:44 +1100934 ret = savage_dispatch_vb_idx(dev_priv,
935 &cmd_header, (const uint16_t *)cmdbuf,
936 (const uint32_t *)vtxbuf, vb_size,
937 vb_stride);
938 cmdbuf += j;
Dave Airlie282a1672005-08-07 15:43:54 +1000939 break;
940 default:
941 /* What's the best return code? EFAULT? */
942 DRM_ERROR("IMPLEMENTATION ERROR: "
943 "non-drawing-command %d\n",
944 cmd_header.cmd.cmd);
Eric Anholt20caafa2007-08-25 19:22:43 +1000945 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000946 }
947
948 if (ret != 0)
949 return ret;
950 }
951 }
952
953 return 0;
954}
955
956int savage_bci_cmdbuf(DRM_IOCTL_ARGS)
957{
958 DRM_DEVICE;
959 drm_savage_private_t *dev_priv = dev->dev_private;
Dave Airliecdd55a22007-07-11 16:32:08 +1000960 struct drm_device_dma *dma = dev->dma;
Dave Airlie056219e2007-07-11 16:17:42 +1000961 struct drm_buf *dmabuf;
Dave Airlie282a1672005-08-07 15:43:54 +1000962 drm_savage_cmdbuf_t cmdbuf;
Dave Airlie3528af12006-01-02 16:11:44 +1100963 drm_savage_cmd_header_t *kcmd_addr = NULL;
964 drm_savage_cmd_header_t *first_draw_cmd;
965 unsigned int *kvb_addr = NULL;
Dave Airlieeddca552007-07-11 16:09:54 +1000966 struct drm_clip_rect *kbox_addr = NULL;
Dave Airlie282a1672005-08-07 15:43:54 +1000967 unsigned int i, j;
968 int ret = 0;
969
970 DRM_DEBUG("\n");
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000971
Eric Anholt6c340ea2007-08-25 20:23:09 +1000972 LOCK_TEST_WITH_RETURN(dev, file_priv);
Dave Airlie282a1672005-08-07 15:43:54 +1000973
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000974 DRM_COPY_FROM_USER_IOCTL(cmdbuf, (drm_savage_cmdbuf_t __user *) data,
Dave Airlie282a1672005-08-07 15:43:54 +1000975 sizeof(cmdbuf));
976
977 if (dma && dma->buflist) {
978 if (cmdbuf.dma_idx > dma->buf_count) {
Dave Airlieb5e89ed2005-09-25 14:28:13 +1000979 DRM_ERROR
980 ("vertex buffer index %u out of range (0-%u)\n",
981 cmdbuf.dma_idx, dma->buf_count - 1);
Eric Anholt20caafa2007-08-25 19:22:43 +1000982 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +1000983 }
984 dmabuf = dma->buflist[cmdbuf.dma_idx];
985 } else {
986 dmabuf = NULL;
987 }
988
Dave Airlie3528af12006-01-02 16:11:44 +1100989 /* Copy the user buffers into kernel temporary areas. This hasn't been
990 * a performance loss compared to VERIFYAREA_READ/
991 * COPY_FROM_USER_UNCHECKED when done in other drivers, and is correct
992 * for locking on FreeBSD.
993 */
994 if (cmdbuf.size) {
995 kcmd_addr = drm_alloc(cmdbuf.size * 8, DRM_MEM_DRIVER);
996 if (kcmd_addr == NULL)
Eric Anholt20caafa2007-08-25 19:22:43 +1000997 return -ENOMEM;
Dave Airlie3528af12006-01-02 16:11:44 +1100998
999 if (DRM_COPY_FROM_USER(kcmd_addr, cmdbuf.cmd_addr,
1000 cmdbuf.size * 8))
1001 {
1002 drm_free(kcmd_addr, cmdbuf.size * 8, DRM_MEM_DRIVER);
Eric Anholt20caafa2007-08-25 19:22:43 +10001003 return -EFAULT;
Dave Airlie3528af12006-01-02 16:11:44 +11001004 }
1005 cmdbuf.cmd_addr = kcmd_addr;
1006 }
1007 if (cmdbuf.vb_size) {
1008 kvb_addr = drm_alloc(cmdbuf.vb_size, DRM_MEM_DRIVER);
1009 if (kvb_addr == NULL) {
Eric Anholt20caafa2007-08-25 19:22:43 +10001010 ret = -ENOMEM;
Dave Airlie3528af12006-01-02 16:11:44 +11001011 goto done;
1012 }
1013
1014 if (DRM_COPY_FROM_USER(kvb_addr, cmdbuf.vb_addr,
1015 cmdbuf.vb_size)) {
Eric Anholt20caafa2007-08-25 19:22:43 +10001016 ret = -EFAULT;
Dave Airlie3528af12006-01-02 16:11:44 +11001017 goto done;
1018 }
1019 cmdbuf.vb_addr = kvb_addr;
1020 }
1021 if (cmdbuf.nbox) {
Dave Airlieeddca552007-07-11 16:09:54 +10001022 kbox_addr = drm_alloc(cmdbuf.nbox * sizeof(struct drm_clip_rect),
Dave Airlie3528af12006-01-02 16:11:44 +11001023 DRM_MEM_DRIVER);
1024 if (kbox_addr == NULL) {
Eric Anholt20caafa2007-08-25 19:22:43 +10001025 ret = -ENOMEM;
Dave Airlie3528af12006-01-02 16:11:44 +11001026 goto done;
1027 }
1028
1029 if (DRM_COPY_FROM_USER(kbox_addr, cmdbuf.box_addr,
Dave Airlieeddca552007-07-11 16:09:54 +10001030 cmdbuf.nbox * sizeof(struct drm_clip_rect))) {
Eric Anholt20caafa2007-08-25 19:22:43 +10001031 ret = -EFAULT;
Dave Airlie3528af12006-01-02 16:11:44 +11001032 goto done;
1033 }
1034 cmdbuf.box_addr = kbox_addr;
1035 }
Dave Airlie282a1672005-08-07 15:43:54 +10001036
1037 /* Make sure writes to DMA buffers are finished before sending
1038 * DMA commands to the graphics hardware. */
1039 DRM_MEMORYBARRIER();
1040
1041 /* Coming from user space. Don't know if the Xserver has
1042 * emitted wait commands. Assuming the worst. */
1043 dev_priv->waiting = 1;
1044
1045 i = 0;
1046 first_draw_cmd = NULL;
1047 while (i < cmdbuf.size) {
1048 drm_savage_cmd_header_t cmd_header;
Dave Airlie3528af12006-01-02 16:11:44 +11001049 cmd_header = *(drm_savage_cmd_header_t *)cmdbuf.cmd_addr;
1050 cmdbuf.cmd_addr++;
Dave Airlie282a1672005-08-07 15:43:54 +10001051 i++;
1052
1053 /* Group drawing commands with same state to minimize
1054 * iterations over clip rects. */
1055 j = 0;
1056 switch (cmd_header.cmd.cmd) {
1057 case SAVAGE_CMD_DMA_IDX:
1058 case SAVAGE_CMD_VB_IDX:
1059 j = (cmd_header.idx.count + 3) / 4;
1060 if (i + j > cmdbuf.size) {
1061 DRM_ERROR("indexed drawing command extends "
1062 "beyond end of command buffer\n");
1063 DMA_FLUSH();
Eric Anholt20caafa2007-08-25 19:22:43 +10001064 return -EINVAL;
Dave Airlie282a1672005-08-07 15:43:54 +10001065 }
1066 /* fall through */
1067 case SAVAGE_CMD_DMA_PRIM:
1068 case SAVAGE_CMD_VB_PRIM:
1069 if (!first_draw_cmd)
Dave Airlie3528af12006-01-02 16:11:44 +11001070 first_draw_cmd = cmdbuf.cmd_addr - 1;
1071 cmdbuf.cmd_addr += j;
Dave Airlie282a1672005-08-07 15:43:54 +10001072 i += j;
1073 break;
1074 default:
1075 if (first_draw_cmd) {
Dave Airlie3528af12006-01-02 16:11:44 +11001076 ret = savage_dispatch_draw(
1077 dev_priv, first_draw_cmd,
1078 cmdbuf.cmd_addr - 1,
1079 dmabuf, cmdbuf.vb_addr, cmdbuf.vb_size,
1080 cmdbuf.vb_stride,
1081 cmdbuf.nbox, cmdbuf.box_addr);
Dave Airlie282a1672005-08-07 15:43:54 +10001082 if (ret != 0)
1083 return ret;
1084 first_draw_cmd = NULL;
1085 }
1086 }
1087 if (first_draw_cmd)
1088 continue;
1089
1090 switch (cmd_header.cmd.cmd) {
1091 case SAVAGE_CMD_STATE:
1092 j = (cmd_header.state.count + 1) / 2;
1093 if (i + j > cmdbuf.size) {
1094 DRM_ERROR("command SAVAGE_CMD_STATE extends "
1095 "beyond end of command buffer\n");
1096 DMA_FLUSH();
Eric Anholt20caafa2007-08-25 19:22:43 +10001097 ret = -EINVAL;
Dave Airlie3528af12006-01-02 16:11:44 +11001098 goto done;
Dave Airlie282a1672005-08-07 15:43:54 +10001099 }
Dave Airlieb5e89ed2005-09-25 14:28:13 +10001100 ret = savage_dispatch_state(dev_priv, &cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +11001101 (const uint32_t *)cmdbuf.cmd_addr);
1102 cmdbuf.cmd_addr += j;
Dave Airlie282a1672005-08-07 15:43:54 +10001103 i += j;
1104 break;
1105 case SAVAGE_CMD_CLEAR:
1106 if (i + 1 > cmdbuf.size) {
1107 DRM_ERROR("command SAVAGE_CMD_CLEAR extends "
1108 "beyond end of command buffer\n");
1109 DMA_FLUSH();
Eric Anholt20caafa2007-08-25 19:22:43 +10001110 ret = -EINVAL;
Dave Airlie3528af12006-01-02 16:11:44 +11001111 goto done;
Dave Airlie282a1672005-08-07 15:43:54 +10001112 }
1113 ret = savage_dispatch_clear(dev_priv, &cmd_header,
Dave Airlie3528af12006-01-02 16:11:44 +11001114 cmdbuf.cmd_addr,
1115 cmdbuf.nbox, cmdbuf.box_addr);
1116 cmdbuf.cmd_addr++;
Dave Airlie282a1672005-08-07 15:43:54 +10001117 i++;
1118 break;
1119 case SAVAGE_CMD_SWAP:
Dave Airlie3528af12006-01-02 16:11:44 +11001120 ret = savage_dispatch_swap(dev_priv, cmdbuf.nbox,
1121 cmdbuf.box_addr);
Dave Airlie282a1672005-08-07 15:43:54 +10001122 break;
1123 default:
1124 DRM_ERROR("invalid command 0x%x\n", cmd_header.cmd.cmd);
1125 DMA_FLUSH();
Eric Anholt20caafa2007-08-25 19:22:43 +10001126 ret = -EINVAL;
Dave Airlie3528af12006-01-02 16:11:44 +11001127 goto done;
Dave Airlie282a1672005-08-07 15:43:54 +10001128 }
1129
1130 if (ret != 0) {
1131 DMA_FLUSH();
Dave Airlie3528af12006-01-02 16:11:44 +11001132 goto done;
Dave Airlie282a1672005-08-07 15:43:54 +10001133 }
1134 }
1135
1136 if (first_draw_cmd) {
Dave Airlie3528af12006-01-02 16:11:44 +11001137 ret = savage_dispatch_draw (
1138 dev_priv, first_draw_cmd, cmdbuf.cmd_addr, dmabuf,
1139 cmdbuf.vb_addr, cmdbuf.vb_size, cmdbuf.vb_stride,
1140 cmdbuf.nbox, cmdbuf.box_addr);
Dave Airlie282a1672005-08-07 15:43:54 +10001141 if (ret != 0) {
1142 DMA_FLUSH();
Dave Airlie3528af12006-01-02 16:11:44 +11001143 goto done;
Dave Airlie282a1672005-08-07 15:43:54 +10001144 }
1145 }
1146
1147 DMA_FLUSH();
1148
1149 if (dmabuf && cmdbuf.discard) {
1150 drm_savage_buf_priv_t *buf_priv = dmabuf->dev_private;
1151 uint16_t event;
1152 event = savage_bci_emit_event(dev_priv, SAVAGE_WAIT_3D);
1153 SET_AGE(&buf_priv->age, event, dev_priv->event_wrap);
1154 savage_freelist_put(dev, dmabuf);
1155 }
1156
Dave Airlie3528af12006-01-02 16:11:44 +11001157done:
1158 /* If we didn't need to allocate them, these'll be NULL */
1159 drm_free(kcmd_addr, cmdbuf.size * 8, DRM_MEM_DRIVER);
1160 drm_free(kvb_addr, cmdbuf.vb_size, DRM_MEM_DRIVER);
Dave Airlieeddca552007-07-11 16:09:54 +10001161 drm_free(kbox_addr, cmdbuf.nbox * sizeof(struct drm_clip_rect),
Dave Airlie3528af12006-01-02 16:11:44 +11001162 DRM_MEM_DRIVER);
1163
1164 return ret;
Dave Airlie282a1672005-08-07 15:43:54 +10001165}