1 /*
2  * Direct3D asm shader parser
3  *
4  * Copyright 2008 Stefan Dösinger
5  * Copyright 2009 Matteo Bruni
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  *
21  */
22 
23 #include "wine/debug.h"
24 
25 #include "d3dcompiler_private.h"
26 
27 WINE_DEFAULT_DEBUG_CHANNEL(asmshader);
28 WINE_DECLARE_DEBUG_CHANNEL(parsed_shader);
29 
30 
31 /* How to map vs 1.0 and 2.0 varyings to 3.0 ones
32  * oTx is mapped to ox, which happens to be an
33  * identical mapping since BWRITERSPR_TEXCRDOUT == BWRITERSPR_OUTPUT
34  * oPos, oFog and point size are mapped to general output regs as well.
35  * the vs 1.x and 2.x parser functions add varying declarations
36  * to the shader, and the 1.x and 2.x output functions check those varyings
37  */
38 #define OT0_REG         0
39 #define OT1_REG         1
40 #define OT2_REG         2
41 #define OT3_REG         3
42 #define OT4_REG         4
43 #define OT5_REG         5
44 #define OT6_REG         6
45 #define OT7_REG         7
46 #define OPOS_REG        8
47 #define OFOG_REG        9
48 #define OFOG_WRITEMASK  BWRITERSP_WRITEMASK_0
49 #define OPTS_REG        9
50 #define OPTS_WRITEMASK  BWRITERSP_WRITEMASK_1
51 #define OD0_REG         10
52 #define OD1_REG         11
53 
54 /* Input color registers 0-1 are identically mapped */
55 #define C0_VARYING      0
56 #define C1_VARYING      1
57 #define T0_VARYING      2
58 #define T1_VARYING      3
59 #define T2_VARYING      4
60 #define T3_VARYING      5
61 #define T4_VARYING      6
62 #define T5_VARYING      7
63 #define T6_VARYING      8
64 #define T7_VARYING      9
65 
66 /****************************************************************
67  * Common(non-version specific) shader parser control code      *
68  ****************************************************************/
69 
asmparser_end(struct asm_parser * This)70 static void asmparser_end(struct asm_parser *This) {
71     TRACE("Finalizing shader\n");
72 }
73 
asmparser_constF(struct asm_parser * This,DWORD reg,float x,float y,float z,float w)74 static void asmparser_constF(struct asm_parser *This, DWORD reg, float x, float y, float z, float w) {
75     if(!This->shader) return;
76     TRACE("Adding float constant %u at pos %u\n", reg, This->shader->num_cf);
77     TRACE_(parsed_shader)("def c%u, %f, %f, %f, %f\n", reg, x, y, z, w);
78     if(!add_constF(This->shader, reg, x, y, z, w)) {
79         ERR("Out of memory\n");
80         set_parse_status(&This->status, PARSE_ERR);
81     }
82 }
83 
asmparser_constB(struct asm_parser * This,DWORD reg,BOOL x)84 static void asmparser_constB(struct asm_parser *This, DWORD reg, BOOL x) {
85     if(!This->shader) return;
86     TRACE("Adding boolean constant %u at pos %u\n", reg, This->shader->num_cb);
87     TRACE_(parsed_shader)("def b%u, %s\n", reg, x ? "true" : "false");
88     if(!add_constB(This->shader, reg, x)) {
89         ERR("Out of memory\n");
90         set_parse_status(&This->status, PARSE_ERR);
91     }
92 }
93 
asmparser_constI(struct asm_parser * This,DWORD reg,INT x,INT y,INT z,INT w)94 static void asmparser_constI(struct asm_parser *This, DWORD reg, INT x, INT y, INT z, INT w) {
95     if(!This->shader) return;
96     TRACE("Adding integer constant %u at pos %u\n", reg, This->shader->num_ci);
97     TRACE_(parsed_shader)("def i%u, %d, %d, %d, %d\n", reg, x, y, z, w);
98     if(!add_constI(This->shader, reg, x, y, z, w)) {
99         ERR("Out of memory\n");
100         set_parse_status(&This->status, PARSE_ERR);
101     }
102 }
103 
asmparser_dcl_output(struct asm_parser * This,DWORD usage,DWORD num,const struct shader_reg * reg)104 static void asmparser_dcl_output(struct asm_parser *This, DWORD usage, DWORD num,
105                                  const struct shader_reg *reg) {
106     if(!This->shader) return;
107     if(This->shader->type == ST_PIXEL) {
108         asmparser_message(This, "Line %u: Output register declared in a pixel shader\n", This->line_no);
109         set_parse_status(&This->status, PARSE_ERR);
110     }
111     if(!record_declaration(This->shader, usage, num, 0, TRUE, reg->regnum, reg->u.writemask, FALSE)) {
112         ERR("Out of memory\n");
113         set_parse_status(&This->status, PARSE_ERR);
114     }
115 }
116 
asmparser_dcl_output_unsupported(struct asm_parser * This,DWORD usage,DWORD num,const struct shader_reg * reg)117 static void asmparser_dcl_output_unsupported(struct asm_parser *This, DWORD usage, DWORD num,
118                                              const struct shader_reg *reg) {
119     asmparser_message(This, "Line %u: Output declaration unsupported in this shader version\n", This->line_no);
120     set_parse_status(&This->status, PARSE_ERR);
121 }
122 
asmparser_dcl_input(struct asm_parser * This,DWORD usage,DWORD num,DWORD mod,const struct shader_reg * reg)123 static void asmparser_dcl_input(struct asm_parser *This, DWORD usage, DWORD num,
124                                 DWORD mod, const struct shader_reg *reg) {
125     struct instruction instr;
126 
127     if(!This->shader) return;
128     if(mod != 0 &&
129        (This->shader->version != BWRITERPS_VERSION(3, 0) ||
130         (mod != BWRITERSPDM_MSAMPCENTROID &&
131          mod != BWRITERSPDM_PARTIALPRECISION))) {
132         asmparser_message(This, "Line %u: Unsupported modifier in dcl instruction\n", This->line_no);
133         set_parse_status(&This->status, PARSE_ERR);
134         return;
135     }
136 
137     /* Check register type and modifiers */
138     instr.dstmod = mod;
139     instr.shift = 0;
140     This->funcs->dstreg(This, &instr, reg);
141 
142     if(!record_declaration(This->shader, usage, num, mod, FALSE, reg->regnum, reg->u.writemask, FALSE)) {
143         ERR("Out of memory\n");
144         set_parse_status(&This->status, PARSE_ERR);
145     }
146 }
147 
asmparser_dcl_input_ps_2(struct asm_parser * This,DWORD usage,DWORD num,DWORD mod,const struct shader_reg * reg)148 static void asmparser_dcl_input_ps_2(struct asm_parser *This, DWORD usage, DWORD num,
149                                      DWORD mod, const struct shader_reg *reg) {
150     struct instruction instr;
151 
152     if(!This->shader) return;
153     instr.dstmod = mod;
154     instr.shift = 0;
155     This->funcs->dstreg(This, &instr, reg);
156     if(!record_declaration(This->shader, usage, num, mod, FALSE, instr.dst.regnum, instr.dst.u.writemask, FALSE)) {
157         ERR("Out of memory\n");
158         set_parse_status(&This->status, PARSE_ERR);
159     }
160 }
161 
asmparser_dcl_input_unsupported(struct asm_parser * This,DWORD usage,DWORD num,DWORD mod,const struct shader_reg * reg)162 static void asmparser_dcl_input_unsupported(struct asm_parser *This,
163         DWORD usage, DWORD num, DWORD mod, const struct shader_reg *reg)
164 {
165     asmparser_message(This, "Line %u: Input declaration unsupported in this shader version\n", This->line_no);
166     set_parse_status(&This->status, PARSE_ERR);
167 }
168 
asmparser_dcl_sampler(struct asm_parser * This,DWORD samptype,DWORD mod,DWORD regnum,unsigned int line_no)169 static void asmparser_dcl_sampler(struct asm_parser *This, DWORD samptype,
170                                   DWORD mod, DWORD regnum,
171                                   unsigned int line_no) {
172     if(!This->shader) return;
173     if(mod != 0 &&
174        (This->shader->version != BWRITERPS_VERSION(3, 0) ||
175         (mod != BWRITERSPDM_MSAMPCENTROID &&
176          mod != BWRITERSPDM_PARTIALPRECISION))) {
177         asmparser_message(This, "Line %u: Unsupported modifier in dcl instruction\n", This->line_no);
178         set_parse_status(&This->status, PARSE_ERR);
179         return;
180     }
181     if(!record_sampler(This->shader, samptype, mod, regnum)) {
182         ERR("Out of memory\n");
183         set_parse_status(&This->status, PARSE_ERR);
184     }
185 }
186 
asmparser_dcl_sampler_unsupported(struct asm_parser * This,DWORD samptype,DWORD mod,DWORD regnum,unsigned int line_no)187 static void asmparser_dcl_sampler_unsupported(struct asm_parser *This,
188         DWORD samptype, DWORD mod, DWORD regnum, unsigned int line_no)
189 {
190     asmparser_message(This, "Line %u: Sampler declaration unsupported in this shader version\n", This->line_no);
191     set_parse_status(&This->status, PARSE_ERR);
192 }
193 
asmparser_sincos(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct src_regs * srcs)194 static void asmparser_sincos(struct asm_parser *This, DWORD mod, DWORD shift,
195                              const struct shader_reg *dst,
196                              const struct src_regs *srcs) {
197     struct instruction *instr;
198 
199     if(!srcs || srcs->count != 3) {
200         asmparser_message(This, "Line %u: sincos (vs 2) has an incorrect number of source registers\n", This->line_no);
201         set_parse_status(&This->status, PARSE_ERR);
202         return;
203     }
204 
205     instr = alloc_instr(3);
206     if(!instr) {
207         ERR("Error allocating memory for the instruction\n");
208         set_parse_status(&This->status, PARSE_ERR);
209         return;
210     }
211 
212     instr->opcode = BWRITERSIO_SINCOS;
213     instr->dstmod = mod;
214     instr->shift = shift;
215     instr->comptype = 0;
216 
217     This->funcs->dstreg(This, instr, dst);
218     This->funcs->srcreg(This, instr, 0, &srcs->reg[0]);
219     This->funcs->srcreg(This, instr, 1, &srcs->reg[1]);
220     This->funcs->srcreg(This, instr, 2, &srcs->reg[2]);
221 
222     if(!add_instruction(This->shader, instr)) {
223         ERR("Out of memory\n");
224         set_parse_status(&This->status, PARSE_ERR);
225     }
226 }
227 
map_oldps_register(const struct shader_reg * reg,BOOL tex_varying)228 static struct shader_reg map_oldps_register(const struct shader_reg *reg, BOOL tex_varying) {
229     struct shader_reg ret;
230     switch(reg->type) {
231         case BWRITERSPR_TEXTURE:
232             if(tex_varying) {
233                 ret = *reg;
234                 ret.type = BWRITERSPR_INPUT;
235                 switch(reg->regnum) {
236                     case 0:     ret.regnum = T0_VARYING; break;
237                     case 1:     ret.regnum = T1_VARYING; break;
238                     case 2:     ret.regnum = T2_VARYING; break;
239                     case 3:     ret.regnum = T3_VARYING; break;
240                     case 4:     ret.regnum = T4_VARYING; break;
241                     case 5:     ret.regnum = T5_VARYING; break;
242                     case 6:     ret.regnum = T6_VARYING; break;
243                     case 7:     ret.regnum = T7_VARYING; break;
244                     default:
245                         FIXME("Unexpected TEXTURE register t%u\n", reg->regnum);
246                         return *reg;
247                 }
248                 return ret;
249             } else {
250                 ret = *reg;
251                 ret.type = BWRITERSPR_TEMP;
252                 switch(reg->regnum) {
253                     case 0:     ret.regnum = T0_REG; break;
254                     case 1:     ret.regnum = T1_REG; break;
255                     case 2:     ret.regnum = T2_REG; break;
256                     case 3:     ret.regnum = T3_REG; break;
257                     default:
258                         FIXME("Unexpected TEXTURE register t%u\n", reg->regnum);
259                         return *reg;
260                 }
261                 return ret;
262             }
263 
264         /* case BWRITERSPR_INPUT - Identical mapping of 1.x/2.0 color varyings
265            to 3.0 ones */
266 
267         default: return *reg;
268     }
269 }
270 
asmparser_texcoord(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct src_regs * srcs)271 static void asmparser_texcoord(struct asm_parser *This, DWORD mod, DWORD shift,
272                                const struct shader_reg *dst,
273                                const struct src_regs *srcs) {
274     struct instruction *instr;
275 
276     if(srcs) {
277         asmparser_message(This, "Line %u: Source registers in texcoord instruction\n", This->line_no);
278         set_parse_status(&This->status, PARSE_ERR);
279         return;
280     }
281 
282     instr = alloc_instr(1);
283     if(!instr) {
284         ERR("Error allocating memory for the instruction\n");
285         set_parse_status(&This->status, PARSE_ERR);
286         return;
287     }
288 
289     /* texcoord copies the texture coord data into a temporary register-like
290      * readable form. In newer shader models this equals a MOV from v0 to r0,
291      * record it as this.
292      */
293     instr->opcode = BWRITERSIO_MOV;
294     instr->dstmod = mod | BWRITERSPDM_SATURATE; /* texcoord clamps to [0;1] */
295     instr->shift = shift;
296     instr->comptype = 0;
297 
298     This->funcs->dstreg(This, instr, dst);
299     /* The src reg needs special care */
300     instr->src[0] = map_oldps_register(dst, TRUE);
301 
302     if(!add_instruction(This->shader, instr)) {
303         ERR("Out of memory\n");
304         set_parse_status(&This->status, PARSE_ERR);
305     }
306 }
307 
asmparser_texcrd(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct src_regs * srcs)308 static void asmparser_texcrd(struct asm_parser *This, DWORD mod, DWORD shift,
309                              const struct shader_reg *dst,
310                              const struct src_regs *srcs) {
311     struct instruction *instr;
312 
313     if(!srcs || srcs->count != 1) {
314         asmparser_message(This, "Line %u: Wrong number of source registers in texcrd instruction\n", This->line_no);
315         set_parse_status(&This->status, PARSE_ERR);
316         return;
317     }
318 
319     instr = alloc_instr(1);
320     if(!instr) {
321         ERR("Error allocating memory for the instruction\n");
322         set_parse_status(&This->status, PARSE_ERR);
323         return;
324     }
325 
326     /* The job of texcrd is done by mov in later shader versions */
327     instr->opcode = BWRITERSIO_MOV;
328     instr->dstmod = mod;
329     instr->shift = shift;
330     instr->comptype = 0;
331 
332     This->funcs->dstreg(This, instr, dst);
333     This->funcs->srcreg(This, instr, 0, &srcs->reg[0]);
334 
335     if(!add_instruction(This->shader, instr)) {
336         ERR("Out of memory\n");
337         set_parse_status(&This->status, PARSE_ERR);
338     }
339 }
340 
asmparser_texkill(struct asm_parser * This,const struct shader_reg * dst)341 static void asmparser_texkill(struct asm_parser *This,
342                               const struct shader_reg *dst) {
343     struct instruction *instr = alloc_instr(0);
344 
345     if(!instr) {
346         ERR("Error allocating memory for the instruction\n");
347         set_parse_status(&This->status, PARSE_ERR);
348         return;
349     }
350 
351     instr->opcode = BWRITERSIO_TEXKILL;
352     instr->dstmod = 0;
353     instr->shift = 0;
354     instr->comptype = 0;
355 
356     /* Do not run the dst register through the normal
357      * register conversion. If used with ps_1_0 to ps_1_3
358      * the texture coordinate from that register is used,
359      * not the temporary register value. In ps_1_4 and
360      * ps_2_0 t0 is always a varying and temporaries can
361      * be used with texkill.
362      */
363     instr->dst = map_oldps_register(dst, TRUE);
364     instr->has_dst = TRUE;
365 
366     if(!add_instruction(This->shader, instr)) {
367         ERR("Out of memory\n");
368         set_parse_status(&This->status, PARSE_ERR);
369     }
370 }
371 
asmparser_texhelper(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct shader_reg * src0)372 static void asmparser_texhelper(struct asm_parser *This, DWORD mod, DWORD shift,
373                                 const struct shader_reg *dst,
374                                 const struct shader_reg *src0) {
375     struct instruction *instr = alloc_instr(2);
376 
377     if(!instr) {
378         ERR("Error allocating memory for the instruction\n");
379         set_parse_status(&This->status, PARSE_ERR);
380         return;
381     }
382 
383     instr->opcode = BWRITERSIO_TEX;
384     instr->dstmod = mod;
385     instr->shift = shift;
386     instr->comptype = 0;
387     /* The dest register can be mapped normally to a temporary register */
388     This->funcs->dstreg(This, instr, dst);
389     /* Use the src passed as parameter by the specific instruction handler */
390     instr->src[0] = *src0;
391 
392     /* The 2nd source register is the sampler register with the
393      * destination's regnum
394      */
395     ZeroMemory(&instr->src[1], sizeof(instr->src[1]));
396     instr->src[1].type = BWRITERSPR_SAMPLER;
397     instr->src[1].regnum = dst->regnum;
398     instr->src[1].u.swizzle = BWRITERVS_NOSWIZZLE;
399     instr->src[1].srcmod = BWRITERSPSM_NONE;
400     instr->src[1].rel_reg = NULL;
401 
402     if(!add_instruction(This->shader, instr)) {
403         ERR("Out of memory\n");
404         set_parse_status(&This->status, PARSE_ERR);
405     }
406 }
407 
asmparser_tex(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst)408 static void asmparser_tex(struct asm_parser *This, DWORD mod, DWORD shift,
409                           const struct shader_reg *dst) {
410     struct shader_reg src;
411 
412     /* The first source register is the varying containing the coordinate */
413     src = map_oldps_register(dst, TRUE);
414     asmparser_texhelper(This, mod, shift, dst, &src);
415 }
416 
asmparser_texld14(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct src_regs * srcs)417 static void asmparser_texld14(struct asm_parser *This, DWORD mod, DWORD shift,
418                               const struct shader_reg *dst,
419                               const struct src_regs *srcs) {
420     struct instruction *instr;
421 
422     if(!srcs || srcs->count != 1) {
423         asmparser_message(This, "Line %u: texld (PS 1.4) has a wrong number of source registers\n", This->line_no);
424         set_parse_status(&This->status, PARSE_ERR);
425         return;
426     }
427 
428     instr = alloc_instr(2);
429     if(!instr) {
430         ERR("Error allocating memory for the instruction\n");
431         set_parse_status(&This->status, PARSE_ERR);
432         return;
433     }
434 
435     /* This code is recording a texld instruction, not tex. However,
436      * texld borrows the opcode of tex
437      */
438     instr->opcode = BWRITERSIO_TEX;
439     instr->dstmod = mod;
440     instr->shift = shift;
441     instr->comptype = 0;
442 
443     This->funcs->dstreg(This, instr, dst);
444     This->funcs->srcreg(This, instr, 0, &srcs->reg[0]);
445 
446     /* The 2nd source register is the sampler register with the
447      * destination's regnum
448      */
449     ZeroMemory(&instr->src[1], sizeof(instr->src[1]));
450     instr->src[1].type = BWRITERSPR_SAMPLER;
451     instr->src[1].regnum = dst->regnum;
452     instr->src[1].u.swizzle = BWRITERVS_NOSWIZZLE;
453     instr->src[1].srcmod = BWRITERSPSM_NONE;
454     instr->src[1].rel_reg = NULL;
455 
456     if(!add_instruction(This->shader, instr)) {
457         ERR("Out of memory\n");
458         set_parse_status(&This->status, PARSE_ERR);
459     }
460 }
461 
asmparser_texreg2ar(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct shader_reg * src0)462 static void asmparser_texreg2ar(struct asm_parser *This, DWORD mod, DWORD shift,
463                                 const struct shader_reg *dst,
464                                 const struct shader_reg *src0) {
465     struct shader_reg src;
466 
467     src = map_oldps_register(src0, FALSE);
468     /* Supply the correct swizzle */
469     src.u.swizzle = BWRITERVS_X_W | BWRITERVS_Y_X | BWRITERVS_Z_X | BWRITERVS_W_X;
470     asmparser_texhelper(This, mod, shift, dst, &src);
471 }
472 
asmparser_texreg2gb(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct shader_reg * src0)473 static void asmparser_texreg2gb(struct asm_parser *This, DWORD mod, DWORD shift,
474                                 const struct shader_reg *dst,
475                                 const struct shader_reg *src0) {
476     struct shader_reg src;
477 
478     src = map_oldps_register(src0, FALSE);
479     /* Supply the correct swizzle */
480     src.u.swizzle = BWRITERVS_X_Y | BWRITERVS_Y_Z | BWRITERVS_Z_Z | BWRITERVS_W_Z;
481     asmparser_texhelper(This, mod, shift, dst, &src);
482 }
483 
asmparser_texreg2rgb(struct asm_parser * This,DWORD mod,DWORD shift,const struct shader_reg * dst,const struct shader_reg * src0)484 static void asmparser_texreg2rgb(struct asm_parser *This, DWORD mod, DWORD shift,
485                                  const struct shader_reg *dst,
486                                  const struct shader_reg *src0) {
487     struct shader_reg src;
488 
489     src = map_oldps_register(src0, FALSE);
490     /* Supply the correct swizzle */
491     src.u.swizzle = BWRITERVS_X_X | BWRITERVS_Y_Y | BWRITERVS_Z_Z | BWRITERVS_W_Z;
492     asmparser_texhelper(This, mod, shift, dst, &src);
493 }
494 
495 /* Complex pixel shader 1.3 instructions like texm3x3tex are tricky - the
496  * bytecode writer works instruction by instruction, so we can't properly
497  * convert these from/to equivalent ps_3_0 instructions. Then simply keep using
498  * the ps_1_3 opcodes and just adapt the registers in the common fashion (i.e.
499  * go through asmparser_instr).
500  */
501 
asmparser_instr(struct asm_parser * This,DWORD opcode,DWORD mod,DWORD shift,enum bwriter_comparison_type comp,const struct shader_reg * dst,const struct src_regs * srcs,int expectednsrcs)502 static void asmparser_instr(struct asm_parser *This, DWORD opcode, DWORD mod, DWORD shift,
503         enum bwriter_comparison_type comp, const struct shader_reg *dst,
504         const struct src_regs *srcs, int expectednsrcs)
505 {
506     struct instruction *instr;
507     unsigned int i;
508     BOOL firstreg = TRUE;
509     unsigned int src_count = srcs ? srcs->count : 0;
510 
511     if(!This->shader) return;
512 
513     TRACE_(parsed_shader)("%s%s%s%s ", debug_print_opcode(opcode),
514                           debug_print_dstmod(mod),
515                           debug_print_shift(shift),
516                           debug_print_comp(comp));
517     if(dst) {
518         TRACE_(parsed_shader)("%s", debug_print_dstreg(dst));
519         firstreg = FALSE;
520     }
521     for(i = 0; i < src_count; i++) {
522         if(!firstreg) TRACE_(parsed_shader)(", ");
523         else firstreg = FALSE;
524         TRACE_(parsed_shader)("%s", debug_print_srcreg(&srcs->reg[i]));
525     }
526     TRACE_(parsed_shader)("\n");
527 
528  /* Check for instructions with different syntaxes in different shader versions */
529     switch(opcode) {
530         case BWRITERSIO_SINCOS:
531             /* The syntax changes between vs 2 and the other shader versions */
532             if(This->shader->version == BWRITERVS_VERSION(2, 0) ||
533                This->shader->version == BWRITERVS_VERSION(2, 1)) {
534                 asmparser_sincos(This, mod, shift, dst, srcs);
535                 return;
536             }
537             /* Use the default handling */
538             break;
539         case BWRITERSIO_TEXCOORD:
540             /* texcoord/texcrd are two instructions present only in PS <= 1.3 and PS 1.4 respectively */
541             if(This->shader->version == BWRITERPS_VERSION(1, 4))
542                 asmparser_texcrd(This, mod, shift, dst, srcs);
543             else asmparser_texcoord(This, mod, shift, dst, srcs);
544             return;
545         case BWRITERSIO_TEX:
546             /* this encodes both the tex PS 1.x instruction and the
547                texld 1.4/2.0+ instruction */
548             if(This->shader->version == BWRITERPS_VERSION(1, 0) ||
549                This->shader->version == BWRITERPS_VERSION(1, 1) ||
550                This->shader->version == BWRITERPS_VERSION(1, 2) ||
551                This->shader->version == BWRITERPS_VERSION(1, 3)) {
552                 asmparser_tex(This, mod, shift, dst);
553                 return;
554             }
555             else if(This->shader->version == BWRITERPS_VERSION(1, 4)) {
556                 asmparser_texld14(This, mod, shift, dst, srcs);
557                 return;
558             }
559             /* else fallback to the standard behavior */
560             break;
561     }
562 
563     if(src_count != expectednsrcs) {
564         asmparser_message(This, "Line %u: Wrong number of source registers\n", This->line_no);
565         set_parse_status(&This->status, PARSE_ERR);
566         return;
567     }
568 
569     /* Handle PS 1.x instructions, "regularizing" them */
570     switch(opcode) {
571         case BWRITERSIO_TEXKILL:
572             asmparser_texkill(This, dst);
573             return;
574         case BWRITERSIO_TEXREG2AR:
575             asmparser_texreg2ar(This, mod, shift, dst, &srcs->reg[0]);
576             return;
577         case BWRITERSIO_TEXREG2GB:
578             asmparser_texreg2gb(This, mod, shift, dst, &srcs->reg[0]);
579             return;
580         case BWRITERSIO_TEXREG2RGB:
581             asmparser_texreg2rgb(This, mod, shift, dst, &srcs->reg[0]);
582             return;
583     }
584 
585     instr = alloc_instr(src_count);
586     if(!instr) {
587         ERR("Error allocating memory for the instruction\n");
588         set_parse_status(&This->status, PARSE_ERR);
589         return;
590     }
591 
592     instr->opcode = opcode;
593     instr->dstmod = mod;
594     instr->shift = shift;
595     instr->comptype = comp;
596     if(dst) This->funcs->dstreg(This, instr, dst);
597     for(i = 0; i < src_count; i++) {
598         This->funcs->srcreg(This, instr, i, &srcs->reg[i]);
599     }
600 
601     if(!add_instruction(This->shader, instr)) {
602         ERR("Out of memory\n");
603         set_parse_status(&This->status, PARSE_ERR);
604     }
605 }
606 
map_oldvs_register(const struct shader_reg * reg)607 static struct shader_reg map_oldvs_register(const struct shader_reg *reg) {
608     struct shader_reg ret;
609     switch(reg->type) {
610         case BWRITERSPR_RASTOUT:
611             ret = *reg;
612             ret.type = BWRITERSPR_OUTPUT;
613             switch(reg->regnum) {
614                 case BWRITERSRO_POSITION:
615                     ret.regnum = OPOS_REG;
616                     break;
617                 case BWRITERSRO_FOG:
618                     ret.regnum = OFOG_REG;
619                     ret.u.writemask = OFOG_WRITEMASK;
620                     break;
621                 case BWRITERSRO_POINT_SIZE:
622                     ret.regnum = OPTS_REG;
623                     ret.u.writemask = OPTS_WRITEMASK;
624                     break;
625                 default:
626                     FIXME("Unhandled RASTOUT register %u\n", reg->regnum);
627                     return *reg;
628             }
629             return ret;
630 
631         case BWRITERSPR_TEXCRDOUT:
632             ret = *reg;
633             ret.type = BWRITERSPR_OUTPUT;
634             switch(reg->regnum) {
635                 case 0: ret.regnum = OT0_REG; break;
636                 case 1: ret.regnum = OT1_REG; break;
637                 case 2: ret.regnum = OT2_REG; break;
638                 case 3: ret.regnum = OT3_REG; break;
639                 case 4: ret.regnum = OT4_REG; break;
640                 case 5: ret.regnum = OT5_REG; break;
641                 case 6: ret.regnum = OT6_REG; break;
642                 case 7: ret.regnum = OT7_REG; break;
643                 default:
644                     FIXME("Unhandled TEXCRDOUT regnum %u\n", reg->regnum);
645                     return *reg;
646             }
647             return ret;
648 
649         case BWRITERSPR_ATTROUT:
650             ret = *reg;
651             ret.type = BWRITERSPR_OUTPUT;
652             switch(reg->regnum) {
653                 case 0: ret.regnum = OD0_REG; break;
654                 case 1: ret.regnum = OD1_REG; break;
655                 default:
656                     FIXME("Unhandled ATTROUT regnum %u\n", reg->regnum);
657                     return *reg;
658             }
659             return ret;
660 
661         default: return *reg;
662     }
663 }
664 
665 /* Checks for unsupported source modifiers in VS (all versions) or
666    PS 2.0 and newer */
check_legacy_srcmod(struct asm_parser * This,DWORD srcmod)667 static void check_legacy_srcmod(struct asm_parser *This, DWORD srcmod) {
668     if(srcmod == BWRITERSPSM_BIAS || srcmod == BWRITERSPSM_BIASNEG ||
669        srcmod == BWRITERSPSM_SIGN || srcmod == BWRITERSPSM_SIGNNEG ||
670        srcmod == BWRITERSPSM_COMP || srcmod == BWRITERSPSM_X2 ||
671        srcmod == BWRITERSPSM_X2NEG || srcmod == BWRITERSPSM_DZ ||
672        srcmod == BWRITERSPSM_DW) {
673         asmparser_message(This, "Line %u: Source modifier %s not supported in this shader version\n",
674                           This->line_no,
675                           debug_print_srcmod(srcmod));
676         set_parse_status(&This->status, PARSE_ERR);
677     }
678 }
679 
check_abs_srcmod(struct asm_parser * This,DWORD srcmod)680 static void check_abs_srcmod(struct asm_parser *This, DWORD srcmod) {
681     if(srcmod == BWRITERSPSM_ABS || srcmod == BWRITERSPSM_ABSNEG) {
682         asmparser_message(This, "Line %u: Source modifier %s not supported in this shader version\n",
683                           This->line_no,
684                           debug_print_srcmod(srcmod));
685         set_parse_status(&This->status, PARSE_ERR);
686     }
687 }
688 
check_loop_swizzle(struct asm_parser * This,const struct shader_reg * src)689 static void check_loop_swizzle(struct asm_parser *This,
690                                const struct shader_reg *src) {
691     if((src->type == BWRITERSPR_LOOP && src->u.swizzle != BWRITERVS_NOSWIZZLE) ||
692        (src->rel_reg && src->rel_reg->type == BWRITERSPR_LOOP &&
693         src->rel_reg->u.swizzle != BWRITERVS_NOSWIZZLE)) {
694         asmparser_message(This, "Line %u: Swizzle not allowed on aL register\n", This->line_no);
695         set_parse_status(&This->status, PARSE_ERR);
696     }
697 }
698 
check_shift_dstmod(struct asm_parser * This,DWORD shift)699 static void check_shift_dstmod(struct asm_parser *This, DWORD shift) {
700     if(shift != 0) {
701         asmparser_message(This, "Line %u: Shift modifiers not supported in this shader version\n",
702                           This->line_no);
703         set_parse_status(&This->status, PARSE_ERR);
704     }
705 }
706 
check_ps_dstmod(struct asm_parser * This,DWORD dstmod)707 static void check_ps_dstmod(struct asm_parser *This, DWORD dstmod) {
708     if(dstmod == BWRITERSPDM_PARTIALPRECISION ||
709        dstmod == BWRITERSPDM_MSAMPCENTROID) {
710         asmparser_message(This, "Line %u: Instruction modifier %s not supported in this shader version\n",
711                           This->line_no,
712                           debug_print_dstmod(dstmod));
713         set_parse_status(&This->status, PARSE_ERR);
714     }
715 }
716 
717 struct allowed_reg_type {
718     DWORD type;
719     DWORD count;
720     BOOL reladdr;
721 };
722 
check_reg_type(const struct shader_reg * reg,const struct allowed_reg_type * allowed)723 static BOOL check_reg_type(const struct shader_reg *reg,
724                            const struct allowed_reg_type *allowed) {
725     unsigned int i = 0;
726 
727     while(allowed[i].type != ~0U) {
728         if(reg->type == allowed[i].type) {
729             if(reg->rel_reg) {
730                 if(allowed[i].reladdr)
731                     return TRUE; /* The relative addressing register
732                                     can have a negative value, we
733                                     can't check the register index */
734                 return FALSE;
735             }
736             if(reg->regnum < allowed[i].count) return TRUE;
737             return FALSE;
738         }
739         i++;
740     }
741     return FALSE;
742 }
743 
744 /* Native assembler doesn't do separate checks for src and dst registers */
745 static const struct allowed_reg_type vs_1_reg_allowed[] = {
746     { BWRITERSPR_TEMP,         12,  FALSE },
747     { BWRITERSPR_INPUT,        16,  FALSE },
748     { BWRITERSPR_CONST,       ~0U,   TRUE },
749     { BWRITERSPR_ADDR,          1,  FALSE },
750     { BWRITERSPR_RASTOUT,       3,  FALSE }, /* oPos, oFog and oPts */
751     { BWRITERSPR_ATTROUT,       2,  FALSE },
752     { BWRITERSPR_TEXCRDOUT,     8,  FALSE },
753     { ~0U, 0 } /* End tag */
754 };
755 
756 /* struct instruction *asmparser_srcreg
757  *
758  * Records a source register in the instruction and does shader version
759  * specific checks and modifications on it
760  *
761  * Parameters:
762  *  This: Shader parser instance
763  *  instr: instruction to store the register in
764  *  num: Number of source register
765  *  src: Pointer to source the register structure. The caller can free
766  *  it afterwards
767  */
asmparser_srcreg_vs_1(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)768 static void asmparser_srcreg_vs_1(struct asm_parser *This,
769                                   struct instruction *instr, int num,
770                                   const struct shader_reg *src) {
771     struct shader_reg reg;
772 
773     if(!check_reg_type(src, vs_1_reg_allowed)) {
774         asmparser_message(This, "Line %u: Source register %s not supported in VS 1\n",
775                           This->line_no,
776                           debug_print_srcreg(src));
777         set_parse_status(&This->status, PARSE_ERR);
778     }
779     check_legacy_srcmod(This, src->srcmod);
780     check_abs_srcmod(This, src->srcmod);
781     reg = map_oldvs_register(src);
782     instr->src[num] = reg;
783 }
784 
785 static const struct allowed_reg_type vs_2_reg_allowed[] = {
786     { BWRITERSPR_TEMP,      12,  FALSE },
787     { BWRITERSPR_INPUT,     16,  FALSE },
788     { BWRITERSPR_CONST,    ~0U,   TRUE },
789     { BWRITERSPR_ADDR,       1,  FALSE },
790     { BWRITERSPR_CONSTBOOL, 16,  FALSE },
791     { BWRITERSPR_CONSTINT,  16,  FALSE },
792     { BWRITERSPR_LOOP,       1,  FALSE },
793     { BWRITERSPR_LABEL,   2048,  FALSE },
794     { BWRITERSPR_PREDICATE,  1,  FALSE },
795     { BWRITERSPR_RASTOUT,    3,  FALSE }, /* oPos, oFog and oPts */
796     { BWRITERSPR_ATTROUT,    2,  FALSE },
797     { BWRITERSPR_TEXCRDOUT,  8,  FALSE },
798     { ~0U, 0 } /* End tag */
799 };
800 
asmparser_srcreg_vs_2(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)801 static void asmparser_srcreg_vs_2(struct asm_parser *This,
802                                   struct instruction *instr, int num,
803                                   const struct shader_reg *src) {
804     struct shader_reg reg;
805 
806     if(!check_reg_type(src, vs_2_reg_allowed)) {
807         asmparser_message(This, "Line %u: Source register %s not supported in VS 2\n",
808                           This->line_no,
809                           debug_print_srcreg(src));
810         set_parse_status(&This->status, PARSE_ERR);
811     }
812     check_loop_swizzle(This, src);
813     check_legacy_srcmod(This, src->srcmod);
814     check_abs_srcmod(This, src->srcmod);
815     reg = map_oldvs_register(src);
816     instr->src[num] = reg;
817 }
818 
819 static const struct allowed_reg_type vs_3_reg_allowed[] = {
820     { BWRITERSPR_TEMP,         32,  FALSE },
821     { BWRITERSPR_INPUT,        16,   TRUE },
822     { BWRITERSPR_CONST,       ~0U,   TRUE },
823     { BWRITERSPR_ADDR,          1,  FALSE },
824     { BWRITERSPR_CONSTBOOL,    16,  FALSE },
825     { BWRITERSPR_CONSTINT,     16,  FALSE },
826     { BWRITERSPR_LOOP,          1,  FALSE },
827     { BWRITERSPR_LABEL,      2048,  FALSE },
828     { BWRITERSPR_PREDICATE,     1,  FALSE },
829     { BWRITERSPR_SAMPLER,       4,  FALSE },
830     { BWRITERSPR_OUTPUT,       12,   TRUE },
831     { ~0U, 0 } /* End tag */
832 };
833 
asmparser_srcreg_vs_3(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)834 static void asmparser_srcreg_vs_3(struct asm_parser *This,
835                                   struct instruction *instr, int num,
836                                   const struct shader_reg *src) {
837     if(!check_reg_type(src, vs_3_reg_allowed)) {
838         asmparser_message(This, "Line %u: Source register %s not supported in VS 3.0\n",
839                           This->line_no,
840                           debug_print_srcreg(src));
841         set_parse_status(&This->status, PARSE_ERR);
842     }
843     check_loop_swizzle(This, src);
844     check_legacy_srcmod(This, src->srcmod);
845     instr->src[num] = *src;
846 }
847 
848 static const struct allowed_reg_type ps_1_0123_reg_allowed[] = {
849     { BWRITERSPR_CONST,     8,  FALSE },
850     { BWRITERSPR_TEMP,      2,  FALSE },
851     { BWRITERSPR_TEXTURE,   4,  FALSE },
852     { BWRITERSPR_INPUT,     2,  FALSE },
853     { ~0U, 0 } /* End tag */
854 };
855 
asmparser_srcreg_ps_1_0123(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)856 static void asmparser_srcreg_ps_1_0123(struct asm_parser *This,
857                                        struct instruction *instr, int num,
858                                        const struct shader_reg *src) {
859     struct shader_reg reg;
860 
861     if(!check_reg_type(src, ps_1_0123_reg_allowed)) {
862         asmparser_message(This, "Line %u: Source register %s not supported in <== PS 1.3\n",
863                           This->line_no,
864                           debug_print_srcreg(src));
865         set_parse_status(&This->status, PARSE_ERR);
866     }
867     check_abs_srcmod(This, src->srcmod);
868     reg = map_oldps_register(src, FALSE);
869     instr->src[num] = reg;
870 }
871 
872 static const struct allowed_reg_type ps_1_4_reg_allowed[] = {
873     { BWRITERSPR_CONST,     8,  FALSE },
874     { BWRITERSPR_TEMP,      6,  FALSE },
875     { BWRITERSPR_TEXTURE,   6,  FALSE },
876     { BWRITERSPR_INPUT,     2,  FALSE },
877     { ~0U, 0 } /* End tag */
878 };
879 
asmparser_srcreg_ps_1_4(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)880 static void asmparser_srcreg_ps_1_4(struct asm_parser *This,
881                                     struct instruction *instr, int num,
882                                     const struct shader_reg *src) {
883     struct shader_reg reg;
884 
885     if(!check_reg_type(src, ps_1_4_reg_allowed)) {
886         asmparser_message(This, "Line %u: Source register %s not supported in PS 1.4\n",
887                           This->line_no,
888                           debug_print_srcreg(src));
889         set_parse_status(&This->status, PARSE_ERR);
890     }
891     check_abs_srcmod(This, src->srcmod);
892     reg = map_oldps_register(src, TRUE);
893     instr->src[num] = reg;
894 }
895 
896 static const struct allowed_reg_type ps_2_0_reg_allowed[] = {
897     { BWRITERSPR_INPUT,         2,  FALSE },
898     { BWRITERSPR_TEMP,         32,  FALSE },
899     { BWRITERSPR_CONST,        32,  FALSE },
900     { BWRITERSPR_CONSTINT,     16,  FALSE },
901     { BWRITERSPR_CONSTBOOL,    16,  FALSE },
902     { BWRITERSPR_SAMPLER,      16,  FALSE },
903     { BWRITERSPR_TEXTURE,       8,  FALSE },
904     { BWRITERSPR_COLOROUT,      4,  FALSE },
905     { BWRITERSPR_DEPTHOUT,      1,  FALSE },
906     { ~0U, 0 } /* End tag */
907 };
908 
asmparser_srcreg_ps_2(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)909 static void asmparser_srcreg_ps_2(struct asm_parser *This,
910                                   struct instruction *instr, int num,
911                                   const struct shader_reg *src) {
912     struct shader_reg reg;
913 
914     if(!check_reg_type(src, ps_2_0_reg_allowed)) {
915         asmparser_message(This, "Line %u: Source register %s not supported in PS 2.0\n",
916                           This->line_no,
917                           debug_print_srcreg(src));
918         set_parse_status(&This->status, PARSE_ERR);
919     }
920     check_legacy_srcmod(This, src->srcmod);
921     check_abs_srcmod(This, src->srcmod);
922     reg = map_oldps_register(src, TRUE);
923     instr->src[num] = reg;
924 }
925 
926 static const struct allowed_reg_type ps_2_x_reg_allowed[] = {
927     { BWRITERSPR_INPUT,         2,  FALSE },
928     { BWRITERSPR_TEMP,         32,  FALSE },
929     { BWRITERSPR_CONST,        32,  FALSE },
930     { BWRITERSPR_CONSTINT,     16,  FALSE },
931     { BWRITERSPR_CONSTBOOL,    16,  FALSE },
932     { BWRITERSPR_PREDICATE,     1,  FALSE },
933     { BWRITERSPR_SAMPLER,      16,  FALSE },
934     { BWRITERSPR_TEXTURE,       8,  FALSE },
935     { BWRITERSPR_LABEL,      2048,  FALSE },
936     { BWRITERSPR_COLOROUT,      4,  FALSE },
937     { BWRITERSPR_DEPTHOUT,      1,  FALSE },
938     { ~0U, 0 } /* End tag */
939 };
940 
asmparser_srcreg_ps_2_x(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)941 static void asmparser_srcreg_ps_2_x(struct asm_parser *This,
942                                     struct instruction *instr, int num,
943                                     const struct shader_reg *src) {
944     struct shader_reg reg;
945 
946     if(!check_reg_type(src, ps_2_x_reg_allowed)) {
947         asmparser_message(This, "Line %u: Source register %s not supported in PS 2.x\n",
948                           This->line_no,
949                           debug_print_srcreg(src));
950         set_parse_status(&This->status, PARSE_ERR);
951     }
952     check_legacy_srcmod(This, src->srcmod);
953     check_abs_srcmod(This, src->srcmod);
954     reg = map_oldps_register(src, TRUE);
955     instr->src[num] = reg;
956 }
957 
958 static const struct allowed_reg_type ps_3_reg_allowed[] = {
959     { BWRITERSPR_INPUT,        10,   TRUE },
960     { BWRITERSPR_TEMP,         32,  FALSE },
961     { BWRITERSPR_CONST,       224,  FALSE },
962     { BWRITERSPR_CONSTINT,     16,  FALSE },
963     { BWRITERSPR_CONSTBOOL,    16,  FALSE },
964     { BWRITERSPR_PREDICATE,     1,  FALSE },
965     { BWRITERSPR_SAMPLER,      16,  FALSE },
966     { BWRITERSPR_MISCTYPE,      2,  FALSE }, /* vPos and vFace */
967     { BWRITERSPR_LOOP,          1,  FALSE },
968     { BWRITERSPR_LABEL,      2048,  FALSE },
969     { BWRITERSPR_COLOROUT,      4,  FALSE },
970     { BWRITERSPR_DEPTHOUT,      1,  FALSE },
971     { ~0U, 0 } /* End tag */
972 };
973 
asmparser_srcreg_ps_3(struct asm_parser * This,struct instruction * instr,int num,const struct shader_reg * src)974 static void asmparser_srcreg_ps_3(struct asm_parser *This,
975                                   struct instruction *instr, int num,
976                                   const struct shader_reg *src) {
977     if(!check_reg_type(src, ps_3_reg_allowed)) {
978         asmparser_message(This, "Line %u: Source register %s not supported in PS 3.0\n",
979                           This->line_no,
980                           debug_print_srcreg(src));
981         set_parse_status(&This->status, PARSE_ERR);
982     }
983     check_loop_swizzle(This, src);
984     check_legacy_srcmod(This, src->srcmod);
985     instr->src[num] = *src;
986 }
987 
asmparser_dstreg_vs_1(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)988 static void asmparser_dstreg_vs_1(struct asm_parser *This,
989                                   struct instruction *instr,
990                                   const struct shader_reg *dst) {
991     struct shader_reg reg;
992 
993     if(!check_reg_type(dst, vs_1_reg_allowed)) {
994         asmparser_message(This, "Line %u: Destination register %s not supported in VS 1\n",
995                           This->line_no,
996                           debug_print_dstreg(dst));
997         set_parse_status(&This->status, PARSE_ERR);
998     }
999     check_ps_dstmod(This, instr->dstmod);
1000     check_shift_dstmod(This, instr->shift);
1001     reg = map_oldvs_register(dst);
1002     instr->dst = reg;
1003     instr->has_dst = TRUE;
1004 }
1005 
asmparser_dstreg_vs_2(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1006 static void asmparser_dstreg_vs_2(struct asm_parser *This,
1007                                   struct instruction *instr,
1008                                   const struct shader_reg *dst) {
1009     struct shader_reg reg;
1010 
1011     if(!check_reg_type(dst, vs_2_reg_allowed)) {
1012         asmparser_message(This, "Line %u: Destination register %s not supported in VS 2.0\n",
1013                           This->line_no,
1014                           debug_print_dstreg(dst));
1015         set_parse_status(&This->status, PARSE_ERR);
1016     }
1017     check_ps_dstmod(This, instr->dstmod);
1018     check_shift_dstmod(This, instr->shift);
1019     reg = map_oldvs_register(dst);
1020     instr->dst = reg;
1021     instr->has_dst = TRUE;
1022 }
1023 
asmparser_dstreg_vs_3(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1024 static void asmparser_dstreg_vs_3(struct asm_parser *This,
1025                                   struct instruction *instr,
1026                                   const struct shader_reg *dst) {
1027     if(!check_reg_type(dst, vs_3_reg_allowed)) {
1028         asmparser_message(This, "Line %u: Destination register %s not supported in VS 3.0\n",
1029                           This->line_no,
1030                           debug_print_dstreg(dst));
1031         set_parse_status(&This->status, PARSE_ERR);
1032     }
1033     check_ps_dstmod(This, instr->dstmod);
1034     check_shift_dstmod(This, instr->shift);
1035     instr->dst = *dst;
1036     instr->has_dst = TRUE;
1037 }
1038 
asmparser_dstreg_ps_1_0123(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1039 static void asmparser_dstreg_ps_1_0123(struct asm_parser *This,
1040                                        struct instruction *instr,
1041                                        const struct shader_reg *dst) {
1042     struct shader_reg reg;
1043 
1044     if(!check_reg_type(dst, ps_1_0123_reg_allowed)) {
1045         asmparser_message(This, "Line %u: Destination register %s not supported in PS 1\n",
1046                           This->line_no,
1047                           debug_print_dstreg(dst));
1048         set_parse_status(&This->status, PARSE_ERR);
1049     }
1050     reg = map_oldps_register(dst, FALSE);
1051     instr->dst = reg;
1052     instr->has_dst = TRUE;
1053 }
1054 
asmparser_dstreg_ps_1_4(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1055 static void asmparser_dstreg_ps_1_4(struct asm_parser *This,
1056                                     struct instruction *instr,
1057                                     const struct shader_reg *dst) {
1058     struct shader_reg reg;
1059 
1060     if(!check_reg_type(dst, ps_1_4_reg_allowed)) {
1061         asmparser_message(This, "Line %u: Destination register %s not supported in PS 1\n",
1062                           This->line_no,
1063                           debug_print_dstreg(dst));
1064         set_parse_status(&This->status, PARSE_ERR);
1065     }
1066     reg = map_oldps_register(dst, TRUE);
1067     instr->dst = reg;
1068     instr->has_dst = TRUE;
1069 }
1070 
asmparser_dstreg_ps_2(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1071 static void asmparser_dstreg_ps_2(struct asm_parser *This,
1072                                   struct instruction *instr,
1073                                   const struct shader_reg *dst) {
1074     struct shader_reg reg;
1075 
1076     if(!check_reg_type(dst, ps_2_0_reg_allowed)) {
1077         asmparser_message(This, "Line %u: Destination register %s not supported in PS 2.0\n",
1078                           This->line_no,
1079                           debug_print_dstreg(dst));
1080         set_parse_status(&This->status, PARSE_ERR);
1081     }
1082     check_shift_dstmod(This, instr->shift);
1083     reg = map_oldps_register(dst, TRUE);
1084     instr->dst = reg;
1085     instr->has_dst = TRUE;
1086 }
1087 
asmparser_dstreg_ps_2_x(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1088 static void asmparser_dstreg_ps_2_x(struct asm_parser *This,
1089                                     struct instruction *instr,
1090                                     const struct shader_reg *dst) {
1091     struct shader_reg reg;
1092 
1093     if(!check_reg_type(dst, ps_2_x_reg_allowed)) {
1094         asmparser_message(This, "Line %u: Destination register %s not supported in PS 2.x\n",
1095                           This->line_no,
1096                           debug_print_dstreg(dst));
1097         set_parse_status(&This->status, PARSE_ERR);
1098     }
1099     check_shift_dstmod(This, instr->shift);
1100     reg = map_oldps_register(dst, TRUE);
1101     instr->dst = reg;
1102     instr->has_dst = TRUE;
1103 }
1104 
asmparser_dstreg_ps_3(struct asm_parser * This,struct instruction * instr,const struct shader_reg * dst)1105 static void asmparser_dstreg_ps_3(struct asm_parser *This,
1106                                   struct instruction *instr,
1107                                   const struct shader_reg *dst) {
1108     if(!check_reg_type(dst, ps_3_reg_allowed)) {
1109         asmparser_message(This, "Line %u: Destination register %s not supported in PS 3.0\n",
1110                           This->line_no,
1111                           debug_print_dstreg(dst));
1112         set_parse_status(&This->status, PARSE_ERR);
1113     }
1114     check_shift_dstmod(This, instr->shift);
1115     instr->dst = *dst;
1116     instr->has_dst = TRUE;
1117 }
1118 
asmparser_predicate_supported(struct asm_parser * This,const struct shader_reg * predicate)1119 static void asmparser_predicate_supported(struct asm_parser *This,
1120                                           const struct shader_reg *predicate) {
1121     /* this sets the predicate of the last instruction added to the shader */
1122     if(!This->shader) return;
1123     if(This->shader->num_instrs == 0) ERR("Predicate without an instruction\n");
1124     This->shader->instr[This->shader->num_instrs - 1]->has_predicate = TRUE;
1125     This->shader->instr[This->shader->num_instrs - 1]->predicate = *predicate;
1126 }
1127 
asmparser_predicate_unsupported(struct asm_parser * This,const struct shader_reg * predicate)1128 static void asmparser_predicate_unsupported(struct asm_parser *This,
1129                                             const struct shader_reg *predicate) {
1130     asmparser_message(This, "Line %u: Predicate not supported in < VS 2.0 or PS 2.x\n", This->line_no);
1131     set_parse_status(&This->status, PARSE_ERR);
1132 }
1133 
asmparser_coissue_supported(struct asm_parser * This)1134 static void asmparser_coissue_supported(struct asm_parser *This) {
1135     /* this sets the coissue flag of the last instruction added to the shader */
1136     if(!This->shader) return;
1137     if(This->shader->num_instrs == 0){
1138         asmparser_message(This, "Line %u: Coissue flag on the first shader instruction\n", This->line_no);
1139         set_parse_status(&This->status, PARSE_ERR);
1140     }
1141     This->shader->instr[This->shader->num_instrs-1]->coissue = TRUE;
1142 }
1143 
asmparser_coissue_unsupported(struct asm_parser * This)1144 static void asmparser_coissue_unsupported(struct asm_parser *This) {
1145     asmparser_message(This, "Line %u: Coissue is only supported in pixel shaders versions <= 1.4\n", This->line_no);
1146     set_parse_status(&This->status, PARSE_ERR);
1147 }
1148 
1149 static const struct asmparser_backend parser_vs_1 = {
1150     asmparser_constF,
1151     asmparser_constI,
1152     asmparser_constB,
1153 
1154     asmparser_dstreg_vs_1,
1155     asmparser_srcreg_vs_1,
1156 
1157     asmparser_predicate_unsupported,
1158     asmparser_coissue_unsupported,
1159 
1160     asmparser_dcl_output_unsupported,
1161     asmparser_dcl_input,
1162     asmparser_dcl_sampler_unsupported,
1163 
1164     asmparser_end,
1165 
1166     asmparser_instr,
1167 };
1168 
1169 static const struct asmparser_backend parser_vs_2 = {
1170     asmparser_constF,
1171     asmparser_constI,
1172     asmparser_constB,
1173 
1174     asmparser_dstreg_vs_2,
1175     asmparser_srcreg_vs_2,
1176 
1177     asmparser_predicate_supported,
1178     asmparser_coissue_unsupported,
1179 
1180     asmparser_dcl_output_unsupported,
1181     asmparser_dcl_input,
1182     asmparser_dcl_sampler_unsupported,
1183 
1184     asmparser_end,
1185 
1186     asmparser_instr,
1187 };
1188 
1189 static const struct asmparser_backend parser_vs_3 = {
1190     asmparser_constF,
1191     asmparser_constI,
1192     asmparser_constB,
1193 
1194     asmparser_dstreg_vs_3,
1195     asmparser_srcreg_vs_3,
1196 
1197     asmparser_predicate_supported,
1198     asmparser_coissue_unsupported,
1199 
1200     asmparser_dcl_output,
1201     asmparser_dcl_input,
1202     asmparser_dcl_sampler,
1203 
1204     asmparser_end,
1205 
1206     asmparser_instr,
1207 };
1208 
1209 static const struct asmparser_backend parser_ps_1_0123 = {
1210     asmparser_constF,
1211     asmparser_constI,
1212     asmparser_constB,
1213 
1214     asmparser_dstreg_ps_1_0123,
1215     asmparser_srcreg_ps_1_0123,
1216 
1217     asmparser_predicate_unsupported,
1218     asmparser_coissue_supported,
1219 
1220     asmparser_dcl_output_unsupported,
1221     asmparser_dcl_input_unsupported,
1222     asmparser_dcl_sampler_unsupported,
1223 
1224     asmparser_end,
1225 
1226     asmparser_instr,
1227 };
1228 
1229 static const struct asmparser_backend parser_ps_1_4 = {
1230     asmparser_constF,
1231     asmparser_constI,
1232     asmparser_constB,
1233 
1234     asmparser_dstreg_ps_1_4,
1235     asmparser_srcreg_ps_1_4,
1236 
1237     asmparser_predicate_unsupported,
1238     asmparser_coissue_supported,
1239 
1240     asmparser_dcl_output_unsupported,
1241     asmparser_dcl_input_unsupported,
1242     asmparser_dcl_sampler_unsupported,
1243 
1244     asmparser_end,
1245 
1246     asmparser_instr,
1247 };
1248 
1249 static const struct asmparser_backend parser_ps_2 = {
1250     asmparser_constF,
1251     asmparser_constI,
1252     asmparser_constB,
1253 
1254     asmparser_dstreg_ps_2,
1255     asmparser_srcreg_ps_2,
1256 
1257     asmparser_predicate_unsupported,
1258     asmparser_coissue_unsupported,
1259 
1260     asmparser_dcl_output_unsupported,
1261     asmparser_dcl_input_ps_2,
1262     asmparser_dcl_sampler,
1263 
1264     asmparser_end,
1265 
1266     asmparser_instr,
1267 };
1268 
1269 static const struct asmparser_backend parser_ps_2_x = {
1270     asmparser_constF,
1271     asmparser_constI,
1272     asmparser_constB,
1273 
1274     asmparser_dstreg_ps_2_x,
1275     asmparser_srcreg_ps_2_x,
1276 
1277     asmparser_predicate_supported,
1278     asmparser_coissue_unsupported,
1279 
1280     asmparser_dcl_output_unsupported,
1281     asmparser_dcl_input_ps_2,
1282     asmparser_dcl_sampler,
1283 
1284     asmparser_end,
1285 
1286     asmparser_instr,
1287 };
1288 
1289 static const struct asmparser_backend parser_ps_3 = {
1290     asmparser_constF,
1291     asmparser_constI,
1292     asmparser_constB,
1293 
1294     asmparser_dstreg_ps_3,
1295     asmparser_srcreg_ps_3,
1296 
1297     asmparser_predicate_supported,
1298     asmparser_coissue_unsupported,
1299 
1300     asmparser_dcl_output_unsupported,
1301     asmparser_dcl_input,
1302     asmparser_dcl_sampler,
1303 
1304     asmparser_end,
1305 
1306     asmparser_instr,
1307 };
1308 
gen_oldvs_output(struct bwriter_shader * shader)1309 static void gen_oldvs_output(struct bwriter_shader *shader) {
1310     record_declaration(shader, BWRITERDECLUSAGE_POSITION, 0, 0, TRUE, OPOS_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1311     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 0, 0, TRUE, OT0_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1312     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 1, 0, TRUE, OT1_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1313     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 2, 0, TRUE, OT2_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1314     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 3, 0, TRUE, OT3_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1315     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 4, 0, TRUE, OT4_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1316     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 5, 0, TRUE, OT5_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1317     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 6, 0, TRUE, OT6_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1318     record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 7, 0, TRUE, OT7_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1319     record_declaration(shader, BWRITERDECLUSAGE_FOG, 0, 0, TRUE, OFOG_REG, OFOG_WRITEMASK, TRUE);
1320     record_declaration(shader, BWRITERDECLUSAGE_PSIZE, 0, 0, TRUE, OPTS_REG, OPTS_WRITEMASK, TRUE);
1321     record_declaration(shader, BWRITERDECLUSAGE_COLOR, 0, 0, TRUE, OD0_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1322     record_declaration(shader, BWRITERDECLUSAGE_COLOR, 1, 0, TRUE, OD1_REG, BWRITERSP_WRITEMASK_ALL, TRUE);
1323 }
1324 
gen_oldps_input(struct bwriter_shader * shader,DWORD texcoords)1325 static void gen_oldps_input(struct bwriter_shader *shader, DWORD texcoords) {
1326     switch(texcoords) {
1327         case 8: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 7, 0, FALSE, T7_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1328             /* fall through */
1329         case 7: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 6, 0, FALSE, T6_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1330             /* fall through */
1331         case 6: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 5, 0, FALSE, T5_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1332             /* fall through */
1333         case 5: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 4, 0, FALSE, T4_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1334             /* fall through */
1335         case 4: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 3, 0, FALSE, T3_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1336             /* fall through */
1337         case 3: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 2, 0, FALSE, T2_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1338             /* fall through */
1339         case 2: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 1, 0, FALSE, T1_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1340             /* fall through */
1341         case 1: record_declaration(shader, BWRITERDECLUSAGE_TEXCOORD, 0, 0, FALSE, T0_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1342     };
1343     record_declaration(shader, BWRITERDECLUSAGE_COLOR, 0, 0, FALSE, C0_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1344     record_declaration(shader, BWRITERDECLUSAGE_COLOR, 1, 0, FALSE, C1_VARYING, BWRITERSP_WRITEMASK_ALL, TRUE);
1345 }
1346 
create_vs10_parser(struct asm_parser * ret)1347 void create_vs10_parser(struct asm_parser *ret) {
1348     TRACE_(parsed_shader)("vs_1_0\n");
1349 
1350     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1351     if(!ret->shader) {
1352         ERR("Failed to allocate memory for the shader\n");
1353         set_parse_status(&ret->status, PARSE_ERR);
1354         return;
1355     }
1356 
1357     ret->shader->type = ST_VERTEX;
1358     ret->shader->version = BWRITERVS_VERSION(1, 0);
1359     ret->funcs = &parser_vs_1;
1360     gen_oldvs_output(ret->shader);
1361 }
1362 
create_vs11_parser(struct asm_parser * ret)1363 void create_vs11_parser(struct asm_parser *ret) {
1364     TRACE_(parsed_shader)("vs_1_1\n");
1365 
1366     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1367     if(!ret->shader) {
1368         ERR("Failed to allocate memory for the shader\n");
1369         set_parse_status(&ret->status, PARSE_ERR);
1370         return;
1371     }
1372 
1373     ret->shader->type = ST_VERTEX;
1374     ret->shader->version = BWRITERVS_VERSION(1, 1);
1375     ret->funcs = &parser_vs_1;
1376     gen_oldvs_output(ret->shader);
1377 }
1378 
create_vs20_parser(struct asm_parser * ret)1379 void create_vs20_parser(struct asm_parser *ret) {
1380     TRACE_(parsed_shader)("vs_2_0\n");
1381 
1382     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1383     if(!ret->shader) {
1384         ERR("Failed to allocate memory for the shader\n");
1385         set_parse_status(&ret->status, PARSE_ERR);
1386         return;
1387     }
1388 
1389     ret->shader->type = ST_VERTEX;
1390     ret->shader->version = BWRITERVS_VERSION(2, 0);
1391     ret->funcs = &parser_vs_2;
1392     gen_oldvs_output(ret->shader);
1393 }
1394 
create_vs2x_parser(struct asm_parser * ret)1395 void create_vs2x_parser(struct asm_parser *ret) {
1396     TRACE_(parsed_shader)("vs_2_x\n");
1397 
1398     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1399     if(!ret->shader) {
1400         ERR("Failed to allocate memory for the shader\n");
1401         set_parse_status(&ret->status, PARSE_ERR);
1402         return;
1403     }
1404 
1405     ret->shader->type = ST_VERTEX;
1406     ret->shader->version = BWRITERVS_VERSION(2, 1);
1407     ret->funcs = &parser_vs_2;
1408     gen_oldvs_output(ret->shader);
1409 }
1410 
create_vs30_parser(struct asm_parser * ret)1411 void create_vs30_parser(struct asm_parser *ret) {
1412     TRACE_(parsed_shader)("vs_3_0\n");
1413 
1414     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1415     if(!ret->shader) {
1416         ERR("Failed to allocate memory for the shader\n");
1417         set_parse_status(&ret->status, PARSE_ERR);
1418         return;
1419     }
1420 
1421     ret->shader->type = ST_VERTEX;
1422     ret->shader->version = BWRITERVS_VERSION(3, 0);
1423     ret->funcs = &parser_vs_3;
1424 }
1425 
create_ps10_parser(struct asm_parser * ret)1426 void create_ps10_parser(struct asm_parser *ret) {
1427     TRACE_(parsed_shader)("ps_1_0\n");
1428 
1429     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1430     if(!ret->shader) {
1431         ERR("Failed to allocate memory for the shader\n");
1432         set_parse_status(&ret->status, PARSE_ERR);
1433         return;
1434     }
1435 
1436     ret->shader->type = ST_PIXEL;
1437     ret->shader->version = BWRITERPS_VERSION(1, 0);
1438     ret->funcs = &parser_ps_1_0123;
1439     gen_oldps_input(ret->shader, 4);
1440 }
1441 
create_ps11_parser(struct asm_parser * ret)1442 void create_ps11_parser(struct asm_parser *ret) {
1443     TRACE_(parsed_shader)("ps_1_1\n");
1444 
1445     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1446     if(!ret->shader) {
1447         ERR("Failed to allocate memory for the shader\n");
1448         set_parse_status(&ret->status, PARSE_ERR);
1449         return;
1450     }
1451 
1452     ret->shader->type = ST_PIXEL;
1453     ret->shader->version = BWRITERPS_VERSION(1, 1);
1454     ret->funcs = &parser_ps_1_0123;
1455     gen_oldps_input(ret->shader, 4);
1456 }
1457 
create_ps12_parser(struct asm_parser * ret)1458 void create_ps12_parser(struct asm_parser *ret) {
1459     TRACE_(parsed_shader)("ps_1_2\n");
1460 
1461     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1462     if(!ret->shader) {
1463         ERR("Failed to allocate memory for the shader\n");
1464         set_parse_status(&ret->status, PARSE_ERR);
1465         return;
1466     }
1467 
1468     ret->shader->type = ST_PIXEL;
1469     ret->shader->version = BWRITERPS_VERSION(1, 2);
1470     ret->funcs = &parser_ps_1_0123;
1471     gen_oldps_input(ret->shader, 4);
1472 }
1473 
create_ps13_parser(struct asm_parser * ret)1474 void create_ps13_parser(struct asm_parser *ret) {
1475     TRACE_(parsed_shader)("ps_1_3\n");
1476 
1477     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1478     if(!ret->shader) {
1479         ERR("Failed to allocate memory for the shader\n");
1480         set_parse_status(&ret->status, PARSE_ERR);
1481         return;
1482     }
1483 
1484     ret->shader->type = ST_PIXEL;
1485     ret->shader->version = BWRITERPS_VERSION(1, 3);
1486     ret->funcs = &parser_ps_1_0123;
1487     gen_oldps_input(ret->shader, 4);
1488 }
1489 
create_ps14_parser(struct asm_parser * ret)1490 void create_ps14_parser(struct asm_parser *ret) {
1491     TRACE_(parsed_shader)("ps_1_4\n");
1492 
1493     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1494     if(!ret->shader) {
1495         ERR("Failed to allocate memory for the shader\n");
1496         set_parse_status(&ret->status, PARSE_ERR);
1497         return;
1498     }
1499 
1500     ret->shader->type = ST_PIXEL;
1501     ret->shader->version = BWRITERPS_VERSION(1, 4);
1502     ret->funcs = &parser_ps_1_4;
1503     gen_oldps_input(ret->shader, 6);
1504 }
1505 
create_ps20_parser(struct asm_parser * ret)1506 void create_ps20_parser(struct asm_parser *ret) {
1507     TRACE_(parsed_shader)("ps_2_0\n");
1508 
1509     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1510     if(!ret->shader) {
1511         ERR("Failed to allocate memory for the shader\n");
1512         set_parse_status(&ret->status, PARSE_ERR);
1513         return;
1514     }
1515 
1516     ret->shader->type = ST_PIXEL;
1517     ret->shader->version = BWRITERPS_VERSION(2, 0);
1518     ret->funcs = &parser_ps_2;
1519     gen_oldps_input(ret->shader, 8);
1520 }
1521 
create_ps2x_parser(struct asm_parser * ret)1522 void create_ps2x_parser(struct asm_parser *ret) {
1523     TRACE_(parsed_shader)("ps_2_x\n");
1524 
1525     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1526     if(!ret->shader) {
1527         ERR("Failed to allocate memory for the shader\n");
1528         set_parse_status(&ret->status, PARSE_ERR);
1529         return;
1530     }
1531 
1532     ret->shader->type = ST_PIXEL;
1533     ret->shader->version = BWRITERPS_VERSION(2, 1);
1534     ret->funcs = &parser_ps_2_x;
1535     gen_oldps_input(ret->shader, 8);
1536 }
1537 
create_ps30_parser(struct asm_parser * ret)1538 void create_ps30_parser(struct asm_parser *ret) {
1539     TRACE_(parsed_shader)("ps_3_0\n");
1540 
1541     ret->shader = d3dcompiler_alloc(sizeof(*ret->shader));
1542     if(!ret->shader) {
1543         ERR("Failed to allocate memory for the shader\n");
1544         set_parse_status(&ret->status, PARSE_ERR);
1545         return;
1546     }
1547 
1548     ret->shader->type = ST_PIXEL;
1549     ret->shader->version = BWRITERPS_VERSION(3, 0);
1550     ret->funcs = &parser_ps_3;
1551 }
1552