gallivm: Updated lp_build_const_mask_aos to input number of channels.
[mesa.git] / src / gallium / drivers / llvmpipe / lp_state_setup.c
1 /**************************************************************************
2 *
3 * Copyright 2010 VMware.
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
8 * "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sub license, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
13 *
14 * The above copyright notice and this permission notice (including the
15 * next paragraph) shall be included in all copies or substantial portions
16 * of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21 * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
22 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
25 *
26 **************************************************************************/
27
28
29 #include "util/u_math.h"
30 #include "util/u_memory.h"
31 #include "util/u_simple_list.h"
32 #include "os/os_time.h"
33 #include "gallivm/lp_bld_arit.h"
34 #include "gallivm/lp_bld_const.h"
35 #include "gallivm/lp_bld_debug.h"
36 #include "gallivm/lp_bld_init.h"
37 #include "gallivm/lp_bld_logic.h"
38 #include "gallivm/lp_bld_intr.h"
39 #include "gallivm/lp_bld_flow.h"
40 #include "gallivm/lp_bld_type.h"
41
42 #include "lp_perf.h"
43 #include "lp_debug.h"
44 #include "lp_flush.h"
45 #include "lp_screen.h"
46 #include "lp_context.h"
47 #include "lp_state.h"
48 #include "lp_state_fs.h"
49 #include "lp_state_setup.h"
50
51
52
53 /* currently organized to interpolate full float[4] attributes even
54 * when some elements are unused. Later, can pack vertex data more
55 * closely.
56 */
57
58
59 struct lp_setup_args
60 {
61 /* Function arguments:
62 */
63 LLVMValueRef v0;
64 LLVMValueRef v1;
65 LLVMValueRef v2;
66 LLVMValueRef facing; /* boolean */
67 LLVMValueRef a0;
68 LLVMValueRef dadx;
69 LLVMValueRef dady;
70
71 /* Derived:
72 */
73 LLVMValueRef x0_center;
74 LLVMValueRef y0_center;
75 LLVMValueRef dy20_ooa;
76 LLVMValueRef dy01_ooa;
77 LLVMValueRef dx20_ooa;
78 LLVMValueRef dx01_ooa;
79 };
80
81
82
83 static LLVMTypeRef
84 type4f(struct gallivm_state *gallivm)
85 {
86 return LLVMVectorType(LLVMFloatTypeInContext(gallivm->context), 4);
87 }
88
89
90 /* Equivalent of _mm_setr_ps(a,b,c,d)
91 */
92 static LLVMValueRef
93 vec4f(struct gallivm_state *gallivm,
94 LLVMValueRef a, LLVMValueRef b, LLVMValueRef c, LLVMValueRef d,
95 const char *name)
96 {
97 LLVMBuilderRef bld = gallivm->builder;
98 LLVMValueRef i0 = lp_build_const_int32(gallivm, 0);
99 LLVMValueRef i1 = lp_build_const_int32(gallivm, 1);
100 LLVMValueRef i2 = lp_build_const_int32(gallivm, 2);
101 LLVMValueRef i3 = lp_build_const_int32(gallivm, 3);
102
103 LLVMValueRef res = LLVMGetUndef(type4f(gallivm));
104
105 res = LLVMBuildInsertElement(bld, res, a, i0, "");
106 res = LLVMBuildInsertElement(bld, res, b, i1, "");
107 res = LLVMBuildInsertElement(bld, res, c, i2, "");
108 res = LLVMBuildInsertElement(bld, res, d, i3, name);
109
110 return res;
111 }
112
113 /* Equivalent of _mm_set1_ps(a)
114 */
115 static LLVMValueRef
116 vec4f_from_scalar(struct gallivm_state *gallivm,
117 LLVMValueRef a,
118 const char *name)
119 {
120 LLVMBuilderRef bld = gallivm->builder;
121 LLVMValueRef res = LLVMGetUndef(type4f(gallivm));
122 int i;
123
124 for(i = 0; i < 4; ++i) {
125 LLVMValueRef index = lp_build_const_int32(gallivm, i);
126 res = LLVMBuildInsertElement(bld, res, a, index, i == 3 ? name : "");
127 }
128
129 return res;
130 }
131
132 static void
133 store_coef(struct gallivm_state *gallivm,
134 struct lp_setup_args *args,
135 unsigned slot,
136 LLVMValueRef a0,
137 LLVMValueRef dadx,
138 LLVMValueRef dady)
139 {
140 LLVMBuilderRef builder = gallivm->builder;
141 LLVMValueRef idx = lp_build_const_int32(gallivm, slot);
142
143 LLVMBuildStore(builder,
144 a0,
145 LLVMBuildGEP(builder, args->a0, &idx, 1, ""));
146
147 LLVMBuildStore(builder,
148 dadx,
149 LLVMBuildGEP(builder, args->dadx, &idx, 1, ""));
150
151 LLVMBuildStore(builder,
152 dady,
153 LLVMBuildGEP(builder, args->dady, &idx, 1, ""));
154 }
155
156
157
158 static void
159 emit_constant_coef4(struct gallivm_state *gallivm,
160 struct lp_setup_args *args,
161 unsigned slot,
162 LLVMValueRef vert)
163 {
164 LLVMValueRef zero = lp_build_const_float(gallivm, 0.0);
165 LLVMValueRef zerovec = vec4f_from_scalar(gallivm, zero, "zero");
166 store_coef(gallivm, args, slot, vert, zerovec, zerovec);
167 }
168
169
170
171 /**
172 * Setup the fragment input attribute with the front-facing value.
173 * \param frontface is the triangle front facing?
174 */
175 static void
176 emit_facing_coef(struct gallivm_state *gallivm,
177 struct lp_setup_args *args,
178 unsigned slot )
179 {
180 LLVMBuilderRef builder = gallivm->builder;
181 LLVMTypeRef float_type = LLVMFloatTypeInContext(gallivm->context);
182 LLVMValueRef a0_0 = args->facing;
183 LLVMValueRef a0_0f = LLVMBuildSIToFP(builder, a0_0, float_type, "");
184 LLVMValueRef zero = lp_build_const_float(gallivm, 0.0);
185 LLVMValueRef a0 = vec4f(gallivm, a0_0f, zero, zero, zero, "facing");
186 LLVMValueRef zerovec = vec4f_from_scalar(gallivm, zero, "zero");
187
188 store_coef(gallivm, args, slot, a0, zerovec, zerovec);
189 }
190
191
192 static LLVMValueRef
193 vert_attrib(struct gallivm_state *gallivm,
194 LLVMValueRef vert,
195 int attr,
196 int elem,
197 const char *name)
198 {
199 LLVMBuilderRef b = gallivm->builder;
200 LLVMValueRef idx[2];
201 idx[0] = lp_build_const_int32(gallivm, attr);
202 idx[1] = lp_build_const_int32(gallivm, elem);
203 return LLVMBuildLoad(b, LLVMBuildGEP(b, vert, idx, 2, ""), name);
204 }
205
206
207 static void
208 lp_twoside(struct gallivm_state *gallivm,
209 struct lp_setup_args *args,
210 const struct lp_setup_variant_key *key,
211 int bcolor_slot,
212 LLVMValueRef attribv[3])
213 {
214 LLVMBuilderRef b = gallivm->builder;
215 LLVMValueRef a0_back, a1_back, a2_back;
216 LLVMValueRef idx2 = lp_build_const_int32(gallivm, bcolor_slot);
217
218 LLVMValueRef facing = args->facing;
219 LLVMValueRef front_facing = LLVMBuildICmp(b, LLVMIntEQ, facing, lp_build_const_int32(gallivm, 0), ""); /** need i1 for if condition */
220
221 a0_back = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v0, &idx2, 1, ""), "v0a_back");
222 a1_back = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v1, &idx2, 1, ""), "v1a_back");
223 a2_back = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v2, &idx2, 1, ""), "v2a_back");
224
225 /* Possibly swap the front and back attrib values,
226 *
227 * Prefer select to if so we don't have to worry about phis or
228 * allocas.
229 */
230 attribv[0] = LLVMBuildSelect(b, front_facing, a0_back, attribv[0], "");
231 attribv[1] = LLVMBuildSelect(b, front_facing, a1_back, attribv[1], "");
232 attribv[2] = LLVMBuildSelect(b, front_facing, a2_back, attribv[2], "");
233
234 }
235
236 static void
237 lp_do_offset_tri(struct gallivm_state *gallivm,
238 struct lp_setup_args *args,
239 const struct lp_setup_variant_key *key,
240 LLVMValueRef inv_det,
241 LLVMValueRef dxyz01,
242 LLVMValueRef dxyz20,
243 LLVMValueRef attribv[3])
244 {
245 LLVMBuilderRef b = gallivm->builder;
246 struct lp_build_context bld;
247 LLVMValueRef zoffset, mult;
248 LLVMValueRef z0_new, z1_new, z2_new;
249 LLVMValueRef dzdxdzdy, dzdx, dzdy, dzxyz20, dyzzx01, dyzzx01_dzxyz20, dzx01_dyz20;
250 LLVMValueRef z0z1, z0z1z2;
251 LLVMValueRef max, max_value, res12;
252 LLVMValueRef shuffles[4];
253 LLVMTypeRef shuf_type = LLVMInt32TypeInContext(gallivm->context);
254 LLVMValueRef onei = lp_build_const_int32(gallivm, 1);
255 LLVMValueRef zeroi = lp_build_const_int32(gallivm, 0);
256 LLVMValueRef twoi = lp_build_const_int32(gallivm, 2);
257 LLVMValueRef threei = lp_build_const_int32(gallivm, 3);
258
259 /* (res12) = cross(e,f).xy */
260 shuffles[0] = twoi;
261 shuffles[1] = zeroi;
262 shuffles[2] = onei;
263 shuffles[3] = twoi;
264 dzxyz20 = LLVMBuildShuffleVector(b, dxyz20, dxyz20, LLVMConstVector(shuffles, 4), "");
265
266 shuffles[0] = onei;
267 shuffles[1] = twoi;
268 shuffles[2] = twoi;
269 shuffles[3] = zeroi;
270 dyzzx01 = LLVMBuildShuffleVector(b, dxyz01, dxyz01, LLVMConstVector(shuffles, 4), "");
271
272 dyzzx01_dzxyz20 = LLVMBuildFMul(b, dzxyz20, dyzzx01, "dyzzx01_dzxyz20");
273
274 shuffles[0] = twoi;
275 shuffles[1] = threei;
276 shuffles[2] = LLVMGetUndef(shuf_type);
277 shuffles[3] = LLVMGetUndef(shuf_type);
278 dzx01_dyz20 = LLVMBuildShuffleVector(b, dyzzx01_dzxyz20, dyzzx01_dzxyz20,
279 LLVMConstVector(shuffles, 4), "");
280
281 res12 = LLVMBuildFSub(b, dyzzx01_dzxyz20, dzx01_dyz20, "res12");
282
283 /* dzdx = fabsf(res1 * inv_det), dydx = fabsf(res2 * inv_det)*/
284 lp_build_context_init(&bld, gallivm, lp_type_float_vec(32, 128));
285 dzdxdzdy = LLVMBuildFMul(b, res12, inv_det, "dzdxdzdy");
286 dzdxdzdy = lp_build_abs(&bld, dzdxdzdy);
287
288 dzdx = LLVMBuildExtractElement(b, dzdxdzdy, zeroi, "");
289 dzdy = LLVMBuildExtractElement(b, dzdxdzdy, onei, "");
290
291 /* zoffset = offset->units + MAX2(dzdx, dzdy) * offset->scale */
292 max = LLVMBuildFCmp(b, LLVMRealUGT, dzdx, dzdy, "");
293 max_value = LLVMBuildSelect(b, max, dzdx, dzdy, "max");
294
295 mult = LLVMBuildFMul(b, max_value, lp_build_const_float(gallivm, key->scale), "");
296 zoffset = LLVMBuildFAdd(b, lp_build_const_float(gallivm, key->units), mult, "zoffset");
297
298 /* yuck */
299 shuffles[0] = twoi;
300 shuffles[1] = lp_build_const_int32(gallivm, 6);
301 shuffles[2] = LLVMGetUndef(shuf_type);
302 shuffles[3] = LLVMGetUndef(shuf_type);
303 z0z1 = LLVMBuildShuffleVector(b, attribv[0], attribv[1], LLVMConstVector(shuffles, 4), "");
304 shuffles[0] = zeroi;
305 shuffles[1] = onei;
306 shuffles[2] = lp_build_const_int32(gallivm, 6);
307 shuffles[3] = LLVMGetUndef(shuf_type);
308 z0z1z2 = LLVMBuildShuffleVector(b, z0z1, attribv[2], LLVMConstVector(shuffles, 4), "");
309 zoffset = vec4f_from_scalar(gallivm, zoffset, "");
310
311 /* clamp and do offset */
312 z0z1z2 = lp_build_clamp(&bld, LLVMBuildFAdd(b, z0z1z2, zoffset, ""), bld.zero, bld.one);
313
314 /* insert into args->a0.z, a1.z, a2.z:
315 */
316 z0_new = LLVMBuildExtractElement(b, z0z1z2, zeroi, "");
317 z1_new = LLVMBuildExtractElement(b, z0z1z2, onei, "");
318 z2_new = LLVMBuildExtractElement(b, z0z1z2, twoi, "");
319 attribv[0] = LLVMBuildInsertElement(b, attribv[0], z0_new, twoi, "");
320 attribv[1] = LLVMBuildInsertElement(b, attribv[1], z1_new, twoi, "");
321 attribv[2] = LLVMBuildInsertElement(b, attribv[2], z2_new, twoi, "");
322 }
323
324 static void
325 load_attribute(struct gallivm_state *gallivm,
326 struct lp_setup_args *args,
327 const struct lp_setup_variant_key *key,
328 unsigned vert_attr,
329 LLVMValueRef attribv[3])
330 {
331 LLVMBuilderRef b = gallivm->builder;
332 LLVMValueRef idx = lp_build_const_int32(gallivm, vert_attr);
333
334 /* Load the vertex data
335 */
336 attribv[0] = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v0, &idx, 1, ""), "v0a");
337 attribv[1] = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v1, &idx, 1, ""), "v1a");
338 attribv[2] = LLVMBuildLoad(b, LLVMBuildGEP(b, args->v2, &idx, 1, ""), "v2a");
339
340
341 /* Potentially modify it according to twoside, etc:
342 */
343 if (key->twoside) {
344 if (vert_attr == key->color_slot && key->bcolor_slot >= 0)
345 lp_twoside(gallivm, args, key, key->bcolor_slot, attribv);
346 else if (vert_attr == key->spec_slot && key->bspec_slot >= 0)
347 lp_twoside(gallivm, args, key, key->bspec_slot, attribv);
348 }
349 }
350
351 static void
352 emit_coef4( struct gallivm_state *gallivm,
353 struct lp_setup_args *args,
354 unsigned slot,
355 LLVMValueRef a0,
356 LLVMValueRef a1,
357 LLVMValueRef a2)
358 {
359 LLVMBuilderRef b = gallivm->builder;
360 LLVMValueRef dy20_ooa = args->dy20_ooa;
361 LLVMValueRef dy01_ooa = args->dy01_ooa;
362 LLVMValueRef dx20_ooa = args->dx20_ooa;
363 LLVMValueRef dx01_ooa = args->dx01_ooa;
364 LLVMValueRef x0_center = args->x0_center;
365 LLVMValueRef y0_center = args->y0_center;
366
367 LLVMValueRef da01 = LLVMBuildFSub(b, a0, a1, "da01");
368 LLVMValueRef da20 = LLVMBuildFSub(b, a2, a0, "da20");
369
370 /* Calculate dadx (vec4f)
371 */
372 LLVMValueRef da01_dy20_ooa = LLVMBuildFMul(b, da01, dy20_ooa, "da01_dy20_ooa");
373 LLVMValueRef da20_dy01_ooa = LLVMBuildFMul(b, da20, dy01_ooa, "da20_dy01_ooa");
374 LLVMValueRef dadx = LLVMBuildFSub(b, da01_dy20_ooa, da20_dy01_ooa, "dadx");
375
376 /* Calculate dady (vec4f)
377 */
378 LLVMValueRef da01_dx20_ooa = LLVMBuildFMul(b, da01, dx20_ooa, "da01_dx20_ooa");
379 LLVMValueRef da20_dx01_ooa = LLVMBuildFMul(b, da20, dx01_ooa, "da20_dx01_ooa");
380 LLVMValueRef dady = LLVMBuildFSub(b, da20_dx01_ooa, da01_dx20_ooa, "dady");
381
382 /* Calculate a0 - the attribute value at the origin
383 */
384 LLVMValueRef dadx_x0 = LLVMBuildFMul(b, dadx, x0_center, "dadx_x0");
385 LLVMValueRef dady_y0 = LLVMBuildFMul(b, dady, y0_center, "dady_y0");
386 LLVMValueRef attr_v0 = LLVMBuildFAdd(b, dadx_x0, dady_y0, "attr_v0");
387 LLVMValueRef attr_0 = LLVMBuildFSub(b, a0, attr_v0, "attr_0");
388
389 store_coef(gallivm, args, slot, attr_0, dadx, dady);
390 }
391
392
393 static void
394 emit_linear_coef( struct gallivm_state *gallivm,
395 struct lp_setup_args *args,
396 unsigned slot,
397 LLVMValueRef attribv[3])
398 {
399 /* nothing to do anymore */
400 emit_coef4(gallivm,
401 args, slot,
402 attribv[0],
403 attribv[1],
404 attribv[2]);
405 }
406
407
408 /**
409 * Compute a0, dadx and dady for a perspective-corrected interpolant,
410 * for a triangle.
411 * We basically multiply the vertex value by 1/w before computing
412 * the plane coefficients (a0, dadx, dady).
413 * Later, when we compute the value at a particular fragment position we'll
414 * divide the interpolated value by the interpolated W at that fragment.
415 */
416 static void
417 apply_perspective_corr( struct gallivm_state *gallivm,
418 struct lp_setup_args *args,
419 unsigned slot,
420 LLVMValueRef attribv[3])
421 {
422 LLVMBuilderRef b = gallivm->builder;
423
424 /* premultiply by 1/w (v[0][3] is always 1/w):
425 */
426 LLVMValueRef v0_oow = vec4f_from_scalar(gallivm, vert_attrib(gallivm, args->v0, 0, 3, ""), "v0_oow");
427 LLVMValueRef v1_oow = vec4f_from_scalar(gallivm, vert_attrib(gallivm, args->v1, 0, 3, ""), "v1_oow");
428 LLVMValueRef v2_oow = vec4f_from_scalar(gallivm, vert_attrib(gallivm, args->v2, 0, 3, ""), "v2_oow");
429
430 attribv[0] = LLVMBuildFMul(b, attribv[0], v0_oow, "v0_oow_v0a");
431 attribv[1] = LLVMBuildFMul(b, attribv[1], v1_oow, "v1_oow_v1a");
432 attribv[2] = LLVMBuildFMul(b, attribv[2], v2_oow, "v2_oow_v2a");
433 }
434
435
436 static void
437 emit_position_coef( struct gallivm_state *gallivm,
438 struct lp_setup_args *args,
439 int slot,
440 LLVMValueRef attribv[3])
441 {
442 emit_linear_coef(gallivm, args, slot, attribv);
443 }
444
445
446 /**
447 * Applys cylindrical wrapping to vertex attributes if enabled.
448 * Input coordinates must be in [0, 1] range, otherwise results are undefined.
449 *
450 * @param cyl_wrap TGSI_CYLINDRICAL_WRAP_x flags
451 */
452 static void
453 emit_apply_cyl_wrap(struct gallivm_state *gallivm,
454 struct lp_setup_args *args,
455 uint cyl_wrap,
456 LLVMValueRef attribv[3])
457
458 {
459 LLVMBuilderRef builder = gallivm->builder;
460 struct lp_type type = lp_float32_vec4_type();
461 LLVMTypeRef float_vec_type = lp_build_vec_type(gallivm, type);
462 LLVMValueRef pos_half;
463 LLVMValueRef neg_half;
464 LLVMValueRef cyl_mask;
465 LLVMValueRef offset;
466 LLVMValueRef delta;
467 LLVMValueRef one;
468
469 if (!cyl_wrap)
470 return;
471
472 /* Constants */
473 pos_half = lp_build_const_vec(gallivm, type, +0.5f);
474 neg_half = lp_build_const_vec(gallivm, type, -0.5f);
475 cyl_mask = lp_build_const_mask_aos(gallivm, type, cyl_wrap, 4);
476
477 one = lp_build_const_vec(gallivm, type, 1.0f);
478 one = LLVMBuildBitCast(builder, one, lp_build_int_vec_type(gallivm, type), "");
479 one = LLVMBuildAnd(builder, one, cyl_mask, "");
480
481 /* Edge v0 -> v1 */
482 delta = LLVMBuildFSub(builder, attribv[1], attribv[0], "");
483
484 offset = lp_build_compare(gallivm, type, PIPE_FUNC_GREATER, delta, pos_half);
485 offset = LLVMBuildAnd(builder, offset, one, "");
486 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
487 attribv[0] = LLVMBuildFAdd(builder, attribv[0], offset, "");
488
489 offset = lp_build_compare(gallivm, type, PIPE_FUNC_LESS, delta, neg_half);
490 offset = LLVMBuildAnd(builder, offset, one, "");
491 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
492 attribv[1] = LLVMBuildFAdd(builder, attribv[1], offset, "");
493
494 /* Edge v1 -> v2 */
495 delta = LLVMBuildFSub(builder, attribv[2], attribv[1], "");
496
497 offset = lp_build_compare(gallivm, type, PIPE_FUNC_GREATER, delta, pos_half);
498 offset = LLVMBuildAnd(builder, offset, one, "");
499 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
500 attribv[1] = LLVMBuildFAdd(builder, attribv[1], offset, "");
501
502 offset = lp_build_compare(gallivm, type, PIPE_FUNC_LESS, delta, neg_half);
503 offset = LLVMBuildAnd(builder, offset, one, "");
504 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
505 attribv[2] = LLVMBuildFAdd(builder, attribv[2], offset, "");
506
507 /* Edge v2 -> v0 */
508 delta = LLVMBuildFSub(builder, attribv[0], attribv[2], "");
509
510 offset = lp_build_compare(gallivm, type, PIPE_FUNC_GREATER, delta, pos_half);
511 offset = LLVMBuildAnd(builder, offset, one, "");
512 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
513 attribv[2] = LLVMBuildFAdd(builder, attribv[2], offset, "");
514
515 offset = lp_build_compare(gallivm, type, PIPE_FUNC_LESS, delta, neg_half);
516 offset = LLVMBuildAnd(builder, offset, one, "");
517 offset = LLVMBuildBitCast(builder, offset, float_vec_type, "");
518 attribv[0] = LLVMBuildFAdd(builder, attribv[0], offset, "");
519 }
520
521
522 /**
523 * Compute the inputs-> dadx, dady, a0 values.
524 */
525 static void
526 emit_tri_coef( struct gallivm_state *gallivm,
527 const struct lp_setup_variant_key *key,
528 struct lp_setup_args *args)
529 {
530 unsigned slot;
531
532 LLVMValueRef attribs[3];
533
534 /* setup interpolation for all the remaining attributes:
535 */
536 for (slot = 0; slot < key->num_inputs; slot++) {
537 switch (key->inputs[slot].interp) {
538 case LP_INTERP_CONSTANT:
539 load_attribute(gallivm, args, key, key->inputs[slot].src_index, attribs);
540 if (key->flatshade_first) {
541 emit_constant_coef4(gallivm, args, slot+1, attribs[0]);
542 }
543 else {
544 emit_constant_coef4(gallivm, args, slot+1, attribs[2]);
545 }
546 break;
547
548 case LP_INTERP_LINEAR:
549 load_attribute(gallivm, args, key, key->inputs[slot].src_index, attribs);
550 emit_apply_cyl_wrap(gallivm, args, key->inputs[slot].cyl_wrap, attribs);
551 emit_linear_coef(gallivm, args, slot+1, attribs);
552 break;
553
554 case LP_INTERP_PERSPECTIVE:
555 load_attribute(gallivm, args, key, key->inputs[slot].src_index, attribs);
556 emit_apply_cyl_wrap(gallivm, args, key->inputs[slot].cyl_wrap, attribs);
557 apply_perspective_corr(gallivm, args, slot+1, attribs);
558 emit_linear_coef(gallivm, args, slot+1, attribs);
559 break;
560
561 case LP_INTERP_POSITION:
562 /*
563 * The generated pixel interpolators will pick up the coeffs from
564 * slot 0.
565 */
566 break;
567
568 case LP_INTERP_FACING:
569 emit_facing_coef(gallivm, args, slot+1);
570 break;
571
572 default:
573 assert(0);
574 }
575 }
576 }
577
578
579 /* XXX: generic code:
580 */
581 static void
582 set_noalias(LLVMBuilderRef builder,
583 LLVMValueRef function,
584 const LLVMTypeRef *arg_types,
585 int nr_args)
586 {
587 int i;
588 for(i = 0; i < nr_args; ++i)
589 if(LLVMGetTypeKind(arg_types[i]) == LLVMPointerTypeKind)
590 LLVMAddAttribute(LLVMGetParam(function, i),
591 LLVMNoAliasAttribute);
592 }
593
594 static void
595 init_args(struct gallivm_state *gallivm,
596 const struct lp_setup_variant_key *key,
597 struct lp_setup_args *args)
598 {
599 LLVMBuilderRef b = gallivm->builder;
600 LLVMTypeRef shuf_type = LLVMInt32TypeInContext(gallivm->context);
601 LLVMValueRef onef = lp_build_const_float(gallivm, 1.0);
602 LLVMValueRef onei = lp_build_const_int32(gallivm, 1);
603 LLVMValueRef zeroi = lp_build_const_int32(gallivm, 0);
604 LLVMValueRef pixel_center, xy0_center, dxy01, dxy20, dyx20;
605 LLVMValueRef e, f, ef, ooa;
606 LLVMValueRef shuffles[4];
607 LLVMValueRef attr_pos[3];
608 struct lp_type typef4 = lp_type_float_vec(32, 128);
609
610 /* The internal position input is in slot zero:
611 */
612 load_attribute(gallivm, args, key, 0, attr_pos);
613
614 pixel_center = lp_build_const_vec(gallivm, typef4,
615 key->pixel_center_half ? 0.5 : 0.0);
616
617 /*
618 * xy are first two elems in v0a/v1a/v2a but just use vec4 arit
619 * also offset_tri uses actually xyz in them
620 */
621 xy0_center = LLVMBuildFSub(b, attr_pos[0], pixel_center, "xy0_center" );
622
623 dxy01 = LLVMBuildFSub(b, attr_pos[0], attr_pos[1], "dxy01");
624 dxy20 = LLVMBuildFSub(b, attr_pos[2], attr_pos[0], "dxy20");
625
626 shuffles[0] = onei;
627 shuffles[1] = zeroi;
628 shuffles[2] = LLVMGetUndef(shuf_type);
629 shuffles[3] = LLVMGetUndef(shuf_type);
630
631 dyx20 = LLVMBuildShuffleVector(b, dxy20, dxy20, LLVMConstVector(shuffles, 4), "");
632
633 ef = LLVMBuildFMul(b, dxy01, dyx20, "ef");
634 e = LLVMBuildExtractElement(b, ef, zeroi, "");
635 f = LLVMBuildExtractElement(b, ef, onei, "");
636
637 ooa = LLVMBuildFDiv(b, onef, LLVMBuildFSub(b, e, f, ""), "ooa");
638
639 ooa = vec4f_from_scalar(gallivm, ooa, "");
640
641 /* tri offset calc shares a lot of arithmetic, do it here */
642 if (key->scale != 0.0f || key->units != 0.0f) {
643 lp_do_offset_tri(gallivm, args, key, ooa, dxy01, dxy20, attr_pos);
644 }
645
646 dxy20 = LLVMBuildFMul(b, dxy20, ooa, "");
647 dxy01 = LLVMBuildFMul(b, dxy01, ooa, "");
648
649 args->dy20_ooa = lp_build_extract_broadcast(gallivm, typef4, typef4, dxy20, onei);
650 args->dy01_ooa = lp_build_extract_broadcast(gallivm, typef4, typef4, dxy01, onei);
651
652 args->dx20_ooa = lp_build_extract_broadcast(gallivm, typef4, typef4, dxy20, zeroi);
653 args->dx01_ooa = lp_build_extract_broadcast(gallivm, typef4, typef4, dxy01, zeroi);
654
655 args->x0_center = lp_build_extract_broadcast(gallivm, typef4, typef4, xy0_center, zeroi);
656 args->y0_center = lp_build_extract_broadcast(gallivm, typef4, typef4, xy0_center, onei);
657
658 /* might want to merge that with other coef emit in the future */
659 emit_position_coef(gallivm, args, 0, attr_pos);
660 }
661
662 /**
663 * Generate the runtime callable function for the coefficient calculation.
664 *
665 */
666 static struct lp_setup_variant *
667 generate_setup_variant(struct lp_setup_variant_key *key,
668 struct llvmpipe_context *lp)
669 {
670 struct lp_setup_variant *variant = NULL;
671 struct gallivm_state *gallivm;
672 struct lp_setup_args args;
673 char func_name[256];
674 LLVMTypeRef vec4f_type;
675 LLVMTypeRef func_type;
676 LLVMTypeRef arg_types[7];
677 LLVMBasicBlockRef block;
678 LLVMBuilderRef builder;
679 int64_t t0 = 0, t1;
680
681 if (0)
682 goto fail;
683
684 variant = CALLOC_STRUCT(lp_setup_variant);
685 if (variant == NULL)
686 goto fail;
687
688 variant->gallivm = gallivm = gallivm_create();
689 if (!variant->gallivm) {
690 goto fail;
691 }
692
693 builder = gallivm->builder;
694
695 if (LP_DEBUG & DEBUG_COUNTERS) {
696 t0 = os_time_get();
697 }
698
699 memcpy(&variant->key, key, key->size);
700 variant->list_item_global.base = variant;
701
702 util_snprintf(func_name, sizeof(func_name), "fs%u_setup%u",
703 0,
704 variant->no);
705
706 /* Currently always deal with full 4-wide vertex attributes from
707 * the vertices.
708 */
709
710 vec4f_type = LLVMVectorType(LLVMFloatTypeInContext(gallivm->context), 4);
711
712 arg_types[0] = LLVMPointerType(vec4f_type, 0); /* v0 */
713 arg_types[1] = LLVMPointerType(vec4f_type, 0); /* v1 */
714 arg_types[2] = LLVMPointerType(vec4f_type, 0); /* v2 */
715 arg_types[3] = LLVMInt32TypeInContext(gallivm->context); /* facing */
716 arg_types[4] = LLVMPointerType(vec4f_type, 0); /* a0, aligned */
717 arg_types[5] = LLVMPointerType(vec4f_type, 0); /* dadx, aligned */
718 arg_types[6] = LLVMPointerType(vec4f_type, 0); /* dady, aligned */
719
720 func_type = LLVMFunctionType(LLVMVoidTypeInContext(gallivm->context),
721 arg_types, Elements(arg_types), 0);
722
723 variant->function = LLVMAddFunction(gallivm->module, func_name, func_type);
724 if (!variant->function)
725 goto fail;
726
727 LLVMSetFunctionCallConv(variant->function, LLVMCCallConv);
728
729 args.v0 = LLVMGetParam(variant->function, 0);
730 args.v1 = LLVMGetParam(variant->function, 1);
731 args.v2 = LLVMGetParam(variant->function, 2);
732 args.facing = LLVMGetParam(variant->function, 3);
733 args.a0 = LLVMGetParam(variant->function, 4);
734 args.dadx = LLVMGetParam(variant->function, 5);
735 args.dady = LLVMGetParam(variant->function, 6);
736
737 lp_build_name(args.v0, "in_v0");
738 lp_build_name(args.v1, "in_v1");
739 lp_build_name(args.v2, "in_v2");
740 lp_build_name(args.facing, "in_facing");
741 lp_build_name(args.a0, "out_a0");
742 lp_build_name(args.dadx, "out_dadx");
743 lp_build_name(args.dady, "out_dady");
744
745 /*
746 * Function body
747 */
748 block = LLVMAppendBasicBlockInContext(gallivm->context,
749 variant->function, "entry");
750 LLVMPositionBuilderAtEnd(builder, block);
751
752 set_noalias(builder, variant->function, arg_types, Elements(arg_types));
753 init_args(gallivm, &variant->key, &args);
754 emit_tri_coef(gallivm, &variant->key, &args);
755
756 LLVMBuildRetVoid(builder);
757
758 gallivm_verify_function(gallivm, variant->function);
759
760 gallivm_compile_module(gallivm);
761
762 variant->jit_function = (lp_jit_setup_triangle)
763 gallivm_jit_function(gallivm, variant->function);
764 if (!variant->jit_function)
765 goto fail;
766
767 /*
768 * Update timing information:
769 */
770 if (LP_DEBUG & DEBUG_COUNTERS) {
771 t1 = os_time_get();
772 LP_COUNT_ADD(llvm_compile_time, t1 - t0);
773 LP_COUNT_ADD(nr_llvm_compiles, 1);
774 }
775
776 return variant;
777
778 fail:
779 if (variant) {
780 if (variant->function) {
781 gallivm_free_function(gallivm,
782 variant->function,
783 variant->jit_function);
784 }
785 if (variant->gallivm) {
786 gallivm_destroy(variant->gallivm);
787 }
788 FREE(variant);
789 }
790
791 return NULL;
792 }
793
794
795
796 static void
797 lp_make_setup_variant_key(struct llvmpipe_context *lp,
798 struct lp_setup_variant_key *key)
799 {
800 struct lp_fragment_shader *fs = lp->fs;
801 unsigned i;
802
803 assert(sizeof key->inputs[0] == sizeof(uint));
804
805 key->num_inputs = fs->info.base.num_inputs;
806 key->flatshade_first = lp->rasterizer->flatshade_first;
807 key->pixel_center_half = lp->rasterizer->gl_rasterization_rules;
808 key->twoside = lp->rasterizer->light_twoside;
809 key->size = Offset(struct lp_setup_variant_key,
810 inputs[key->num_inputs]);
811
812 key->color_slot = lp->color_slot [0];
813 key->bcolor_slot = lp->bcolor_slot[0];
814 key->spec_slot = lp->color_slot [1];
815 key->bspec_slot = lp->bcolor_slot[1];
816 assert(key->color_slot == lp->color_slot [0]);
817 assert(key->bcolor_slot == lp->bcolor_slot[0]);
818 assert(key->spec_slot == lp->color_slot [1]);
819 assert(key->bspec_slot == lp->bcolor_slot[1]);
820
821 key->units = (float) (lp->rasterizer->offset_units * lp->mrd);
822 key->scale = lp->rasterizer->offset_scale;
823 key->pad = 0;
824 memcpy(key->inputs, fs->inputs, key->num_inputs * sizeof key->inputs[0]);
825 for (i = 0; i < key->num_inputs; i++) {
826 if (key->inputs[i].interp == LP_INTERP_COLOR) {
827 if (lp->rasterizer->flatshade)
828 key->inputs[i].interp = LP_INTERP_CONSTANT;
829 else
830 key->inputs[i].interp = LP_INTERP_PERSPECTIVE;
831 }
832 }
833
834 }
835
836
837 static void
838 remove_setup_variant(struct llvmpipe_context *lp,
839 struct lp_setup_variant *variant)
840 {
841 if (gallivm_debug & GALLIVM_DEBUG_IR) {
842 debug_printf("llvmpipe: del setup_variant #%u total %u\n",
843 variant->no, lp->nr_setup_variants);
844 }
845
846 if (variant->function) {
847 gallivm_free_function(variant->gallivm,
848 variant->function,
849 variant->jit_function);
850 }
851
852 if (variant->gallivm) {
853 gallivm_destroy(variant->gallivm);
854 }
855
856 remove_from_list(&variant->list_item_global);
857 lp->nr_setup_variants--;
858 FREE(variant);
859 }
860
861
862
863 /* When the number of setup variants exceeds a threshold, cull a
864 * fraction (currently a quarter) of them.
865 */
866 static void
867 cull_setup_variants(struct llvmpipe_context *lp)
868 {
869 struct pipe_context *pipe = &lp->pipe;
870 int i;
871
872 /*
873 * XXX: we need to flush the context until we have some sort of reference
874 * counting in fragment shaders as they may still be binned
875 * Flushing alone might not be sufficient we need to wait on it too.
876 */
877 llvmpipe_finish(pipe, __FUNCTION__);
878
879 for (i = 0; i < LP_MAX_SETUP_VARIANTS / 4; i++) {
880 struct lp_setup_variant_list_item *item;
881 if (is_empty_list(&lp->setup_variants_list)) {
882 break;
883 }
884 item = last_elem(&lp->setup_variants_list);
885 assert(item);
886 assert(item->base);
887 remove_setup_variant(lp, item->base);
888 }
889 }
890
891
892 /**
893 * Update fragment/vertex shader linkage state. This is called just
894 * prior to drawing something when some fragment-related state has
895 * changed.
896 */
897 void
898 llvmpipe_update_setup(struct llvmpipe_context *lp)
899 {
900 struct lp_setup_variant_key *key = &lp->setup_variant.key;
901 struct lp_setup_variant *variant = NULL;
902 struct lp_setup_variant_list_item *li;
903
904 lp_make_setup_variant_key(lp, key);
905
906 foreach(li, &lp->setup_variants_list) {
907 if(li->base->key.size == key->size &&
908 memcmp(&li->base->key, key, key->size) == 0) {
909 variant = li->base;
910 break;
911 }
912 }
913
914 if (variant) {
915 move_to_head(&lp->setup_variants_list, &variant->list_item_global);
916 }
917 else {
918 if (lp->nr_setup_variants >= LP_MAX_SETUP_VARIANTS) {
919 cull_setup_variants(lp);
920 }
921
922 variant = generate_setup_variant(key, lp);
923 if (variant) {
924 insert_at_head(&lp->setup_variants_list, &variant->list_item_global);
925 lp->nr_setup_variants++;
926 llvmpipe_variant_count++;
927 }
928 }
929
930 lp_setup_set_setup_variant(lp->setup,
931 variant);
932 }
933
934 void
935 lp_delete_setup_variants(struct llvmpipe_context *lp)
936 {
937 struct lp_setup_variant_list_item *li;
938 li = first_elem(&lp->setup_variants_list);
939 while(!at_end(&lp->setup_variants_list, li)) {
940 struct lp_setup_variant_list_item *next = next_elem(li);
941 remove_setup_variant(lp, li->base);
942 li = next;
943 }
944 }
945
946 void
947 lp_dump_setup_coef( const struct lp_setup_variant_key *key,
948 const float (*sa0)[4],
949 const float (*sdadx)[4],
950 const float (*sdady)[4])
951 {
952 int i, slot;
953
954 for (i = 0; i < TGSI_NUM_CHANNELS; i++) {
955 float a0 = sa0 [0][i];
956 float dadx = sdadx[0][i];
957 float dady = sdady[0][i];
958
959 debug_printf("POS.%c: a0 = %f, dadx = %f, dady = %f\n",
960 "xyzw"[i],
961 a0, dadx, dady);
962 }
963
964 for (slot = 0; slot < key->num_inputs; slot++) {
965 unsigned usage_mask = key->inputs[slot].usage_mask;
966 for (i = 0; i < TGSI_NUM_CHANNELS; i++) {
967 if (usage_mask & (1 << i)) {
968 float a0 = sa0 [1 + slot][i];
969 float dadx = sdadx[1 + slot][i];
970 float dady = sdady[1 + slot][i];
971
972 debug_printf("IN[%u].%c: a0 = %f, dadx = %f, dady = %f\n",
973 slot,
974 "xyzw"[i],
975 a0, dadx, dady);
976 }
977 }
978 }
979 }