Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 1 | %{ |
| 2 | /* |
| 3 | * Copyright © 2010 Intel Corporation |
| 4 | * |
| 5 | * Permission is hereby granted, free of charge, to any person obtaining a |
| 6 | * copy of this software and associated documentation files (the "Software"), |
| 7 | * to deal in the Software without restriction, including without limitation |
| 8 | * the rights to use, copy, modify, merge, publish, distribute, sublicense, |
| 9 | * and/or sell copies of the Software, and to permit persons to whom the |
| 10 | * Software is furnished to do so, subject to the following conditions: |
| 11 | * |
| 12 | * The above copyright notice and this permission notice (including the next |
| 13 | * paragraph) shall be included in all copies or substantial portions of the |
| 14 | * Software. |
| 15 | * |
| 16 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 17 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 18 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| 19 | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| 20 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
| 21 | * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
| 22 | * DEALINGS IN THE SOFTWARE. |
| 23 | */ |
| 24 | |
| 25 | #include <stdio.h> |
| 26 | #include <string.h> |
Kenneth Graunke | 7e908a6 | 2010-07-02 15:31:26 -0700 | [diff] [blame] | 27 | #include <ctype.h> |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 28 | |
Carl Worth | a1e32bc | 2010-05-10 13:17:25 -0700 | [diff] [blame] | 29 | #include "glcpp.h" |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 30 | #include "glcpp-parse.h" |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 31 | |
Carl Worth | a9bb4bc | 2010-07-20 15:03:20 -0700 | [diff] [blame^] | 32 | /* Flex annoyingly generates some functions without making them |
| 33 | * static. Let's declare them here. */ |
| 34 | int glcpp_get_column (yyscan_t yyscanner); |
| 35 | void glcpp_set_column (int column_no , yyscan_t yyscanner); |
| 36 | |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 37 | #define YY_USER_ACTION \ |
| 38 | do { \ |
| 39 | yylloc->source = 0; \ |
| 40 | yylloc->first_column = yycolumn + 1; \ |
Kenneth Graunke | 8f32221 | 2010-06-18 15:23:50 -0700 | [diff] [blame] | 41 | yylloc->first_line = yylineno; \ |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 42 | yycolumn += yyleng; \ |
| 43 | } while(0); |
Kenneth Graunke | 388ab9f | 2010-07-07 11:40:51 -0700 | [diff] [blame] | 44 | #define YY_USER_INIT yylineno = 0; yycolumn = 0; |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 45 | %} |
| 46 | |
Kenneth Graunke | 465e03e | 2010-06-16 16:35:57 -0700 | [diff] [blame] | 47 | %option bison-bridge bison-locations reentrant noyywrap |
Carl Worth | 5070a20 | 2010-05-12 12:45:33 -0700 | [diff] [blame] | 48 | %option extra-type="glcpp_parser_t *" |
Kenneth Graunke | 254a485 | 2010-06-16 11:51:43 -0700 | [diff] [blame] | 49 | %option prefix="glcpp_" |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 50 | %option stack |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 51 | |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 52 | %x DONE COMMENT |
Kenneth Graunke | f82d673 | 2010-06-16 12:53:19 -0700 | [diff] [blame] | 53 | |
Carl Worth | 0b27b5f | 2010-05-10 16:16:06 -0700 | [diff] [blame] | 54 | SPACE [[:space:]] |
| 55 | NONSPACE [^[:space:]] |
Carl Worth | 33cc400 | 2010-05-12 12:17:10 -0700 | [diff] [blame] | 56 | NEWLINE [\n] |
Carl Worth | 0b27b5f | 2010-05-10 16:16:06 -0700 | [diff] [blame] | 57 | HSPACE [ \t] |
Carl Worth | e36a4d5 | 2010-05-14 17:29:24 -0700 | [diff] [blame] | 58 | HASH ^{HSPACE}*#{HSPACE}* |
Carl Worth | 0b27b5f | 2010-05-10 16:16:06 -0700 | [diff] [blame] | 59 | IDENTIFIER [_a-zA-Z][_a-zA-Z0-9]* |
Carl Worth | 6310169 | 2010-05-29 05:07:24 -0700 | [diff] [blame] | 60 | PUNCTUATION [][(){}.&*~!/%<>^|;,=+-] |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 61 | OTHER [^][(){}.&*~!/%<>^|;,=#[:space:]+-]+ |
Carl Worth | 33cc400 | 2010-05-12 12:17:10 -0700 | [diff] [blame] | 62 | |
Carl Worth | 03f6d5d | 2010-05-24 11:29:02 -0700 | [diff] [blame] | 63 | DECIMAL_INTEGER [1-9][0-9]*[uU]? |
| 64 | OCTAL_INTEGER 0[0-7]*[uU]? |
| 65 | HEXADECIMAL_INTEGER 0[xX][0-9a-fA-F]+[uU]? |
| 66 | |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 67 | %% |
| 68 | |
Carl Worth | 2571415 | 2010-06-01 12:18:43 -0700 | [diff] [blame] | 69 | /* Single-line comments */ |
Kenneth Graunke | e6ae7af | 2010-06-21 15:11:01 -0700 | [diff] [blame] | 70 | "//"[^\n]*\n { |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 71 | yylineno++; |
| 72 | yycolumn = 0; |
Carl Worth | 2571415 | 2010-06-01 12:18:43 -0700 | [diff] [blame] | 73 | return NEWLINE; |
| 74 | } |
| 75 | |
| 76 | /* Multi-line comments */ |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 77 | "/*" { yy_push_state(COMMENT, yyscanner); } |
| 78 | <COMMENT>[^*\n]* |
| 79 | <COMMENT>[^*\n]*\n { yylineno++; yycolumn = 0; } |
| 80 | <COMMENT>"*"+[^*/\n]* |
| 81 | <COMMENT>"*"+[^*/\n]*\n { yylineno++; yycolumn = 0; } |
| 82 | <COMMENT>"*"+"/" { |
| 83 | yy_pop_state(yyscanner); |
Carl Worth | 2571415 | 2010-06-01 12:18:43 -0700 | [diff] [blame] | 84 | if (yyextra->space_tokens) |
| 85 | return SPACE; |
| 86 | } |
| 87 | |
Kenneth Graunke | 3b73ea3 | 2010-06-16 12:21:17 -0700 | [diff] [blame] | 88 | /* glcpp doesn't handle #extension, #version, or #pragma directives. |
| 89 | * Simply pass them through to the main compiler's lexer/parser. */ |
Kenneth Graunke | 03ee338 | 2010-06-18 15:41:00 -0700 | [diff] [blame] | 90 | {HASH}(extension|version|pragma)[^\n]+ { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 91 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Kenneth Graunke | 8f32221 | 2010-06-18 15:23:50 -0700 | [diff] [blame] | 92 | yylineno++; |
| 93 | yycolumn = 0; |
Kenneth Graunke | 3b73ea3 | 2010-06-16 12:21:17 -0700 | [diff] [blame] | 94 | return OTHER; |
| 95 | } |
| 96 | |
Kenneth Graunke | 77260fc | 2010-06-17 14:36:34 -0700 | [diff] [blame] | 97 | {HASH}ifdef/.*\n { |
Carl Worth | 1d7e03e | 2010-07-20 14:13:32 -0700 | [diff] [blame] | 98 | yyextra->lexing_if = 1; |
Kenneth Graunke | 77260fc | 2010-06-17 14:36:34 -0700 | [diff] [blame] | 99 | yyextra->space_tokens = 0; |
| 100 | return HASH_IFDEF; |
| 101 | } |
| 102 | |
| 103 | {HASH}ifndef/.*\n { |
Carl Worth | 1d7e03e | 2010-07-20 14:13:32 -0700 | [diff] [blame] | 104 | yyextra->lexing_if = 1; |
Kenneth Graunke | 77260fc | 2010-06-17 14:36:34 -0700 | [diff] [blame] | 105 | yyextra->space_tokens = 0; |
| 106 | return HASH_IFNDEF; |
| 107 | } |
| 108 | |
Carl Worth | 1d7e03e | 2010-07-20 14:13:32 -0700 | [diff] [blame] | 109 | {HASH}if/[^_a-zA-Z0-9].*\n { |
Carl Worth | a771a40 | 2010-06-01 11:20:18 -0700 | [diff] [blame] | 110 | yyextra->lexing_if = 1; |
| 111 | yyextra->space_tokens = 0; |
| 112 | return HASH_IF; |
| 113 | } |
| 114 | |
| 115 | {HASH}elif/.*\n { |
| 116 | yyextra->lexing_if = 1; |
| 117 | yyextra->space_tokens = 0; |
| 118 | return HASH_ELIF; |
| 119 | } |
| 120 | |
| 121 | {HASH}else/.*\n { |
| 122 | yyextra->space_tokens = 0; |
| 123 | return HASH_ELSE; |
| 124 | } |
| 125 | |
| 126 | {HASH}endif/.*\n { |
| 127 | yyextra->space_tokens = 0; |
| 128 | return HASH_ENDIF; |
| 129 | } |
| 130 | |
| 131 | /* When skipping (due to an #if 0 or similar) consume anything |
Carl Worth | 1d7e03e | 2010-07-20 14:13:32 -0700 | [diff] [blame] | 132 | * up to a newline. We do this with less priority than any |
Carl Worth | a771a40 | 2010-06-01 11:20:18 -0700 | [diff] [blame] | 133 | * #if-related directive (#if, #elif, #else, #endif), but with |
| 134 | * more priority than any other directive or token to avoid |
| 135 | * any side-effects from skipped content. |
| 136 | * |
| 137 | * We use the lexing_if flag to avoid skipping any part of an |
| 138 | * if conditional expression. */ |
| 139 | [^\n]+/\n { |
Kenneth Graunke | 8f32221 | 2010-06-18 15:23:50 -0700 | [diff] [blame] | 140 | /* Since this rule always matches, YY_USER_ACTION gets called for it, |
| 141 | * wrongly incrementing yycolumn. We undo that effect here. */ |
| 142 | yycolumn -= yyleng; |
Carl Worth | a771a40 | 2010-06-01 11:20:18 -0700 | [diff] [blame] | 143 | if (yyextra->lexing_if || |
| 144 | yyextra->skip_stack == NULL || |
| 145 | yyextra->skip_stack->type == SKIP_NO_SKIP) |
| 146 | { |
| 147 | REJECT; |
| 148 | } |
| 149 | } |
| 150 | |
Kenneth Graunke | 7e908a6 | 2010-07-02 15:31:26 -0700 | [diff] [blame] | 151 | {HASH}error.* { |
| 152 | char *p; |
| 153 | for (p = yytext; !isalpha(p[0]); p++); /* skip " # " */ |
| 154 | p += 5; /* skip "error" */ |
| 155 | glcpp_error(yylloc, yyextra, "#error%s", p); |
| 156 | } |
| 157 | |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 158 | {HASH}define{HSPACE}+/{IDENTIFIER}"(" { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 159 | yyextra->space_tokens = 0; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 160 | return HASH_DEFINE_FUNC; |
Carl Worth | b20d33c | 2010-05-20 22:27:07 -0700 | [diff] [blame] | 161 | } |
| 162 | |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 163 | {HASH}define { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 164 | yyextra->space_tokens = 0; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 165 | return HASH_DEFINE_OBJ; |
Carl Worth | b20d33c | 2010-05-20 22:27:07 -0700 | [diff] [blame] | 166 | } |
| 167 | |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 168 | {HASH}undef { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 169 | yyextra->space_tokens = 0; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 170 | return HASH_UNDEF; |
Carl Worth | 03f6d5d | 2010-05-24 11:29:02 -0700 | [diff] [blame] | 171 | } |
| 172 | |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 173 | {HASH} { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 174 | yyextra->space_tokens = 0; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 175 | return HASH; |
Carl Worth | 81f0143 | 2010-05-14 17:08:45 -0700 | [diff] [blame] | 176 | } |
| 177 | |
Carl Worth | 8fed1cd | 2010-05-26 09:32:12 -0700 | [diff] [blame] | 178 | {DECIMAL_INTEGER} { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 179 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Carl Worth | 050e3de | 2010-05-27 14:36:29 -0700 | [diff] [blame] | 180 | return INTEGER_STRING; |
Carl Worth | 8fed1cd | 2010-05-26 09:32:12 -0700 | [diff] [blame] | 181 | } |
| 182 | |
| 183 | {OCTAL_INTEGER} { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 184 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Carl Worth | 050e3de | 2010-05-27 14:36:29 -0700 | [diff] [blame] | 185 | return INTEGER_STRING; |
Carl Worth | 8fed1cd | 2010-05-26 09:32:12 -0700 | [diff] [blame] | 186 | } |
| 187 | |
| 188 | {HEXADECIMAL_INTEGER} { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 189 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Carl Worth | 050e3de | 2010-05-27 14:36:29 -0700 | [diff] [blame] | 190 | return INTEGER_STRING; |
Carl Worth | 8fed1cd | 2010-05-26 09:32:12 -0700 | [diff] [blame] | 191 | } |
| 192 | |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 193 | "<<" { |
| 194 | return LEFT_SHIFT; |
Carl Worth | b1854fd | 2010-05-25 16:28:26 -0700 | [diff] [blame] | 195 | } |
| 196 | |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 197 | ">>" { |
| 198 | return RIGHT_SHIFT; |
Carl Worth | b1854fd | 2010-05-25 16:28:26 -0700 | [diff] [blame] | 199 | } |
| 200 | |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 201 | "<=" { |
| 202 | return LESS_OR_EQUAL; |
| 203 | } |
| 204 | |
| 205 | ">=" { |
| 206 | return GREATER_OR_EQUAL; |
| 207 | } |
| 208 | |
| 209 | "==" { |
| 210 | return EQUAL; |
| 211 | } |
| 212 | |
| 213 | "!=" { |
| 214 | return NOT_EQUAL; |
| 215 | } |
| 216 | |
| 217 | "&&" { |
| 218 | return AND; |
| 219 | } |
| 220 | |
| 221 | "||" { |
| 222 | return OR; |
| 223 | } |
| 224 | |
| 225 | "##" { |
| 226 | return PASTE; |
| 227 | } |
| 228 | |
Carl Worth | 8fed1cd | 2010-05-26 09:32:12 -0700 | [diff] [blame] | 229 | "defined" { |
| 230 | return DEFINED; |
| 231 | } |
| 232 | |
Carl Worth | 16c1e98 | 2010-05-26 09:35:34 -0700 | [diff] [blame] | 233 | {IDENTIFIER} { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 234 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Carl Worth | 16c1e98 | 2010-05-26 09:35:34 -0700 | [diff] [blame] | 235 | return IDENTIFIER; |
| 236 | } |
| 237 | |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 238 | {PUNCTUATION} { |
| 239 | return yytext[0]; |
Carl Worth | b1854fd | 2010-05-25 16:28:26 -0700 | [diff] [blame] | 240 | } |
| 241 | |
Carl Worth | 9fb8b7a | 2010-05-25 15:04:32 -0700 | [diff] [blame] | 242 | {OTHER}+ { |
Kenneth Graunke | e0e429f | 2010-06-16 16:26:28 -0700 | [diff] [blame] | 243 | yylval->str = xtalloc_strdup (yyextra, yytext); |
Carl Worth | 9fb8b7a | 2010-05-25 15:04:32 -0700 | [diff] [blame] | 244 | return OTHER; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 245 | } |
| 246 | |
Carl Worth | 9fb8b7a | 2010-05-25 15:04:32 -0700 | [diff] [blame] | 247 | {HSPACE}+ { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 248 | if (yyextra->space_tokens) { |
Carl Worth | f34a000 | 2010-05-25 16:59:02 -0700 | [diff] [blame] | 249 | return SPACE; |
| 250 | } |
Carl Worth | 0a93cbb | 2010-05-13 10:29:07 -0700 | [diff] [blame] | 251 | } |
Carl Worth | fcbbb46 | 2010-05-13 09:36:23 -0700 | [diff] [blame] | 252 | |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 253 | \n { |
Carl Worth | a771a40 | 2010-06-01 11:20:18 -0700 | [diff] [blame] | 254 | yyextra->lexing_if = 0; |
Kenneth Graunke | db93810 | 2010-06-16 17:41:12 -0700 | [diff] [blame] | 255 | yylineno++; |
| 256 | yycolumn = 0; |
Carl Worth | 3ff8167 | 2010-05-25 13:09:03 -0700 | [diff] [blame] | 257 | return NEWLINE; |
Carl Worth | 33cc400 | 2010-05-12 12:17:10 -0700 | [diff] [blame] | 258 | } |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 259 | |
Kenneth Graunke | f82d673 | 2010-06-16 12:53:19 -0700 | [diff] [blame] | 260 | /* Handle missing newline at EOF. */ |
| 261 | <INITIAL><<EOF>> { |
| 262 | BEGIN DONE; /* Don't keep matching this rule forever. */ |
| 263 | yyextra->lexing_if = 0; |
| 264 | return NEWLINE; |
| 265 | } |
| 266 | |
Carl Worth | 3a37b87 | 2010-05-10 11:44:09 -0700 | [diff] [blame] | 267 | %% |
Kenneth Graunke | 1b1f43e | 2010-06-16 12:01:17 -0700 | [diff] [blame] | 268 | |
| 269 | void |
| 270 | glcpp_lex_set_source_string(glcpp_parser_t *parser, const char *shader) |
| 271 | { |
| 272 | yy_scan_string(shader, parser->scanner); |
| 273 | } |