2 * Mesa 3-D graphics library
5 * Copyright (C) 1999-2003 Brian Paul All Rights Reserved.
7 * Permission is hereby granted, free of charge, to any person obtaining a
8 * copy of this software and associated documentation files (the "Software"),
9 * to deal in the Software without restriction, including without limitation
10 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11 * and/or sell copies of the Software, and to permit persons to whom the
12 * Software is furnished to do so, subject to the following conditions:
14 * The above copyright notice and this permission notice shall be included
15 * in all copies or substantial portions of the Software.
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20 * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
21 * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 * NVIDIA vertex program parser.
38 #include "nvprogram.h"
39 #include "nvvertparse.h"
40 #include "nvvertprog.h"
45 * Current parsing state. This structure is passed among the parsing
46 * functions and keeps track of the current parser position and various
53 const GLubyte
*curLine
;
54 GLboolean isStateProgram
;
55 GLboolean isPositionInvariant
;
56 GLboolean isVersion1_1
;
58 GLuint outputsWritten
;
59 GLboolean anyProgRegsWritten
;
60 GLuint numInst
; /* number of instructions parsed */
65 * Called whenever we find an error during parsing.
68 record_error(struct parse_state
*parseState
, const char *msg
, int lineNo
)
72 const GLubyte
*lineStr
;
73 lineStr
= _mesa_find_line_column(parseState
->start
,
74 parseState
->pos
, &line
, &column
);
75 _mesa_debug(parseState
->ctx
,
76 "nvfragparse.c(%d): line %d, column %d:%s (%s)\n",
77 lineNo
, line
, column
, (char *) lineStr
, msg
);
78 _mesa_free((void *) lineStr
);
83 /* Check that no error was already recorded. Only record the first one. */
84 if (parseState
->ctx
->Program
.ErrorString
[0] == 0) {
85 _mesa_set_program_error(parseState
->ctx
,
86 parseState
->pos
- parseState
->start
,
92 #define RETURN_ERROR \
94 record_error(parseState, "Unexpected end of input.", __LINE__); \
98 #define RETURN_ERROR1(msg) \
100 record_error(parseState, msg, __LINE__); \
104 #define RETURN_ERROR2(msg1, msg2) \
107 _mesa_sprintf(err, "%s %s", msg1, msg2); \
108 record_error(parseState, err, __LINE__); \
116 static GLboolean
IsLetter(GLubyte b
)
118 return (b
>= 'a' && b
<= 'z') || (b
>= 'A' && b
<= 'Z');
122 static GLboolean
IsDigit(GLubyte b
)
124 return b
>= '0' && b
<= '9';
128 static GLboolean
IsWhitespace(GLubyte b
)
130 return b
== ' ' || b
== '\t' || b
== '\n' || b
== '\r';
135 * Starting at 'str' find the next token. A token can be an integer,
136 * an identifier or punctuation symbol.
137 * \return <= 0 we found an error, else, return number of characters parsed.
140 GetToken(struct parse_state
*parseState
, GLubyte
*token
)
142 const GLubyte
*str
= parseState
->pos
;
147 /* skip whitespace and comments */
148 while (str
[i
] && (IsWhitespace(str
[i
]) || str
[i
] == '#')) {
151 while (str
[i
] && (str
[i
] != '\n' && str
[i
] != '\r')) {
154 if (str
[i
] == '\n' || str
[i
] == '\r')
155 parseState
->curLine
= str
+ i
+ 1;
158 /* skip whitespace */
159 if (str
[i
] == '\n' || str
[i
] == '\r')
160 parseState
->curLine
= str
+ i
+ 1;
168 /* try matching an integer */
169 while (str
[i
] && IsDigit(str
[i
])) {
170 token
[j
++] = str
[i
++];
172 if (j
> 0 || !str
[i
]) {
177 /* try matching an identifier */
178 if (IsLetter(str
[i
])) {
179 while (str
[i
] && (IsLetter(str
[i
]) || IsDigit(str
[i
]))) {
180 token
[j
++] = str
[i
++];
186 /* punctuation character */
200 * Get next token from input stream and increment stream pointer past token.
203 Parse_Token(struct parse_state
*parseState
, GLubyte
*token
)
206 i
= GetToken(parseState
, token
);
208 parseState
->pos
+= (-i
);
211 parseState
->pos
+= i
;
217 * Get next token from input stream but don't increment stream pointer.
220 Peek_Token(struct parse_state
*parseState
, GLubyte
*token
)
223 i
= GetToken(parseState
, token
);
225 parseState
->pos
+= (-i
);
228 len
= _mesa_strlen((const char *) token
);
229 parseState
->pos
+= (i
- len
);
235 * Try to match 'pattern' as the next token after any whitespace/comments.
236 * Advance the current parsing position only if we match the pattern.
237 * \return GL_TRUE if pattern is matched, GL_FALSE otherwise.
240 Parse_String(struct parse_state
*parseState
, const char *pattern
)
245 /* skip whitespace and comments */
246 while (IsWhitespace(*parseState
->pos
) || *parseState
->pos
== '#') {
247 if (*parseState
->pos
== '#') {
248 while (*parseState
->pos
&& (*parseState
->pos
!= '\n' && *parseState
->pos
!= '\r')) {
249 parseState
->pos
+= 1;
251 if (*parseState
->pos
== '\n' || *parseState
->pos
== '\r')
252 parseState
->curLine
= parseState
->pos
+ 1;
255 /* skip whitespace */
256 if (*parseState
->pos
== '\n' || *parseState
->pos
== '\r')
257 parseState
->curLine
= parseState
->pos
+ 1;
258 parseState
->pos
+= 1;
262 /* Try to match the pattern */
264 for (i
= 0; pattern
[i
]; i
++) {
265 if (*m
!= (GLubyte
) pattern
[i
])
271 return GL_TRUE
; /* success */
275 /**********************************************************************/
277 static const char *InputRegisters
[MAX_NV_VERTEX_PROGRAM_INPUTS
+ 1] = {
278 "OPOS", "WGHT", "NRML", "COL0", "COL1", "FOGC", "6", "7",
279 "TEX0", "TEX1", "TEX2", "TEX3", "TEX4", "TEX5", "TEX6", "TEX7", NULL
282 static const char *OutputRegisters
[MAX_NV_VERTEX_PROGRAM_OUTPUTS
+ 1] = {
283 "HPOS", "COL0", "COL1", "BFC0", "BFC1", "FOGC", "PSIZ",
284 "TEX0", "TEX1", "TEX2", "TEX3", "TEX4", "TEX5", "TEX6", "TEX7", NULL
287 /* NOTE: the order here must match opcodes in nvvertprog.h */
288 static const char *Opcodes
[] = {
289 "MOV", "LIT", "RCP", "RSQ", "EXP", "LOG", "MUL", "ADD", "DP3", "DP4",
290 "DST", "MIN", "MAX", "SLT", "SGE", "MAD", "ARL", "DPH", "RCC", "SUB",
292 /* GL_ARB_vertex_program */
293 "FLR", "FRC", "EX2", "LG2", "POW", "XPD", "SWZ",
300 * Parse a temporary register: Rnn
303 Parse_TempReg(struct parse_state
*parseState
, GLint
*tempRegNum
)
307 /* Should be 'R##' */
308 if (!Parse_Token(parseState
, token
))
311 RETURN_ERROR1("Expected R##");
313 if (IsDigit(token
[1])) {
314 GLint reg
= _mesa_atoi((char *) (token
+ 1));
315 if (reg
>= MAX_NV_VERTEX_PROGRAM_TEMPS
)
316 RETURN_ERROR1("Bad temporary register name");
320 RETURN_ERROR1("Bad temporary register name");
328 * Parse address register "A0.x"
331 Parse_AddrReg(struct parse_state
*parseState
)
334 if (!Parse_String(parseState
, "A0"))
338 if (!Parse_String(parseState
, "."))
342 if (!Parse_String(parseState
, "x"))
350 * Parse absolute program parameter register "c[##]"
353 Parse_AbsParamReg(struct parse_state
*parseState
, GLint
*regNum
)
357 if (!Parse_String(parseState
, "c"))
360 if (!Parse_String(parseState
, "["))
363 if (!Parse_Token(parseState
, token
))
366 if (IsDigit(token
[0])) {
367 /* a numbered program parameter register */
368 GLint reg
= _mesa_atoi((char *) token
);
369 if (reg
>= MAX_NV_VERTEX_PROGRAM_PARAMS
)
370 RETURN_ERROR1("Bad program parameter number");
377 if (!Parse_String(parseState
, "]"))
385 Parse_ParamReg(struct parse_state
*parseState
, struct vp_src_register
*srcReg
)
389 if (!Parse_String(parseState
, "c"))
392 if (!Parse_String(parseState
, "["))
395 if (!Peek_Token(parseState
, token
))
398 if (IsDigit(token
[0])) {
399 /* a numbered program parameter register */
401 (void) Parse_Token(parseState
, token
);
402 reg
= _mesa_atoi((char *) token
);
403 if (reg
>= MAX_NV_VERTEX_PROGRAM_PARAMS
)
404 RETURN_ERROR1("Bad program parameter number");
405 srcReg
->File
= PROGRAM_ENV_PARAM
;
408 else if (_mesa_strcmp((const char *) token
, "A0") == 0) {
409 /* address register "A0.x" */
410 if (!Parse_AddrReg(parseState
))
413 srcReg
->RelAddr
= GL_TRUE
;
414 srcReg
->File
= PROGRAM_ENV_PARAM
;
415 /* Look for +/-N offset */
416 if (!Peek_Token(parseState
, token
))
419 if (token
[0] == '-' || token
[0] == '+') {
420 const GLubyte sign
= token
[0];
421 (void) Parse_Token(parseState
, token
); /* consume +/- */
423 /* an integer should be next */
424 if (!Parse_Token(parseState
, token
))
427 if (IsDigit(token
[0])) {
428 const GLint k
= _mesa_atoi((char *) token
);
431 RETURN_ERROR1("Bad address offset");
436 RETURN_ERROR1("Bad address offset");
445 /* probably got a ']', catch it below */
452 /* Match closing ']' */
453 if (!Parse_String(parseState
, "]"))
461 * Parse v[#] or v[<name>]
464 Parse_AttribReg(struct parse_state
*parseState
, GLint
*tempRegNum
)
470 if (!Parse_String(parseState
, "v"))
474 if (!Parse_String(parseState
, "["))
477 /* match number or named register */
478 if (!Parse_Token(parseState
, token
))
481 if (parseState
->isStateProgram
&& token
[0] != '0')
482 RETURN_ERROR1("Only v[0] accessible in vertex state programs");
484 if (IsDigit(token
[0])) {
485 GLint reg
= _mesa_atoi((char *) token
);
486 if (reg
>= MAX_NV_VERTEX_PROGRAM_INPUTS
)
487 RETURN_ERROR1("Bad vertex attribute register name");
491 for (j
= 0; InputRegisters
[j
]; j
++) {
492 if (_mesa_strcmp((const char *) token
, InputRegisters
[j
]) == 0) {
497 if (!InputRegisters
[j
]) {
498 /* unknown input register label */
499 RETURN_ERROR2("Bad register name", token
);
504 if (!Parse_String(parseState
, "]"))
512 Parse_OutputReg(struct parse_state
*parseState
, GLint
*outputRegNum
)
518 if (!Parse_String(parseState
, "o"))
522 if (!Parse_String(parseState
, "["))
525 /* Get output reg name */
526 if (!Parse_Token(parseState
, token
))
529 if (parseState
->isPositionInvariant
)
530 start
= 1; /* skip HPOS register name */
534 /* try to match an output register name */
535 for (j
= start
; OutputRegisters
[j
]; j
++) {
536 if (_mesa_strcmp((const char *) token
, OutputRegisters
[j
]) == 0) {
541 if (!OutputRegisters
[j
])
542 RETURN_ERROR1("Unrecognized output register name");
545 if (!Parse_String(parseState
, "]"))
546 RETURN_ERROR1("Expected ]");
553 Parse_MaskedDstReg(struct parse_state
*parseState
, struct vp_dst_register
*dstReg
)
557 /* Dst reg can be R<n> or o[n] */
558 if (!Peek_Token(parseState
, token
))
561 if (token
[0] == 'R') {
562 /* a temporary register */
563 dstReg
->File
= PROGRAM_TEMPORARY
;
564 if (!Parse_TempReg(parseState
, &dstReg
->Index
))
567 else if (!parseState
->isStateProgram
&& token
[0] == 'o') {
568 /* an output register */
569 dstReg
->File
= PROGRAM_OUTPUT
;
570 if (!Parse_OutputReg(parseState
, &dstReg
->Index
))
573 else if (parseState
->isStateProgram
&& token
[0] == 'c') {
574 /* absolute program parameter register */
575 dstReg
->File
= PROGRAM_ENV_PARAM
;
576 if (!Parse_AbsParamReg(parseState
, &dstReg
->Index
))
580 RETURN_ERROR1("Bad destination register name");
583 /* Parse optional write mask */
584 if (!Peek_Token(parseState
, token
))
587 if (token
[0] == '.') {
591 if (!Parse_String(parseState
, "."))
594 if (!Parse_Token(parseState
, token
))
597 dstReg
->WriteMask
[0] = GL_FALSE
;
598 dstReg
->WriteMask
[1] = GL_FALSE
;
599 dstReg
->WriteMask
[2] = GL_FALSE
;
600 dstReg
->WriteMask
[3] = GL_FALSE
;
602 if (token
[k
] == 'x') {
603 dstReg
->WriteMask
[0] = GL_TRUE
;
606 if (token
[k
] == 'y') {
607 dstReg
->WriteMask
[1] = GL_TRUE
;
610 if (token
[k
] == 'z') {
611 dstReg
->WriteMask
[2] = GL_TRUE
;
614 if (token
[k
] == 'w') {
615 dstReg
->WriteMask
[3] = GL_TRUE
;
619 RETURN_ERROR1("Bad writemask character");
624 dstReg
->WriteMask
[0] = GL_TRUE
;
625 dstReg
->WriteMask
[1] = GL_TRUE
;
626 dstReg
->WriteMask
[2] = GL_TRUE
;
627 dstReg
->WriteMask
[3] = GL_TRUE
;
634 Parse_SwizzleSrcReg(struct parse_state
*parseState
, struct vp_src_register
*srcReg
)
638 srcReg
->RelAddr
= GL_FALSE
;
641 if (!Peek_Token(parseState
, token
))
643 if (token
[0] == '-') {
644 (void) Parse_String(parseState
, "-");
645 srcReg
->Negate
= GL_TRUE
;
646 if (!Peek_Token(parseState
, token
))
650 srcReg
->Negate
= GL_FALSE
;
653 /* Src reg can be R<n>, c[n], c[n +/- offset], or a named vertex attrib */
654 if (token
[0] == 'R') {
655 srcReg
->File
= PROGRAM_TEMPORARY
;
656 if (!Parse_TempReg(parseState
, &srcReg
->Index
))
659 else if (token
[0] == 'c') {
660 if (!Parse_ParamReg(parseState
, srcReg
))
663 else if (token
[0] == 'v') {
664 srcReg
->File
= PROGRAM_INPUT
;
665 if (!Parse_AttribReg(parseState
, &srcReg
->Index
))
669 RETURN_ERROR2("Bad source register name", token
);
672 /* init swizzle fields */
673 srcReg
->Swizzle
[0] = 0;
674 srcReg
->Swizzle
[1] = 1;
675 srcReg
->Swizzle
[2] = 2;
676 srcReg
->Swizzle
[3] = 3;
678 /* Look for optional swizzle suffix */
679 if (!Peek_Token(parseState
, token
))
681 if (token
[0] == '.') {
682 (void) Parse_String(parseState
, "."); /* consume . */
684 if (!Parse_Token(parseState
, token
))
688 /* single letter swizzle */
690 ASSIGN_4V(srcReg
->Swizzle
, 0, 0, 0, 0);
691 else if (token
[0] == 'y')
692 ASSIGN_4V(srcReg
->Swizzle
, 1, 1, 1, 1);
693 else if (token
[0] == 'z')
694 ASSIGN_4V(srcReg
->Swizzle
, 2, 2, 2, 2);
695 else if (token
[0] == 'w')
696 ASSIGN_4V(srcReg
->Swizzle
, 3, 3, 3, 3);
698 RETURN_ERROR1("Expected x, y, z, or w");
701 /* 2, 3 or 4-component swizzle */
703 for (k
= 0; token
[k
] && k
< 5; k
++) {
705 srcReg
->Swizzle
[k
] = 0;
706 else if (token
[k
] == 'y')
707 srcReg
->Swizzle
[k
] = 1;
708 else if (token
[k
] == 'z')
709 srcReg
->Swizzle
[k
] = 2;
710 else if (token
[k
] == 'w')
711 srcReg
->Swizzle
[k
] = 3;
725 Parse_ScalarSrcReg(struct parse_state
*parseState
, struct vp_src_register
*srcReg
)
729 srcReg
->RelAddr
= GL_FALSE
;
732 if (!Peek_Token(parseState
, token
))
734 if (token
[0] == '-') {
735 srcReg
->Negate
= GL_TRUE
;
736 (void) Parse_String(parseState
, "-"); /* consume '-' */
737 if (!Peek_Token(parseState
, token
))
741 srcReg
->Negate
= GL_FALSE
;
744 /* Src reg can be R<n>, c[n], c[n +/- offset], or a named vertex attrib */
745 if (token
[0] == 'R') {
746 srcReg
->File
= PROGRAM_TEMPORARY
;
747 if (!Parse_TempReg(parseState
, &srcReg
->Index
))
750 else if (token
[0] == 'c') {
751 if (!Parse_ParamReg(parseState
, srcReg
))
754 else if (token
[0] == 'v') {
755 srcReg
->File
= PROGRAM_INPUT
;
756 if (!Parse_AttribReg(parseState
, &srcReg
->Index
))
760 RETURN_ERROR2("Bad source register name", token
);
763 /* Look for .[xyzw] suffix */
764 if (!Parse_String(parseState
, "."))
767 if (!Parse_Token(parseState
, token
))
770 if (token
[0] == 'x' && token
[1] == 0) {
771 srcReg
->Swizzle
[0] = 0;
773 else if (token
[0] == 'y' && token
[1] == 0) {
774 srcReg
->Swizzle
[0] = 1;
776 else if (token
[0] == 'z' && token
[1] == 0) {
777 srcReg
->Swizzle
[0] = 2;
779 else if (token
[0] == 'w' && token
[1] == 0) {
780 srcReg
->Swizzle
[0] = 3;
783 RETURN_ERROR1("Bad scalar source suffix");
785 srcReg
->Swizzle
[1] = srcReg
->Swizzle
[2] = srcReg
->Swizzle
[3] = 0;
792 Parse_UnaryOpInstruction(struct parse_state
*parseState
,
793 struct vp_instruction
*inst
, enum vp_opcode opcode
)
795 if (opcode
== VP_OPCODE_ABS
&& !parseState
->isVersion1_1
)
796 RETURN_ERROR1("ABS illegal for vertex program 1.0");
798 inst
->Opcode
= opcode
;
799 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
802 if (!Parse_MaskedDstReg(parseState
, &inst
->DstReg
))
806 if (!Parse_String(parseState
, ","))
810 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[0]))
814 if (!Parse_String(parseState
, ";"))
822 Parse_BiOpInstruction(struct parse_state
*parseState
,
823 struct vp_instruction
*inst
, enum vp_opcode opcode
)
825 if (opcode
== VP_OPCODE_DPH
&& !parseState
->isVersion1_1
)
826 RETURN_ERROR1("DPH illegal for vertex program 1.0");
827 if (opcode
== VP_OPCODE_SUB
&& !parseState
->isVersion1_1
)
828 RETURN_ERROR1("SUB illegal for vertex program 1.0");
830 inst
->Opcode
= opcode
;
831 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
834 if (!Parse_MaskedDstReg(parseState
, &inst
->DstReg
))
838 if (!Parse_String(parseState
, ","))
842 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[0]))
846 if (!Parse_String(parseState
, ","))
850 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[1]))
854 if (!Parse_String(parseState
, ";"))
857 /* make sure we don't reference more than one program parameter register */
858 if (inst
->SrcReg
[0].File
== PROGRAM_ENV_PARAM
&&
859 inst
->SrcReg
[1].File
== PROGRAM_ENV_PARAM
&&
860 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[1].Index
)
861 RETURN_ERROR1("Can't reference two program parameter registers");
863 /* make sure we don't reference more than one vertex attribute register */
864 if (inst
->SrcReg
[0].File
== PROGRAM_INPUT
&&
865 inst
->SrcReg
[1].File
== PROGRAM_INPUT
&&
866 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[1].Index
)
867 RETURN_ERROR1("Can't reference two vertex attribute registers");
874 Parse_TriOpInstruction(struct parse_state
*parseState
,
875 struct vp_instruction
*inst
, enum vp_opcode opcode
)
877 inst
->Opcode
= opcode
;
878 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
881 if (!Parse_MaskedDstReg(parseState
, &inst
->DstReg
))
885 if (!Parse_String(parseState
, ","))
889 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[0]))
893 if (!Parse_String(parseState
, ","))
897 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[1]))
901 if (!Parse_String(parseState
, ","))
905 if (!Parse_SwizzleSrcReg(parseState
, &inst
->SrcReg
[2]))
909 if (!Parse_String(parseState
, ";"))
912 /* make sure we don't reference more than one program parameter register */
913 if ((inst
->SrcReg
[0].File
== PROGRAM_ENV_PARAM
&&
914 inst
->SrcReg
[1].File
== PROGRAM_ENV_PARAM
&&
915 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[1].Index
) ||
916 (inst
->SrcReg
[0].File
== PROGRAM_ENV_PARAM
&&
917 inst
->SrcReg
[2].File
== PROGRAM_ENV_PARAM
&&
918 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[2].Index
) ||
919 (inst
->SrcReg
[1].File
== PROGRAM_ENV_PARAM
&&
920 inst
->SrcReg
[2].File
== PROGRAM_ENV_PARAM
&&
921 inst
->SrcReg
[1].Index
!= inst
->SrcReg
[2].Index
))
922 RETURN_ERROR1("Can only reference one program register");
924 /* make sure we don't reference more than one vertex attribute register */
925 if ((inst
->SrcReg
[0].File
== PROGRAM_INPUT
&&
926 inst
->SrcReg
[1].File
== PROGRAM_INPUT
&&
927 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[1].Index
) ||
928 (inst
->SrcReg
[0].File
== PROGRAM_INPUT
&&
929 inst
->SrcReg
[2].File
== PROGRAM_INPUT
&&
930 inst
->SrcReg
[0].Index
!= inst
->SrcReg
[2].Index
) ||
931 (inst
->SrcReg
[1].File
== PROGRAM_INPUT
&&
932 inst
->SrcReg
[2].File
== PROGRAM_INPUT
&&
933 inst
->SrcReg
[1].Index
!= inst
->SrcReg
[2].Index
))
934 RETURN_ERROR1("Can only reference one input register");
941 Parse_ScalarInstruction(struct parse_state
*parseState
,
942 struct vp_instruction
*inst
, enum vp_opcode opcode
)
944 if (opcode
== VP_OPCODE_RCC
&& !parseState
->isVersion1_1
)
945 RETURN_ERROR1("RCC illegal for vertex program 1.0");
947 inst
->Opcode
= opcode
;
948 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
951 if (!Parse_MaskedDstReg(parseState
, &inst
->DstReg
))
955 if (!Parse_String(parseState
, ","))
959 if (!Parse_ScalarSrcReg(parseState
, &inst
->SrcReg
[0]))
963 if (!Parse_String(parseState
, ";"))
971 Parse_AddressInstruction(struct parse_state
*parseState
, struct vp_instruction
*inst
)
973 inst
->Opcode
= VP_OPCODE_ARL
;
974 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
977 if (!Parse_AddrReg(parseState
))
981 if (!Parse_String(parseState
, ","))
985 if (!Parse_ScalarSrcReg(parseState
, &inst
->SrcReg
[0]))
989 if (!Parse_String(parseState
, ";"))
997 Parse_EndInstruction(struct parse_state
*parseState
, struct vp_instruction
*inst
)
1001 inst
->Opcode
= VP_OPCODE_END
;
1002 inst
->StringPos
= parseState
->curLine
- parseState
->start
;
1004 /* this should fail! */
1005 if (Parse_Token(parseState
, token
))
1006 RETURN_ERROR2("Unexpected token after END:", token
);
1013 Parse_OptionSequence(struct parse_state
*parseState
,
1014 struct vp_instruction program
[])
1017 if (!Parse_String(parseState
, "OPTION"))
1018 return GL_TRUE
; /* ok, not an OPTION statement */
1019 if (Parse_String(parseState
, "NV_position_invariant")) {
1020 parseState
->isPositionInvariant
= GL_TRUE
;
1023 RETURN_ERROR1("unexpected OPTION statement");
1025 if (!Parse_String(parseState
, ";"))
1032 Parse_InstructionSequence(struct parse_state
*parseState
,
1033 struct vp_instruction program
[])
1036 struct vp_instruction
*inst
= program
+ parseState
->numInst
;
1038 /* Initialize the instruction */
1039 inst
->SrcReg
[0].File
= (enum register_file
) -1;
1040 inst
->SrcReg
[1].File
= (enum register_file
) -1;
1041 inst
->SrcReg
[2].File
= (enum register_file
) -1;
1042 inst
->DstReg
.File
= (enum register_file
) -1;
1044 if (Parse_String(parseState
, "MOV")) {
1045 if (!Parse_UnaryOpInstruction(parseState
, inst
, VP_OPCODE_MOV
))
1048 else if (Parse_String(parseState
, "LIT")) {
1049 if (!Parse_UnaryOpInstruction(parseState
, inst
, VP_OPCODE_LIT
))
1052 else if (Parse_String(parseState
, "ABS")) {
1053 if (!Parse_UnaryOpInstruction(parseState
, inst
, VP_OPCODE_ABS
))
1056 else if (Parse_String(parseState
, "MUL")) {
1057 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_MUL
))
1060 else if (Parse_String(parseState
, "ADD")) {
1061 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_ADD
))
1064 else if (Parse_String(parseState
, "DP3")) {
1065 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_DP3
))
1068 else if (Parse_String(parseState
, "DP4")) {
1069 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_DP4
))
1072 else if (Parse_String(parseState
, "DST")) {
1073 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_DST
))
1076 else if (Parse_String(parseState
, "MIN")) {
1077 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_MIN
))
1080 else if (Parse_String(parseState
, "MAX")) {
1081 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_MAX
))
1084 else if (Parse_String(parseState
, "SLT")) {
1085 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_SLT
))
1088 else if (Parse_String(parseState
, "SGE")) {
1089 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_SGE
))
1092 else if (Parse_String(parseState
, "DPH")) {
1093 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_DPH
))
1096 else if (Parse_String(parseState
, "SUB")) {
1097 if (!Parse_BiOpInstruction(parseState
, inst
, VP_OPCODE_SUB
))
1100 else if (Parse_String(parseState
, "MAD")) {
1101 if (!Parse_TriOpInstruction(parseState
, inst
, VP_OPCODE_MAD
))
1104 else if (Parse_String(parseState
, "RCP")) {
1105 if (!Parse_ScalarInstruction(parseState
, inst
, VP_OPCODE_RCP
))
1108 else if (Parse_String(parseState
, "RSQ")) {
1109 if (!Parse_ScalarInstruction(parseState
, inst
, VP_OPCODE_RSQ
))
1112 else if (Parse_String(parseState
, "EXP")) {
1113 if (!Parse_ScalarInstruction(parseState
, inst
, VP_OPCODE_EXP
))
1116 else if (Parse_String(parseState
, "LOG")) {
1117 if (!Parse_ScalarInstruction(parseState
, inst
, VP_OPCODE_LOG
))
1120 else if (Parse_String(parseState
, "RCC")) {
1121 if (!Parse_ScalarInstruction(parseState
, inst
, VP_OPCODE_RCC
))
1124 else if (Parse_String(parseState
, "ARL")) {
1125 if (!Parse_AddressInstruction(parseState
, inst
))
1128 else if (Parse_String(parseState
, "END")) {
1129 if (!Parse_EndInstruction(parseState
, inst
))
1132 parseState
->numInst
++;
1133 return GL_TRUE
; /* all done */
1137 /* bad instruction name */
1138 RETURN_ERROR1("Unexpected token");
1141 /* examine input/output registers */
1142 if (inst
->DstReg
.File
== PROGRAM_OUTPUT
)
1143 parseState
->outputsWritten
|= (1 << inst
->DstReg
.Index
);
1144 else if (inst
->DstReg
.File
== PROGRAM_ENV_PARAM
)
1145 parseState
->anyProgRegsWritten
= GL_TRUE
;
1147 if (inst
->SrcReg
[0].File
== PROGRAM_INPUT
)
1148 parseState
->inputsRead
|= (1 << inst
->SrcReg
[0].Index
);
1149 if (inst
->SrcReg
[1].File
== PROGRAM_INPUT
)
1150 parseState
->inputsRead
|= (1 << inst
->SrcReg
[1].Index
);
1151 if (inst
->SrcReg
[2].File
== PROGRAM_INPUT
)
1152 parseState
->inputsRead
|= (1 << inst
->SrcReg
[2].Index
);
1154 parseState
->numInst
++;
1156 if (parseState
->numInst
>= MAX_NV_VERTEX_PROGRAM_INSTRUCTIONS
)
1157 RETURN_ERROR1("Program too long");
1165 Parse_Program(struct parse_state
*parseState
,
1166 struct vp_instruction instBuffer
[])
1168 if (parseState
->isVersion1_1
) {
1169 if (!Parse_OptionSequence(parseState
, instBuffer
)) {
1173 return Parse_InstructionSequence(parseState
, instBuffer
);
1178 * Parse/compile the 'str' returning the compiled 'program'.
1179 * ctx->Program.ErrorPos will be -1 if successful. Otherwise, ErrorPos
1180 * indicates the position of the error in 'str'.
1183 _mesa_parse_nv_vertex_program(GLcontext
*ctx
, GLenum dstTarget
,
1184 const GLubyte
*str
, GLsizei len
,
1185 struct vertex_program
*program
)
1187 struct parse_state parseState
;
1188 struct vp_instruction instBuffer
[MAX_NV_VERTEX_PROGRAM_INSTRUCTIONS
];
1189 struct vp_instruction
*newInst
;
1191 GLubyte
*programString
;
1193 /* Make a null-terminated copy of the program string */
1194 programString
= (GLubyte
*) MALLOC(len
+ 1);
1195 if (!programString
) {
1196 _mesa_error(ctx
, GL_OUT_OF_MEMORY
, "glLoadProgramNV");
1199 MEMCPY(programString
, str
, len
);
1200 programString
[len
] = 0;
1202 /* Get ready to parse */
1203 parseState
.ctx
= ctx
;
1204 parseState
.start
= programString
;
1205 parseState
.isPositionInvariant
= GL_FALSE
;
1206 parseState
.isVersion1_1
= GL_FALSE
;
1207 parseState
.numInst
= 0;
1208 parseState
.inputsRead
= 0;
1209 parseState
.outputsWritten
= 0;
1210 parseState
.anyProgRegsWritten
= GL_FALSE
;
1212 /* Reset error state */
1213 _mesa_set_program_error(ctx
, -1, NULL
);
1215 /* check the program header */
1216 if (_mesa_strncmp((const char *) programString
, "!!VP1.0", 7) == 0) {
1217 target
= GL_VERTEX_PROGRAM_NV
;
1218 parseState
.pos
= programString
+ 7;
1219 parseState
.isStateProgram
= GL_FALSE
;
1221 else if (_mesa_strncmp((const char *) programString
, "!!VP1.1", 7) == 0) {
1222 target
= GL_VERTEX_PROGRAM_NV
;
1223 parseState
.pos
= programString
+ 7;
1224 parseState
.isStateProgram
= GL_FALSE
;
1225 parseState
.isVersion1_1
= GL_TRUE
;
1227 else if (_mesa_strncmp((const char *) programString
, "!!VSP1.0", 8) == 0) {
1228 target
= GL_VERTEX_STATE_PROGRAM_NV
;
1229 parseState
.pos
= programString
+ 8;
1230 parseState
.isStateProgram
= GL_TRUE
;
1233 /* invalid header */
1234 ctx
->Program
.ErrorPos
= 0;
1235 _mesa_error(ctx
, GL_INVALID_OPERATION
, "glLoadProgramNV(bad header)");
1239 /* make sure target and header match */
1240 if (target
!= dstTarget
) {
1241 _mesa_error(ctx
, GL_INVALID_OPERATION
,
1242 "glLoadProgramNV(target mismatch)");
1247 if (Parse_Program(&parseState
, instBuffer
)) {
1248 /* successful parse! */
1250 if (parseState
.isStateProgram
) {
1251 if (!parseState
.anyProgRegsWritten
) {
1252 _mesa_error(ctx
, GL_INVALID_OPERATION
,
1253 "glLoadProgramNV(c[#] not written)");
1258 if (!parseState
.isPositionInvariant
&&
1259 !(parseState
.outputsWritten
& 1)) {
1260 /* bit 1 = HPOS register */
1261 _mesa_error(ctx
, GL_INVALID_OPERATION
,
1262 "glLoadProgramNV(HPOS not written)");
1267 /* copy the compiled instructions */
1268 assert(parseState
.numInst
<= MAX_NV_VERTEX_PROGRAM_INSTRUCTIONS
);
1269 newInst
= (struct vp_instruction
*)
1270 MALLOC(parseState
.numInst
* sizeof(struct vp_instruction
));
1272 _mesa_error(ctx
, GL_OUT_OF_MEMORY
, "glLoadProgramNV");
1273 FREE(programString
);
1274 return; /* out of memory */
1276 MEMCPY(newInst
, instBuffer
,
1277 parseState
.numInst
* sizeof(struct vp_instruction
));
1279 /* install the program */
1280 program
->Base
.Target
= target
;
1281 if (program
->Base
.String
) {
1282 FREE(program
->Base
.String
);
1284 program
->Base
.String
= programString
;
1285 program
->Base
.Format
= GL_PROGRAM_FORMAT_ASCII_ARB
;
1286 if (program
->Instructions
) {
1287 FREE(program
->Instructions
);
1289 program
->Instructions
= newInst
;
1290 program
->InputsRead
= parseState
.inputsRead
;
1291 program
->OutputsWritten
= parseState
.outputsWritten
;
1292 program
->IsPositionInvariant
= parseState
.isPositionInvariant
;
1295 _mesa_printf("--- glLoadProgramNV result ---\n");
1296 _mesa_print_nv_vertex_program(program
);
1297 _mesa_printf("------------------------------\n");
1302 _mesa_error(ctx
, GL_INVALID_OPERATION
, "glLoadProgramNV");
1303 /* NOTE: _mesa_set_program_error would have been called already */
1304 /* GL_NV_vertex_program isn't supposed to set the error string
1305 * so we reset it here.
1307 _mesa_set_program_error(ctx
, ctx
->Program
.ErrorPos
, NULL
);
1313 PrintSrcReg(const struct vp_src_register
*src
)
1315 static const char comps
[5] = "xyzw";
1320 _mesa_printf("c[A0.x + %d]", src
->Index
);
1321 else if (src
->Index
< 0)
1322 _mesa_printf("c[A0.x - %d]", -src
->Index
);
1324 _mesa_printf("c[A0.x]");
1326 else if (src
->File
== PROGRAM_OUTPUT
) {
1327 _mesa_printf("o[%s]", OutputRegisters
[src
->Index
]);
1329 else if (src
->File
== PROGRAM_INPUT
) {
1330 _mesa_printf("v[%s]", InputRegisters
[src
->Index
]);
1332 else if (src
->File
== PROGRAM_ENV_PARAM
) {
1333 _mesa_printf("c[%d]", src
->Index
);
1336 ASSERT(src
->File
== PROGRAM_TEMPORARY
);
1337 _mesa_printf("R%d", src
->Index
);
1340 if (src
->Swizzle
[0] == src
->Swizzle
[1] &&
1341 src
->Swizzle
[0] == src
->Swizzle
[2] &&
1342 src
->Swizzle
[0] == src
->Swizzle
[3]) {
1343 _mesa_printf(".%c", comps
[src
->Swizzle
[0]]);
1345 else if (src
->Swizzle
[0] != 0 ||
1346 src
->Swizzle
[1] != 1 ||
1347 src
->Swizzle
[2] != 2 ||
1348 src
->Swizzle
[3] != 3) {
1349 _mesa_printf(".%c%c%c%c",
1350 comps
[src
->Swizzle
[0]],
1351 comps
[src
->Swizzle
[1]],
1352 comps
[src
->Swizzle
[2]],
1353 comps
[src
->Swizzle
[3]]);
1359 PrintDstReg(const struct vp_dst_register
*dst
)
1361 GLint w
= dst
->WriteMask
[0] + dst
->WriteMask
[1]
1362 + dst
->WriteMask
[2] + dst
->WriteMask
[3];
1364 if (dst
->File
== PROGRAM_OUTPUT
) {
1365 _mesa_printf("o[%s]", OutputRegisters
[dst
->Index
]);
1367 else if (dst
->File
== PROGRAM_INPUT
) {
1368 _mesa_printf("v[%s]", InputRegisters
[dst
->Index
]);
1370 else if (dst
->File
== PROGRAM_ENV_PARAM
) {
1371 _mesa_printf("c[%d]", dst
->Index
);
1374 ASSERT(dst
->File
== PROGRAM_TEMPORARY
);
1375 _mesa_printf("R%d", dst
->Index
);
1378 if (w
!= 0 && w
!= 4) {
1380 if (dst
->WriteMask
[0])
1382 if (dst
->WriteMask
[1])
1384 if (dst
->WriteMask
[2])
1386 if (dst
->WriteMask
[3])
1393 * Print a single NVIDIA vertex program instruction.
1396 _mesa_print_nv_vertex_instruction(const struct vp_instruction
*inst
)
1398 switch (inst
->Opcode
) {
1407 _mesa_printf("%s ", Opcodes
[(int) inst
->Opcode
]);
1408 PrintDstReg(&inst
->DstReg
);
1410 PrintSrcReg(&inst
->SrcReg
[0]);
1411 _mesa_printf(";\n");
1424 _mesa_printf("%s ", Opcodes
[(int) inst
->Opcode
]);
1425 PrintDstReg(&inst
->DstReg
);
1427 PrintSrcReg(&inst
->SrcReg
[0]);
1429 PrintSrcReg(&inst
->SrcReg
[1]);
1430 _mesa_printf(";\n");
1433 _mesa_printf("MAD ");
1434 PrintDstReg(&inst
->DstReg
);
1436 PrintSrcReg(&inst
->SrcReg
[0]);
1438 PrintSrcReg(&inst
->SrcReg
[1]);
1440 PrintSrcReg(&inst
->SrcReg
[2]);
1441 _mesa_printf(";\n");
1444 _mesa_printf("ARL A0.x, ");
1445 PrintSrcReg(&inst
->SrcReg
[0]);
1446 _mesa_printf(";\n");
1449 _mesa_printf("END\n");
1452 _mesa_printf("BAD INSTRUCTION\n");
1458 * Print (unparse) the given vertex program. Just for debugging.
1461 _mesa_print_nv_vertex_program(const struct vertex_program
*program
)
1463 const struct vp_instruction
*inst
;
1465 for (inst
= program
->Instructions
; ; inst
++) {
1466 _mesa_print_nv_vertex_instruction(inst
);
1467 if (inst
->Opcode
== VP_OPCODE_END
)
1474 _mesa_nv_vertex_input_register_name(GLuint i
)
1476 ASSERT(i
< MAX_NV_VERTEX_PROGRAM_INPUTS
);
1477 return InputRegisters
[i
];
1482 _mesa_nv_vertex_output_register_name(GLuint i
)
1484 ASSERT(i
< MAX_NV_VERTEX_PROGRAM_OUTPUTS
);
1485 return OutputRegisters
[i
];