Clone of mesa.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

nv50_vbo.c 8.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. /*
  2. * Copyright 2008 Ben Skeggs
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a
  5. * copy of this software and associated documentation files (the "Software"),
  6. * to deal in the Software without restriction, including without limitation
  7. * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. * and/or sell copies of the Software, and to permit persons to whom the
  9. * Software is furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
  17. * THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  18. * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
  19. * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. * SOFTWARE.
  21. */
  22. #include "pipe/p_context.h"
  23. #include "pipe/p_state.h"
  24. #include "pipe/p_inlines.h"
  25. #include "nv50_context.h"
  26. static INLINE unsigned
  27. nv50_prim(unsigned mode)
  28. {
  29. switch (mode) {
  30. case PIPE_PRIM_POINTS: return NV50TCL_VERTEX_BEGIN_POINTS;
  31. case PIPE_PRIM_LINES: return NV50TCL_VERTEX_BEGIN_LINES;
  32. case PIPE_PRIM_LINE_LOOP: return NV50TCL_VERTEX_BEGIN_LINE_LOOP;
  33. case PIPE_PRIM_LINE_STRIP: return NV50TCL_VERTEX_BEGIN_LINE_STRIP;
  34. case PIPE_PRIM_TRIANGLES: return NV50TCL_VERTEX_BEGIN_TRIANGLES;
  35. case PIPE_PRIM_TRIANGLE_STRIP:
  36. return NV50TCL_VERTEX_BEGIN_TRIANGLE_STRIP;
  37. case PIPE_PRIM_TRIANGLE_FAN: return NV50TCL_VERTEX_BEGIN_TRIANGLE_FAN;
  38. case PIPE_PRIM_QUADS: return NV50TCL_VERTEX_BEGIN_QUADS;
  39. case PIPE_PRIM_QUAD_STRIP: return NV50TCL_VERTEX_BEGIN_QUAD_STRIP;
  40. case PIPE_PRIM_POLYGON: return NV50TCL_VERTEX_BEGIN_POLYGON;
  41. default:
  42. break;
  43. }
  44. NOUVEAU_ERR("invalid primitive type %d\n", mode);
  45. return NV50TCL_VERTEX_BEGIN_POINTS;
  46. }
  47. static INLINE uint32_t
  48. nv50_vbo_type_to_hw(unsigned type)
  49. {
  50. switch (type) {
  51. case PIPE_FORMAT_TYPE_FLOAT:
  52. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_FLOAT;
  53. case PIPE_FORMAT_TYPE_UNORM:
  54. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_UNORM;
  55. case PIPE_FORMAT_TYPE_SNORM:
  56. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_SNORM;
  57. case PIPE_FORMAT_TYPE_USCALED:
  58. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_USCALED;
  59. case PIPE_FORMAT_TYPE_SSCALED:
  60. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_SSCALED;
  61. /*
  62. case PIPE_FORMAT_TYPE_UINT:
  63. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_UINT;
  64. case PIPE_FORMAT_TYPE_SINT:
  65. return NV50TCL_VERTEX_ARRAY_ATTRIB_TYPE_SINT; */
  66. default:
  67. return 0;
  68. }
  69. }
  70. static INLINE uint32_t
  71. nv50_vbo_size_to_hw(unsigned size, unsigned nr_c)
  72. {
  73. static const uint32_t hw_values[] = {
  74. 0, 0, 0, 0,
  75. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_8,
  76. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_8_8,
  77. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_8_8_8,
  78. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_8_8_8_8,
  79. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_16,
  80. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_16_16,
  81. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_16_16_16,
  82. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_16_16_16_16,
  83. 0, 0, 0, 0,
  84. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_32,
  85. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_32_32,
  86. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_32_32_32,
  87. NV50TCL_VERTEX_ARRAY_ATTRIB_SIZE_32_32_32_32 };
  88. /* we'd also have R11G11B10 and R10G10B10A2 */
  89. assert(nr_c > 0 && nr_c <= 4);
  90. if (size > 32)
  91. return 0;
  92. size >>= (3 - 2);
  93. return hw_values[size + (nr_c - 1)];
  94. }
  95. static INLINE uint32_t
  96. nv50_vbo_vtxelt_to_hw(struct pipe_vertex_element *ve)
  97. {
  98. uint32_t hw_type, hw_size;
  99. enum pipe_format pf = ve->src_format;
  100. unsigned size = pf_size_x(pf) << pf_exp2(pf);
  101. hw_type = nv50_vbo_type_to_hw(pf_type(pf));
  102. hw_size = nv50_vbo_size_to_hw(size, ve->nr_components);
  103. if (!hw_type || !hw_size) {
  104. NOUVEAU_ERR("unsupported vbo format: %s\n", pf_name(pf));
  105. abort();
  106. return 0x24e80000;
  107. }
  108. if (pf_swizzle_x(pf) == 2) /* BGRA */
  109. hw_size |= (1 << 31); /* no real swizzle bits :-( */
  110. return (hw_type | hw_size);
  111. }
  112. boolean
  113. nv50_draw_arrays(struct pipe_context *pipe, unsigned mode, unsigned start,
  114. unsigned count)
  115. {
  116. struct nv50_context *nv50 = nv50_context(pipe);
  117. struct nouveau_channel *chan = nv50->screen->tesla->channel;
  118. struct nouveau_grobj *tesla = nv50->screen->tesla;
  119. nv50_state_validate(nv50);
  120. BEGIN_RING(chan, tesla, 0x142c, 1);
  121. OUT_RING (chan, 0);
  122. BEGIN_RING(chan, tesla, 0x142c, 1);
  123. OUT_RING (chan, 0);
  124. BEGIN_RING(chan, tesla, 0x1440, 1);
  125. OUT_RING (chan, 0);
  126. BEGIN_RING(chan, tesla, 0x1334, 1);
  127. OUT_RING (chan, 0);
  128. BEGIN_RING(chan, tesla, NV50TCL_VERTEX_BEGIN, 1);
  129. OUT_RING (chan, nv50_prim(mode));
  130. BEGIN_RING(chan, tesla, NV50TCL_VERTEX_BUFFER_FIRST, 2);
  131. OUT_RING (chan, start);
  132. OUT_RING (chan, count);
  133. BEGIN_RING(chan, tesla, NV50TCL_VERTEX_END, 1);
  134. OUT_RING (chan, 0);
  135. pipe->flush(pipe, 0, NULL);
  136. return TRUE;
  137. }
  138. static INLINE void
  139. nv50_draw_elements_inline_u08(struct nv50_context *nv50, uint8_t *map,
  140. unsigned start, unsigned count)
  141. {
  142. struct nouveau_channel *chan = nv50->screen->tesla->channel;
  143. struct nouveau_grobj *tesla = nv50->screen->tesla;
  144. map += start;
  145. if (count & 1) {
  146. BEGIN_RING(chan, tesla, 0x15e8, 1);
  147. OUT_RING (chan, map[0]);
  148. map++;
  149. count--;
  150. }
  151. while (count) {
  152. unsigned nr = count > 2046 ? 2046 : count;
  153. int i;
  154. BEGIN_RING(chan, tesla, 0x400015f0, nr >> 1);
  155. for (i = 0; i < nr; i += 2)
  156. OUT_RING (chan, (map[1] << 16) | map[0]);
  157. count -= nr;
  158. map += nr;
  159. }
  160. }
  161. static INLINE void
  162. nv50_draw_elements_inline_u16(struct nv50_context *nv50, uint16_t *map,
  163. unsigned start, unsigned count)
  164. {
  165. struct nouveau_channel *chan = nv50->screen->tesla->channel;
  166. struct nouveau_grobj *tesla = nv50->screen->tesla;
  167. map += start;
  168. if (count & 1) {
  169. BEGIN_RING(chan, tesla, 0x15e8, 1);
  170. OUT_RING (chan, map[0]);
  171. map++;
  172. count--;
  173. }
  174. while (count) {
  175. unsigned nr = count > 2046 ? 2046 : count;
  176. int i;
  177. BEGIN_RING(chan, tesla, 0x400015f0, nr >> 1);
  178. for (i = 0; i < nr; i += 2)
  179. OUT_RING (chan, (map[1] << 16) | map[0]);
  180. count -= nr;
  181. map += nr;
  182. }
  183. }
  184. static INLINE void
  185. nv50_draw_elements_inline_u32(struct nv50_context *nv50, uint32_t *map,
  186. unsigned start, unsigned count)
  187. {
  188. struct nouveau_channel *chan = nv50->screen->tesla->channel;
  189. struct nouveau_grobj *tesla = nv50->screen->tesla;
  190. map += start;
  191. while (count) {
  192. unsigned nr = count > 2047 ? 2047 : count;
  193. BEGIN_RING(chan, tesla, 0x400015e8, nr);
  194. OUT_RINGp (chan, map, nr);
  195. count -= nr;
  196. map += nr;
  197. }
  198. }
  199. boolean
  200. nv50_draw_elements(struct pipe_context *pipe,
  201. struct pipe_buffer *indexBuffer, unsigned indexSize,
  202. unsigned mode, unsigned start, unsigned count)
  203. {
  204. struct nv50_context *nv50 = nv50_context(pipe);
  205. struct nouveau_channel *chan = nv50->screen->tesla->channel;
  206. struct nouveau_grobj *tesla = nv50->screen->tesla;
  207. struct pipe_screen *pscreen = pipe->screen;
  208. void *map;
  209. map = pipe_buffer_map(pscreen, indexBuffer, PIPE_BUFFER_USAGE_CPU_READ);
  210. nv50_state_validate(nv50);
  211. BEGIN_RING(chan, tesla, 0x142c, 1);
  212. OUT_RING (chan, 0);
  213. BEGIN_RING(chan, tesla, 0x142c, 1);
  214. OUT_RING (chan, 0);
  215. BEGIN_RING(chan, tesla, NV50TCL_VERTEX_BEGIN, 1);
  216. OUT_RING (chan, nv50_prim(mode));
  217. switch (indexSize) {
  218. case 1:
  219. nv50_draw_elements_inline_u08(nv50, map, start, count);
  220. break;
  221. case 2:
  222. nv50_draw_elements_inline_u16(nv50, map, start, count);
  223. break;
  224. case 4:
  225. nv50_draw_elements_inline_u32(nv50, map, start, count);
  226. break;
  227. default:
  228. assert(0);
  229. }
  230. BEGIN_RING(chan, tesla, NV50TCL_VERTEX_END, 1);
  231. OUT_RING (chan, 0);
  232. pipe_buffer_unmap(pscreen, indexBuffer);
  233. pipe->flush(pipe, 0, NULL);
  234. return TRUE;
  235. }
  236. void
  237. nv50_vbo_validate(struct nv50_context *nv50)
  238. {
  239. struct nouveau_grobj *tesla = nv50->screen->tesla;
  240. struct nouveau_stateobj *vtxbuf, *vtxfmt;
  241. int i;
  242. /* don't validate if Gallium took away our buffers */
  243. if (nv50->vtxbuf_nr == 0)
  244. return;
  245. vtxbuf = so_new(nv50->vtxelt_nr * 4, nv50->vtxelt_nr * 2);
  246. vtxfmt = so_new(nv50->vtxelt_nr + 1, 0);
  247. so_method(vtxfmt, tesla, NV50TCL_VERTEX_ARRAY_ATTRIB(0),
  248. nv50->vtxelt_nr);
  249. for (i = 0; i < nv50->vtxelt_nr; i++) {
  250. struct pipe_vertex_element *ve = &nv50->vtxelt[i];
  251. struct pipe_vertex_buffer *vb =
  252. &nv50->vtxbuf[ve->vertex_buffer_index];
  253. struct nouveau_bo *bo = nouveau_bo(vb->buffer);
  254. uint32_t hw = nv50_vbo_vtxelt_to_hw(ve);
  255. so_data(vtxfmt, hw | i);
  256. so_method(vtxbuf, tesla, NV50TCL_VERTEX_ARRAY_FORMAT(i), 3);
  257. so_data (vtxbuf, 0x20000000 | vb->stride);
  258. so_reloc (vtxbuf, bo, vb->buffer_offset +
  259. ve->src_offset, NOUVEAU_BO_VRAM | NOUVEAU_BO_GART |
  260. NOUVEAU_BO_RD | NOUVEAU_BO_HIGH, 0, 0);
  261. so_reloc (vtxbuf, bo, vb->buffer_offset +
  262. ve->src_offset, NOUVEAU_BO_VRAM | NOUVEAU_BO_GART |
  263. NOUVEAU_BO_RD | NOUVEAU_BO_LOW, 0, 0);
  264. }
  265. so_ref (vtxfmt, &nv50->state.vtxfmt);
  266. so_ref (vtxbuf, &nv50->state.vtxbuf);
  267. so_ref (NULL, &vtxbuf);
  268. so_ref (NULL, &vtxfmt);
  269. }