| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | /* | 
					
						
							|  |  |  |  * HLSL preprocessor | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2020 Zebediah Figura for CodeWeavers | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with this library; if not, write to the Free Software | 
					
						
							|  |  |  |  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "preproc.tab.h" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 15:09:51 +01:00
										 |  |  | #undef ERROR  /* defined in wingdi.h */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | #define YYSTYPE PREPROC_YYSTYPE | 
					
						
							|  |  |  | #define YYLTYPE PREPROC_YYLTYPE | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define YY_DECL static int preproc_lexer_lex(YYSTYPE *yylval_param, YYLTYPE *yylloc_param, yyscan_t yyscanner) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  | static struct preproc_expansion *preproc_get_top_expansion(struct preproc_ctx *ctx) | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!ctx->expansion_count) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |     return &ctx->expansion_stack[ctx->expansion_count - 1]; | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  | static void update_location(struct preproc_ctx *ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define YY_USER_ACTION update_location(yyget_extra(yyscanner)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | %} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %option 8bit | 
					
						
							|  |  |  | %option bison-bridge | 
					
						
							|  |  |  | %option bison-locations | 
					
						
							|  |  |  | %option extra-type="struct preproc_ctx *" | 
					
						
							|  |  |  | %option never-interactive | 
					
						
							| 
									
										
										
										
											2023-05-08 12:57:22 -05:00
										 |  |  | %option nodefault | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | %option noinput | 
					
						
							|  |  |  | %option nounput | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  | %option noyy_top_state | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | %option noyywrap | 
					
						
							|  |  |  | %option prefix="preproc_yy" | 
					
						
							|  |  |  | %option reentrant | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  | %option stack | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Because these can both be terminated by EOF, we need states for them. */ | 
					
						
							|  |  |  | %s C_COMMENT | 
					
						
							|  |  |  | %s CXX_COMMENT | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:50 -06:00
										 |  |  | %s ERROR | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | %s INCLUDE | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:56 -06:00
										 |  |  | %s LINE | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | NEWLINE         \r?\n | 
					
						
							| 
									
										
										
										
											2024-05-22 14:07:00 -03:00
										 |  |  | WS              [ \t\r] | 
					
						
							| 
									
										
										
										
											2024-09-20 18:35:38 +03:00
										 |  |  | IDENTIFIER      (::)?[A-Za-z_]((::)?[A-Za-z0-9_]+)* | 
					
						
							| 
									
										
										
										
											2022-02-02 13:46:16 +01:00
										 |  |  | INT_SUFFIX      [uUlL]{0,2} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | %% | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-03 14:02:46 -05:00
										 |  |  | <INITIAL,INCLUDE,LINE>"//"          {yy_push_state(CXX_COMMENT, yyscanner);} | 
					
						
							|  |  |  | <INITIAL,INCLUDE,LINE>"/*"          {yy_push_state(C_COMMENT, yyscanner);} | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | <CXX_COMMENT>\\{NEWLINE}            {} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  | <CXX_COMMENT>\n                     { | 
					
						
							|  |  |  |         yy_pop_state(yyscanner); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |         BEGIN(INITIAL); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         return T_NEWLINE; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | <C_COMMENT>"*/"                     {yy_pop_state(yyscanner);} | 
					
						
							| 
									
										
										
										
											2025-06-03 14:05:30 -05:00
										 |  |  | <C_COMMENT,CXX_COMMENT><<EOF>>      { | 
					
						
							|  |  |  |         yy_pop_state(yyscanner); | 
					
						
							|  |  |  |         BEGIN(INITIAL); | 
					
						
							|  |  |  |         yyterminate(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  | <C_COMMENT,CXX_COMMENT>.            {} | 
					
						
							| 
									
										
										
										
											2023-05-08 13:06:31 -05:00
										 |  |  | <C_COMMENT>\n                       {} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:31 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:50 -06:00
										 |  |  | <ERROR>(\\{NEWLINE}|[^\n])*         {return T_STRING;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:51 -06:00
										 |  |  | <INITIAL>defined/\(                 {return T_DEFINED;} | 
					
						
							|  |  |  | <INITIAL>defined                    {return T_DEFINED;} | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:18 -06:00
										 |  |  | <INITIAL>{IDENTIFIER}/\(            {return T_IDENTIFIER_PAREN;} | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  | <INITIAL>{IDENTIFIER}               {return T_IDENTIFIER;} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-19 11:18:21 -06:00
										 |  |  | <INITIAL>"<="                       {return T_LE;} | 
					
						
							|  |  |  | <INITIAL>">="                       {return T_GE;} | 
					
						
							| 
									
										
										
										
											2021-01-21 16:10:43 -06:00
										 |  |  | <INITIAL>"=="                       {return T_EQ;} | 
					
						
							|  |  |  | <INITIAL>"!="                       {return T_NE;} | 
					
						
							| 
									
										
										
										
											2021-01-21 16:10:45 -06:00
										 |  |  | <INITIAL>"&&"                       {return T_AND;} | 
					
						
							|  |  |  | <INITIAL>"||"                       {return T_OR;} | 
					
						
							| 
									
										
										
										
											2021-01-19 11:18:21 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |     /* We have no use for floats, but shouldn't parse them as integers. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  | <INITIAL>[0-9]*\.[0-9]+([eE][+-]?[0-9]+)?[hHfF]?    {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>[0-9]+\.([eE][+-]?[0-9]+)?[hHfF]?          {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>[0-9]+([eE][+-]?[0-9]+)?[hHfF]             {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>[0-9]+[eE][+-]?[0-9]+                      {return T_TEXT;} | 
					
						
							| 
									
										
										
										
											2022-02-02 13:46:16 +01:00
										 |  |  | <INITIAL,LINE>0[xX][0-9a-fA-f]+{INT_SUFFIX}         {return T_INTEGER;} | 
					
						
							|  |  |  | <INITIAL,LINE>0[0-7]*{INT_SUFFIX}                   {return T_INTEGER;} | 
					
						
							|  |  |  | <INITIAL,LINE>[1-9][0-9]*{INT_SUFFIX}               {return T_INTEGER;} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:20 -06:00
										 |  |  | <INITIAL>##                         {return T_CONCAT;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  | <INITIAL>"++"                       {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>"--"                       {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>"<<"=?                     {return T_TEXT;} | 
					
						
							|  |  |  | <INITIAL>">>"=?                     {return T_TEXT;} | 
					
						
							| 
									
										
										
										
											2021-01-21 16:10:43 -06:00
										 |  |  | <INITIAL>[-+*/%&|^]=                {return T_TEXT;} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:58 -06:00
										 |  |  |     /* Native doesn't preserve these tokens when running the preprocessor on its | 
					
						
							|  |  |  |      * own, but there's no good reason to emulate that difference yet. */ | 
					
						
							|  |  |  | <INITIAL>[pv]s\.[123]\.[0-4x]       {return T_TEXT;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:56 -06:00
										 |  |  | <INCLUDE,LINE>\"[^"]*\"             {return T_STRING;} | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | <INCLUDE>\<[^>]*\>                  {return T_STRING;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:32 -06:00
										 |  |  |     /* C strings (including escaped quotes). */ | 
					
						
							|  |  |  | <INITIAL>\"([^"\\]|\\.)*\"          {return T_TEXT;} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | <INITIAL>#{WS}*{IDENTIFIER}         { | 
					
						
							|  |  |  |         struct preproc_ctx *ctx = yyget_extra(yyscanner); | 
					
						
							|  |  |  |         const char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!ctx->last_was_newline) | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |             struct preproc_expansion *exp; | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             /* Stringification is only done for function-like macro bodies. | 
					
						
							|  |  |  |              * Anywhere else, we need to parse it as two separate tokens. | 
					
						
							|  |  |  |              * We could use a state for this, but yyless() is easier and cheap. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |             if ((exp = preproc_get_top_expansion(ctx)) && exp->macro && exp->macro->arg_count) | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  |                 return T_HASHSTRING; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             yyless(1); | 
					
						
							|  |  |  |             return T_TEXT; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (p = yytext + 1; strchr(" \t", *p); ++p) | 
					
						
							|  |  |  |             ; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:50 -06:00
										 |  |  |         if (!strcmp(p, "error")) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             BEGIN(ERROR); | 
					
						
							|  |  |  |             return T_ERROR; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |         if (!strcmp(p, "include")) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             BEGIN(INCLUDE); | 
					
						
							|  |  |  |             return T_INCLUDE; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:56 -06:00
										 |  |  |         if (!strcmp(p, "line")) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             BEGIN(LINE); | 
					
						
							|  |  |  |             return T_LINE; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |         if (!strcmp(p, "define")) | 
					
						
							|  |  |  |             return T_DEFINE; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:23 -06:00
										 |  |  |         if (!strcmp(p, "elif")) | 
					
						
							|  |  |  |             return T_ELIF; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:22 -06:00
										 |  |  |         if (!strcmp(p, "else")) | 
					
						
							|  |  |  |             return T_ELSE; | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         if (!strcmp(p, "endif")) | 
					
						
							|  |  |  |             return T_ENDIF; | 
					
						
							|  |  |  |         if (!strcmp(p, "if")) | 
					
						
							|  |  |  |             return T_IF; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:07 -06:00
										 |  |  |         if (!strcmp(p, "ifdef")) | 
					
						
							|  |  |  |             return T_IFDEF; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:08 -06:00
										 |  |  |         if (!strcmp(p, "ifndef")) | 
					
						
							|  |  |  |             return T_IFNDEF; | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:49 -06:00
										 |  |  |         if (!strcmp(p, "pragma")) | 
					
						
							|  |  |  |             return T_PRAGMA; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:10 -06:00
										 |  |  |         if (!strcmp(p, "undef")) | 
					
						
							|  |  |  |             return T_UNDEF; | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         preproc_warning(ctx, yyget_lloc(yyscanner), VKD3D_SHADER_WARNING_PP_UNKNOWN_DIRECTIVE, | 
					
						
							|  |  |  |                 "Ignoring unknown directive \"%s\".", yytext); | 
					
						
							|  |  |  |         return T_TEXT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:56 -06:00
										 |  |  | <INITIAL,INCLUDE,LINE>\\{NEWLINE}       {} | 
					
						
							|  |  |  | <INITIAL,INCLUDE,ERROR,LINE>{NEWLINE}   { | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |         BEGIN(INITIAL); | 
					
						
							|  |  |  |         return T_NEWLINE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-03 14:05:30 -05:00
										 |  |  | <INITIAL,INCLUDE,ERROR,LINE><<EOF>> { | 
					
						
							|  |  |  |         BEGIN(INITIAL); | 
					
						
							|  |  |  |         yyterminate(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-08 12:52:10 -05:00
										 |  |  | <INITIAL,INCLUDE,LINE>{WS}+         {} | 
					
						
							| 
									
										
										
										
											2021-01-21 16:10:46 -06:00
										 |  |  | <INITIAL>[-()\[\]{},+!*/<>&|^?:]    {return yytext[0];} | 
					
						
							| 
									
										
										
										
											2023-05-08 12:52:10 -05:00
										 |  |  | <INITIAL,INCLUDE,LINE>.             {return T_TEXT;} | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | %% | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  | static void update_location(struct preproc_ctx *ctx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     struct preproc_buffer *buffer = &preproc_get_top_file(ctx)->buffer; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  |     unsigned int i, leng = yyget_leng(ctx->scanner); | 
					
						
							|  |  |  |     const char *text = yyget_text(ctx->scanner); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* We want to do this here, rather than before calling yylex(), because | 
					
						
							|  |  |  |      * some tokens are skipped by the lexer. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     *yyget_lloc(ctx->scanner) = buffer->location; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < leng; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |         ++buffer->location.column; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  |         if (text[i] == '\n') | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |             buffer->location.column = 1; | 
					
						
							|  |  |  |             ++buffer->location.line; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | static bool preproc_is_writing(struct preproc_ctx *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct preproc_file *file; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* This can happen while checking for unterminated macro invocation. */ | 
					
						
							|  |  |  |     if (!ctx->file_count) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     file = preproc_get_top_file(ctx); | 
					
						
							|  |  |  |     if (!file->if_count) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     return file->if_stack[file->if_count - 1].current_true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:20 -06:00
										 |  |  | /* Concatenation is not done for object-like macros, but is done for both | 
					
						
							|  |  |  |  * function-like macro bodies and their arguments. */ | 
					
						
							|  |  |  | static bool should_concat(struct preproc_ctx *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct preproc_macro *macro; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ctx->expansion_count) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     macro = ctx->expansion_stack[ctx->expansion_count - 1].macro; | 
					
						
							|  |  |  |     return !macro || macro->arg_count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  | static struct preproc_buffer *preproc_get_top_buffer(struct preproc_ctx *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ctx->expansion_count) | 
					
						
							|  |  |  |         return &ctx->expansion_stack[ctx->expansion_count - 1].buffer; | 
					
						
							|  |  |  |     if (ctx->file_count) | 
					
						
							|  |  |  |         return &ctx->file_stack[ctx->file_count - 1].buffer; | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | static void preproc_pop_buffer(struct preproc_ctx *ctx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |     struct preproc_buffer *buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |     if (ctx->expansion_count) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         struct preproc_expansion *exp = &ctx->expansion_stack[ctx->expansion_count - 1]; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         yy_delete_buffer(exp->buffer.lexer_buffer, ctx->scanner); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |         if (exp->macro) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             for (unsigned int i = 0; i < exp->macro->arg_count; ++i) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_cleanup(&exp->arg_values[i].text); | 
					
						
							|  |  |  |             free(exp->arg_values); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         --ctx->expansion_count; | 
					
						
							|  |  |  |         TRACE("Expansion stack size is now %zu.\n", ctx->expansion_count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         struct preproc_file *file = preproc_get_top_file(ctx); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         if (ctx->file_count > 1) | 
					
						
							|  |  |  |             preproc_close_include(ctx, &file->code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (file->if_count) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const struct vkd3d_shader_location loc = {.source_name = file->filename}; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |             preproc_warning(ctx, &loc, VKD3D_SHADER_WARNING_PP_UNTERMINATED_IF, "Unterminated #if block."); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         vkd3d_free(file->if_stack); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         vkd3d_free(file->filename); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         yy_delete_buffer(file->buffer.lexer_buffer, ctx->scanner); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         --ctx->file_count; | 
					
						
							|  |  |  |         TRACE("File stack size is now %zu.\n", ctx->file_count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |     if ((buffer = preproc_get_top_buffer(ctx))) | 
					
						
							|  |  |  |         yy_switch_to_buffer(buffer->lexer_buffer, ctx->scanner); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | static int return_token(int token, YYSTYPE *lval, const char *text) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (token) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:21 -06:00
										 |  |  |         case T_HASHSTRING: | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |         case T_IDENTIFIER: | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:18 -06:00
										 |  |  |         case T_IDENTIFIER_PAREN: | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         case T_INTEGER: | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |         case T_STRING: | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         case T_TEXT: | 
					
						
							|  |  |  |             if (!(lval->string = vkd3d_strdup(text))) | 
					
						
							|  |  |  |                 return 0; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return token; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | static const struct preproc_text *find_arg_expansion(struct preproc_ctx *ctx, const char *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |     struct preproc_expansion *exp; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |     if ((exp = preproc_get_top_expansion(ctx)) && exp->macro) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |         for (i = 0; i < exp->macro->arg_count; ++i) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |             if (!strcmp(s, exp->macro->arg_names[i])) | 
					
						
							|  |  |  |                 return &exp->arg_values[i]; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void preproc_text_add(struct preproc_text *text, const char *string) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&text->text, "%s", string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool preproc_push_expansion(struct preproc_ctx *ctx, | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |         const struct preproc_text *text, struct preproc_macro *macro, struct preproc_text *arg_values) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  | { | 
					
						
							|  |  |  |     struct preproc_expansion *exp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!vkd3d_array_reserve((void **)&ctx->expansion_stack, &ctx->expansion_stack_size, | 
					
						
							|  |  |  |             ctx->expansion_count + 1, sizeof(*ctx->expansion_stack))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     exp = &ctx->expansion_stack[ctx->expansion_count++]; | 
					
						
							|  |  |  |     exp->text = text; | 
					
						
							|  |  |  |     exp->buffer.lexer_buffer = yy_scan_bytes(text->text.buffer, text->text.content_size, ctx->scanner); | 
					
						
							|  |  |  |     exp->buffer.location = text->location; | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |     exp->buffer.eof = false; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |     exp->macro = macro; | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |     exp->arg_values = arg_values; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |     TRACE("Expansion stack size is now %zu.\n", ctx->expansion_count); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-29 18:14:23 -05:00
										 |  |  | static void preproc_stringify(struct preproc_ctx *ctx, struct vkd3d_string_buffer *buffer, const char *text) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct preproc_text *expansion; | 
					
						
							|  |  |  |     const char *p = text + 1; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (*p == ' ' || *p == '\t') | 
					
						
							|  |  |  |         ++p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(buffer, "\""); | 
					
						
							|  |  |  |     if ((expansion = find_arg_expansion(ctx, p))) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-07-29 18:32:04 -05:00
										 |  |  |         size_t len = expansion->text.content_size; | 
					
						
							|  |  |  |         size_t start = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (len && strchr(" \t\r\n", expansion->text.buffer[len - 1])) | 
					
						
							|  |  |  |             --len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (start < len && strchr(" \t\r\n", expansion->text.buffer[start])) | 
					
						
							|  |  |  |             ++start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (i = start; i < len; ++i) | 
					
						
							| 
									
										
										
										
											2023-07-29 18:14:23 -05:00
										 |  |  |         { | 
					
						
							|  |  |  |             char c = expansion->text.buffer[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (c == '\\' || c == '"') | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(buffer, "\\"); | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(buffer, "%c", c); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(buffer, "%s", p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(buffer, "\""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | int yylex(YYSTYPE *lval, YYLTYPE *lloc, yyscan_t scanner) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct preproc_ctx *ctx = yyget_extra(scanner); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (;;) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         struct preproc_func_state *func_state; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |         const char *text; | 
					
						
							|  |  |  |         int token; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         if (ctx->lookahead_token) | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |             token = ctx->lookahead_token; | 
					
						
							|  |  |  |             text = yyget_text(scanner); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |             if (preproc_get_top_buffer(ctx)->eof) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |             { | 
					
						
							|  |  |  |                 preproc_pop_buffer(ctx); | 
					
						
							|  |  |  |                 if (!ctx->file_count) | 
					
						
							|  |  |  |                     return 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 10:34:34 +02:00
										 |  |  |             VKD3D_ASSERT(ctx->file_count); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |             if (!(token = preproc_lexer_lex(lval, lloc, scanner))) | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |                 preproc_get_top_buffer(ctx)->eof = true; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 /* If we have reached the end of an included file, inject a newline. */ | 
					
						
							|  |  |  |                 if (ctx->expansion_count) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 token = T_NEWLINE; | 
					
						
							|  |  |  |                 text = "\n"; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 text = yyget_text(scanner); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (ctx->last_was_newline) | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 switch (token) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     case T_DEFINE: | 
					
						
							|  |  |  |                     case T_ELIF: | 
					
						
							|  |  |  |                     case T_ELSE: | 
					
						
							|  |  |  |                     case T_ENDIF: | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:50 -06:00
										 |  |  |                     case T_ERROR: | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                     case T_IF: | 
					
						
							|  |  |  |                     case T_IFDEF: | 
					
						
							|  |  |  |                     case T_IFNDEF: | 
					
						
							|  |  |  |                     case T_INCLUDE: | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:56 -06:00
										 |  |  |                     case T_LINE: | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:49 -06:00
										 |  |  |                     case T_PRAGMA: | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                     case T_UNDEF: | 
					
						
							|  |  |  |                         ctx->current_directive = token; | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     default: | 
					
						
							|  |  |  |                         ctx->current_directive = 0; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |             ctx->last_was_newline = (token == T_NEWLINE); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:51 -06:00
										 |  |  |         if (ctx->current_directive && token == T_DEFINED) | 
					
						
							|  |  |  |             ctx->last_was_defined = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         func_state = ctx->current_directive ? &ctx->directive_func : &ctx->text_func; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         TRACE("Parsing token %d%s, line %d, in directive %d, state %#x, string %s.\n", | 
					
						
							|  |  |  |                 token, ctx->lookahead_token ? " (lookahead)" : "", lloc->line, | 
					
						
							|  |  |  |                 ctx->current_directive, func_state->state, debugstr_a(text)); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         ctx->lookahead_token = 0; | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |         switch (ctx->current_directive) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case T_ELIF: | 
					
						
							|  |  |  |             case T_ELSE: | 
					
						
							|  |  |  |             case T_ENDIF: | 
					
						
							|  |  |  |             case T_IF: | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:07 -06:00
										 |  |  |             case T_IFDEF: | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:08 -06:00
										 |  |  |             case T_IFNDEF: | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 if (!preproc_is_writing(ctx)) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:49 -06:00
										 |  |  |         if (ctx->current_directive == T_PRAGMA) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             /* Print all tokens verbatim. */ | 
					
						
							|  |  |  |             if (token == T_PRAGMA) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&ctx->buffer, "#pragma "); | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&ctx->buffer, "%s", text); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |         switch (func_state->state) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |             case STATE_NONE: | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:20 -06:00
										 |  |  |                 if (token == T_CONCAT && should_concat(ctx)) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     while (ctx->buffer.content_size | 
					
						
							|  |  |  |                             && strchr(" \t\r\n", ctx->buffer.buffer[ctx->buffer.content_size - 1])) | 
					
						
							|  |  |  |                         --ctx->buffer.content_size; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-29 18:08:29 -05:00
										 |  |  |                 if (token == T_HASHSTRING) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:21 -06:00
										 |  |  |                 { | 
					
						
							| 
									
										
										
										
											2023-07-29 18:14:23 -05:00
										 |  |  |                     struct vkd3d_string_buffer buffer; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:21 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     if (ctx->current_directive) | 
					
						
							|  |  |  |                         return return_token(token, lval, text); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-29 18:14:23 -05:00
										 |  |  |                     vkd3d_string_buffer_init(&buffer); | 
					
						
							|  |  |  |                     preproc_stringify(ctx, &buffer, text); | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&ctx->buffer, "%s", buffer.buffer); | 
					
						
							|  |  |  |                     vkd3d_string_buffer_cleanup(&buffer); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:21 -06:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 if (token == T_IDENTIFIER || token == T_IDENTIFIER_PAREN) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     const struct preproc_text *expansion; | 
					
						
							|  |  |  |                     struct preproc_macro *macro; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     switch (ctx->current_directive) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         case T_DEFINE: | 
					
						
							|  |  |  |                         case T_IFDEF: | 
					
						
							|  |  |  |                         case T_IFNDEF: | 
					
						
							|  |  |  |                         case T_UNDEF: | 
					
						
							|  |  |  |                             /* Return identifiers verbatim. */ | 
					
						
							|  |  |  |                             return return_token(token, lval, text); | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:51 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         case T_IF: | 
					
						
							|  |  |  |                         case T_ELIF: | 
					
						
							|  |  |  |                             /* Return identifiers verbatim only if they're the | 
					
						
							|  |  |  |                              * argument to "defined". */ | 
					
						
							|  |  |  |                             if (ctx->last_was_defined) | 
					
						
							|  |  |  |                             { | 
					
						
							|  |  |  |                                 ctx->last_was_defined = false; | 
					
						
							|  |  |  |                                 return return_token(token, lval, text); | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                             break; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     /* Otherwise, expand a macro if there is one. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if ((expansion = find_arg_expansion(ctx, text))) | 
					
						
							|  |  |  |                     { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                         preproc_push_expansion(ctx, expansion, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if ((macro = preproc_find_macro(ctx, text))) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         if (!macro->arg_count) | 
					
						
							|  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                             preproc_push_expansion(ctx, ¯o->body, macro, NULL); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                         } | 
					
						
							|  |  |  |                         else | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             func_state->state = STATE_IDENTIFIER; | 
					
						
							|  |  |  |                             func_state->macro = macro; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:47 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:48 -06:00
										 |  |  |                     if (!strcmp(text, "__FILE__")) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         const struct preproc_file *file = preproc_get_top_file(ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         /* Not the current file name, but rather the file name | 
					
						
							|  |  |  |                          * before invoking any macros. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (ctx->current_directive) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             char *string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             if (!(string = vkd3d_malloc(strlen(file->filename) + 3))) | 
					
						
							|  |  |  |                                 return 0; | 
					
						
							|  |  |  |                             sprintf(string, "\"%s\"", file->filename); | 
					
						
							|  |  |  |                             lval->string = string; | 
					
						
							|  |  |  |                             return T_STRING; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (preproc_is_writing(ctx)) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&ctx->buffer, "\"%s\" ", file->filename); | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-14 14:47:47 -06:00
										 |  |  |                     if (!strcmp(text, "__LINE__")) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         const struct preproc_file *file = preproc_get_top_file(ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         /* Not the current line number, but rather the line | 
					
						
							|  |  |  |                          * number before invoking any macros. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (ctx->current_directive) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             char string[13]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             sprintf(string, "%d", file->buffer.location.line); | 
					
						
							|  |  |  |                             return return_token(T_INTEGER, lval, string); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (preproc_is_writing(ctx)) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&ctx->buffer, "%d ", file->buffer.location.line); | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (ctx->current_directive) | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |                     return return_token(token, lval, text); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-30 23:38:32 +02:00
										 |  |  |                 if (isspace(text[0])) | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&ctx->buffer, "%s", text); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&ctx->buffer, "%s ", text); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |             case STATE_IDENTIFIER: | 
					
						
							|  |  |  |                 if (token == '(') | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                     struct preproc_text *arg_values; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (!(arg_values = calloc(func_state->macro->arg_count, sizeof(*arg_values)))) | 
					
						
							|  |  |  |                         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     for (unsigned int i = 0; i < func_state->macro->arg_count; ++i) | 
					
						
							|  |  |  |                         vkd3d_string_buffer_init(&arg_values[i].text); | 
					
						
							|  |  |  |                     arg_values[0].location = *lloc; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     func_state->arg_count = 0; | 
					
						
							|  |  |  |                     func_state->paren_depth = 1; | 
					
						
							|  |  |  |                     func_state->state = STATE_ARGS; | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                     func_state->arg_values = arg_values; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     const char *name = func_state->macro->name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     ctx->lookahead_token = token; | 
					
						
							|  |  |  |                     func_state->macro = NULL; | 
					
						
							|  |  |  |                     func_state->state = STATE_NONE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (ctx->current_directive) | 
					
						
							|  |  |  |                         return return_token(T_IDENTIFIER, lval, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&ctx->buffer, "%s ", name); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             case STATE_ARGS: | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 struct preproc_text *current_arg = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 10:34:34 +02:00
										 |  |  |                 VKD3D_ASSERT(func_state->macro->arg_count); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (func_state->arg_count < func_state->macro->arg_count) | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                     current_arg = &func_state->arg_values[func_state->arg_count]; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 switch (token) | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2023-07-29 18:13:41 -05:00
										 |  |  |                     /* Most text gets left alone (e.g. if it contains macros, | 
					
						
							|  |  |  |                      * the macros should be evaluated later). | 
					
						
							|  |  |  |                      * Arguments are a special case, and are replaced with | 
					
						
							|  |  |  |                      * their values immediately. */ | 
					
						
							|  |  |  |                     case T_IDENTIFIER: | 
					
						
							|  |  |  |                     case T_IDENTIFIER_PAREN: | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         const struct preproc_text *expansion; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if ((expansion = find_arg_expansion(ctx, text))) | 
					
						
							|  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                             preproc_push_expansion(ctx, expansion, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2023-07-29 18:13:41 -05:00
										 |  |  |                             continue; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (current_arg) | 
					
						
							|  |  |  |                             preproc_text_add(current_arg, text); | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-29 18:15:14 -05:00
										 |  |  |                     /* Stringification is another special case. Unsurprisingly, | 
					
						
							|  |  |  |                      * we need to stringify if this is an argument. More | 
					
						
							|  |  |  |                      * surprisingly, we need to stringify even if it's not. */ | 
					
						
							|  |  |  |                     case T_HASHSTRING: | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         struct vkd3d_string_buffer buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         vkd3d_string_buffer_init(&buffer); | 
					
						
							|  |  |  |                         preproc_stringify(ctx, &buffer, text); | 
					
						
							|  |  |  |                         if (current_arg) | 
					
						
							|  |  |  |                             preproc_text_add(current_arg, buffer.buffer); | 
					
						
							|  |  |  |                         vkd3d_string_buffer_cleanup(&buffer); | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                     case T_NEWLINE: | 
					
						
							|  |  |  |                         if (current_arg) | 
					
						
							|  |  |  |                             preproc_text_add(current_arg, " "); | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     case ')': | 
					
						
							|  |  |  |                     case ']': | 
					
						
							|  |  |  |                     case '}': | 
					
						
							|  |  |  |                         if (!--func_state->paren_depth) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             if (++func_state->arg_count == func_state->macro->arg_count) | 
					
						
							|  |  |  |                             { | 
					
						
							| 
									
										
										
										
											2024-09-20 15:41:35 -05:00
										 |  |  |                                 preproc_push_expansion(ctx, &func_state->macro->body, | 
					
						
							|  |  |  |                                         func_state->macro, func_state->arg_values); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                             } | 
					
						
							|  |  |  |                             else | 
					
						
							|  |  |  |                             { | 
					
						
							|  |  |  |                                 preproc_warning(ctx, lloc, VKD3D_SHADER_WARNING_PP_ARGUMENT_COUNT_MISMATCH, | 
					
						
							|  |  |  |                                         "Wrong number of arguments to macro \"%s\": expected %zu, got %zu.", | 
					
						
							|  |  |  |                                         func_state->macro->name, func_state->macro->arg_count, func_state->arg_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                                 if (ctx->current_directive) | 
					
						
							|  |  |  |                                     return return_token(T_IDENTIFIER, lval, func_state->macro->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                                 vkd3d_string_buffer_printf(&ctx->buffer, "%s ", func_state->macro->name); | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  |                             func_state->macro = NULL; | 
					
						
							|  |  |  |                             func_state->state = STATE_NONE; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         else | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             if (current_arg) | 
					
						
							|  |  |  |                                 preproc_text_add(current_arg, text); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     case ',': | 
					
						
							|  |  |  |                         if (func_state->paren_depth == 1) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             ++func_state->arg_count; | 
					
						
							|  |  |  |                             if (current_arg) | 
					
						
							|  |  |  |                                 current_arg->location = *lloc; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         else if (current_arg) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             preproc_text_add(current_arg, text); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     case '(': | 
					
						
							|  |  |  |                     case '[': | 
					
						
							|  |  |  |                     case '{': | 
					
						
							|  |  |  |                         ++func_state->paren_depth; | 
					
						
							|  |  |  |                         /* fall through */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     default: | 
					
						
							|  |  |  |                         if (current_arg) | 
					
						
							|  |  |  |                             preproc_text_add(current_arg, text); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2023-07-29 18:17:35 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (current_arg) | 
					
						
							|  |  |  |                     preproc_text_add(current_arg, " "); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | bool preproc_push_include(struct preproc_ctx *ctx, char *filename, const struct vkd3d_shader_code *code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct preproc_file *file; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!vkd3d_array_reserve((void **)&ctx->file_stack, &ctx->file_stack_size, | 
					
						
							|  |  |  |             ctx->file_count + 1, sizeof(*ctx->file_stack))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     file = &ctx->file_stack[ctx->file_count++]; | 
					
						
							|  |  |  |     memset(file, 0, sizeof(*file)); | 
					
						
							|  |  |  |     file->code = *code; | 
					
						
							|  |  |  |     file->filename = filename; | 
					
						
							|  |  |  |     file->buffer.lexer_buffer = yy_scan_bytes(code->code, code->size, ctx->scanner); | 
					
						
							|  |  |  |     file->buffer.location.source_name = file->filename; | 
					
						
							|  |  |  |     file->buffer.location.line = 1; | 
					
						
							|  |  |  |     file->buffer.location.column = 1; | 
					
						
							| 
									
										
										
										
											2025-06-03 14:07:54 -05:00
										 |  |  |     file->buffer.eof = false; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     TRACE("File stack size is now %zu.\n", ctx->file_count); | 
					
						
							|  |  |  |     ctx->last_was_newline = true; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  | static int preproc_macro_compare(const void *key, const struct rb_entry *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct preproc_macro *macro = RB_ENTRY_VALUE(entry, struct preproc_macro, entry); | 
					
						
							|  |  |  |     const char *name = key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return strcmp(name, macro->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void preproc_macro_rb_free(struct rb_entry *entry, void *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     preproc_free_macro(RB_ENTRY_VALUE(entry, struct preproc_macro, entry)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | int preproc_lexer_parse(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |         struct vkd3d_shader_code *out, struct vkd3d_shader_message_context *message_context) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     static const struct vkd3d_shader_preprocess_info default_preprocess_info = {0}; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |     struct preproc_ctx ctx = {0}; | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |     char *source_name = NULL; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |     void *output_code; | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |     unsigned int i; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_init(&ctx.buffer); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |     rb_init(&ctx.macros, preproc_macro_compare); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     if (!(ctx.preprocess_info = vkd3d_find_struct(compile_info->next, PREPROCESS_INFO))) | 
					
						
							|  |  |  |         ctx.preprocess_info = &default_preprocess_info; | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  |     ctx.message_context = message_context; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!(source_name = vkd3d_strdup(compile_info->source_name ? compile_info->source_name : "<anonymous>"))) | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < ctx.preprocess_info->macro_count; ++i) | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |         const struct vkd3d_shader_location loc = {.source_name = source_name}; | 
					
						
							|  |  |  |         struct vkd3d_string_buffer body; | 
					
						
							|  |  |  |         char *name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_init(&body); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&body, "%s", ctx.preprocess_info->macros[i].value); | 
					
						
							|  |  |  |         if (!(name = vkd3d_strdup(ctx.preprocess_info->macros[i].name))) | 
					
						
							| 
									
										
										
										
											2021-01-26 20:26:33 +01:00
										 |  |  |         { | 
					
						
							|  |  |  |             vkd3d_string_buffer_cleanup(&body); | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-01-26 20:26:33 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |         if (!preproc_add_macro(&ctx, &loc, name, NULL, 0, &loc, &body)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             vkd3d_free(name); | 
					
						
							| 
									
										
										
										
											2021-01-26 20:26:33 +01:00
										 |  |  |             vkd3d_string_buffer_cleanup(&body); | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     yylex_init_extra(&ctx, &ctx.scanner); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     if (!preproc_push_include(&ctx, source_name, &compile_info->source)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         yylex_destroy(ctx.scanner); | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  |         goto fail; | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     preproc_yyparse(ctx.scanner, &ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:19 -06:00
										 |  |  |     switch (ctx.text_func.state) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case STATE_NONE: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case STATE_ARGS: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const struct vkd3d_shader_location loc = {.source_name = source_name}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             preproc_warning(&ctx, &loc, VKD3D_SHADER_WARNING_PP_UNTERMINATED_MACRO, | 
					
						
							|  |  |  |                     "Unterminated macro invocation."); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         /* fall through */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case STATE_IDENTIFIER: | 
					
						
							|  |  |  |             if (preproc_is_writing(&ctx)) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&ctx.buffer, "%s ", ctx.text_func.macro->name); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     while (ctx.file_count) | 
					
						
							|  |  |  |         preproc_pop_buffer(&ctx); | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |     yylex_destroy(ctx.scanner); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:06 -06:00
										 |  |  |     rb_destroy(&ctx.macros, preproc_macro_rb_free, NULL); | 
					
						
							| 
									
										
										
										
											2021-01-07 11:48:09 -06:00
										 |  |  |     vkd3d_free(ctx.file_stack); | 
					
						
							| 
									
										
										
										
											2021-01-12 16:14:17 -06:00
										 |  |  |     vkd3d_free(ctx.expansion_stack); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 17:13:20 -06:00
										 |  |  |     if (ctx.error) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         WARN("Failed to preprocess.\n"); | 
					
						
							|  |  |  |         vkd3d_string_buffer_cleanup(&ctx.buffer); | 
					
						
							|  |  |  |         return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |     if (!(output_code = vkd3d_malloc(ctx.buffer.content_size))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_cleanup(&ctx.buffer); | 
					
						
							|  |  |  |         return VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     memcpy(output_code, ctx.buffer.buffer, ctx.buffer.content_size); | 
					
						
							|  |  |  |     out->size = ctx.buffer.content_size; | 
					
						
							|  |  |  |     out->code = output_code; | 
					
						
							|  |  |  |     vkd3d_string_buffer_trace(&ctx.buffer); | 
					
						
							|  |  |  |     vkd3d_string_buffer_cleanup(&ctx.buffer); | 
					
						
							|  |  |  |     return VKD3D_OK; | 
					
						
							| 
									
										
										
										
											2021-01-25 11:23:57 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | fail: | 
					
						
							|  |  |  |     rb_destroy(&ctx.macros, preproc_macro_rb_free, NULL); | 
					
						
							|  |  |  |     vkd3d_free(source_name); | 
					
						
							|  |  |  |     vkd3d_string_buffer_cleanup(&ctx.buffer); | 
					
						
							|  |  |  |     return VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  | } |