%{ /*---------------------------------------------------------------------------* Project: TwlSDK - tools - makelcf File: spec.l Copyright 2003-2008 Nintendo. All rights reserved. These coded instructions, statements, and computer programs contain proprietary information of Nintendo of America Inc. and/or Nintendo Company Ltd., and are protected by Federal copyright law. They may not be disclosed to third parties or copied or duplicated in any form, in whole or in part, without the prior written consent of Nintendo. $Date:: 2008-09-18#$ $Rev: 8573 $ $Author: okubata_ryoma $ *---------------------------------------------------------------------------*/ #include #include #include "makelcf.h" #include "defval.h" #include "spec.tab.h" #define RETURN(x) { debug_printf( "%s [%s]\n", #x, spec_yytext ); return(x); } static void YYText_ResolveDevVal( void ); static char* getSection( void ); %} ws [ \t] nl \r?\n identifier [A-Za-z_][A-Za-z_0-9]* decnum -?[1-9][0-9]* octnum -?0[0-7]* hexnum -?0[xX][0-9A-Fa-f]+ qstring \"[^\"\n\r]*\" string_wo_ws [^ \t\n\r]+ %s START %s PARAM %s PARAM_CONTINUE %option pointer %option yylineno %option noyywrap %% //====================================================================== // SPEC TOKENS //====================================================================== #.* ; /* Comment */ \/\/.* ; /* Comment */ .* { YYText_ResolveDevVal(); BEGIN START; } .* { YYText_ResolveDevVal(); BEGIN PARAM; } (Static|STATIC|static) { BEGIN PARAM; RETURN(tSTATIC); } (Autoload|AUTOLOAD|autoload) { BEGIN PARAM; RETURN(tAUTOLOAD); } (Overlay|OVERLAY|overlay) { BEGIN PARAM; RETURN(tOVERLAY); } (Ltd|LTD|ltd)(Autoload|AUTOLOAD|autoload) { BEGIN PARAM; RETURN(tLTDAUTOLOAD); } (Ltd|LTD|ltd)(Overlay|OVERLAY|overlay) { BEGIN PARAM; RETURN(tLTDOVERLAY); } (Property|PROPERTY|property) { BEGIN PARAM; RETURN(tPROPERTY); } (Properties|PROPERTIES|properties) { BEGIN PARAM; RETURN(tPROPERTY); } (Group|GROUP|group) { BEGIN PARAM; RETURN(tGROUP); } (Address|ADDRESS|address) { BEGIN PARAM; RETURN(tADDRESS); } (After|AFTER|after) { BEGIN PARAM; RETURN(tAFTER); } (Object|OBJECT|object)[sS]? { BEGIN PARAM; RETURN(tOBJECT); } (Librar|LIBRAR|librar)(y|Y|ies|IES)? { BEGIN PARAM; RETURN(tLIBRARY); } (Search|SEARCH|search)_?(Symbol|SYMBOL|symbol)[sS]? { BEGIN PARAM; RETURN(tSEARCHSYMBOL); } (StackSize|STACKSIZE|stacksize) { BEGIN PARAM; RETURN(tSTACKSIZE); } (OverlayDefs|OVERLAYDEFS|overlaydefs) { BEGIN PARAM; RETURN(tOVERLAYDEFS); } (OverlayTable|OVERLAYTABLE|overlaytable) { BEGIN PARAM; RETURN(tOVERLAYTABLE); } (Ltd|LTD|ltd)(OverlayDefs|OVERLAYDEFS|overlaydefs) { BEGIN PARAM; RETURN(tLTDOVERLAYDEFS); } (Ltd|LTD|ltd)(OverlayTable|OVERLAYTABLE|overlaytable) { BEGIN PARAM; RETURN(tLTDOVERLAYTABLE); } (Sur?ffix|SUR?FFIX|sur?ffix) { BEGIN PARAM; RETURN(tSUFFIX); } (Flx|FLX|flx)(Sur?ffix|SUR?FFIX|sur?ffix) { BEGIN PARAM; RETURN(tFLXSUFFIX); } (Ltd|LTD|ltd)(Sur?ffix|SUR?FFIX|sur?ffix) { BEGIN PARAM; RETURN(tLTDSUFFIX); } (ITCM|itcm|Itcm) { spec_yylval.string = strdup("ITCM"); RETURN(tSTRING_ID); } (DTCM|dtcm|Dtcm) { spec_yylval.string = strdup("DTCM"); RETURN(tSTRING_ID); } (WRAM|wram|Wram) { spec_yylval.string = strdup("WRAM"); RETURN(tSTRING_ID); } \{ { BEGIN PARAM; RETURN(tBEGIN); } \} { BEGIN PARAM; RETURN(tEND); } {ws}*\\{ws}*{nl} { BEGIN PARAM_CONTINUE; } /* Continue to next line */ \({ws}*\.[a-z]*{ws}*\) { spec_yylval.string = getSection(); RETURN(tSECTIONNAME); } \({ws}*\.[a-z]*\.[a-z]*{ws}*\) { spec_yylval.string = getSection(); RETURN(tSECTIONNAME); } \* { spec_yylval.string = strdup(spec_yytext); RETURN(tSTRING_STAR); } GROUP\([^\)]*\) { spec_yylval.string = strdup(spec_yytext); RETURN(tSTRING_GROUP); } OBJECT\([^\)]*\) { spec_yylval.string = strdup(spec_yytext); RETURN(tSTRING_FUNCTION); } {decnum} | {octnum} | {hexnum} { /* Integer Value */ spec_yylval.integer = strtol((char*)spec_yytext, (char**)NULL, 0); RETURN(tNUMBER); } {identifier} { /* String for identifier */ spec_yylval.string = strdup(spec_yytext); RETURN(tSTRING_ID); } {qstring} { /* String quated by "" */ spec_yytext[spec_yyleng-1] = '\0'; spec_yylval.string = strdup(spec_yytext+1); RETURN(tSTRING_QUATED); } {string_wo_ws} { /* String without white space */ spec_yylval.string = strdup(spec_yytext); RETURN(tSTRING_NOSPACE); } <> { BEGIN INITIAL; RETURN(tNL); } /* End of File */ {nl} { BEGIN INITIAL; RETURN(tNL); } /* Newline */ {ws}+ ; /* White space */ . { RETURN(spec_yytext[0]); } /* Default */ %% /*============================================================================ * Utilities */ void spec_yyerror( const char* str ) { int line = spec_yylineno; if ( spec_yytext[0] == '\n' ) line --; fprintf( stderr, "makelcf: line %d: %s\n", line, str ); } static char* getSection( void ) { char* p; char* head = NULL; for ( p = spec_yytext; *p; p ++ ) { switch ( *p ) { case '(': while ( *p == ' ' ) p ++; head = p + 1; break; case ')': case ' ': if ( head ) { *p = '\0'; return strdup( head ); } break; default: break; } } return strdup( "*" ); } static void YYText_ResolveDevVal( void ) { int i; char* s = ResolveDefVal( spec_yytext ); for ( i = strlen(s)-1; i >= 0; i -- ) { unput( s[i] ); } free(s); return; } /*============================================================================ * PARSE SPEC FILE */ int ParseSpecFile( const char* filename ) { FILE *fp; int result; if ( NULL == ( fp = fopen( filename, "r" ) ) ) { fprintf( stderr, "makelcf: cannot open %s\n", filename ); return 2; } spec_yyin = fp; result = spec_yyparse(); fclose( fp ); if ( result ) return 1; return CheckSpec() ? 0 : 1; }