/* * Copyright 2006-2018 Adrian Thurston * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to * deal in the Software without restriction, including without limitation the * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or * sell copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include "pcheck.h" #include "common.h" #include "stdlib.h" #include #include #include "ragel.h" /* * C */ const char *defaultOutFnC( const char *inputFileName ) { const char *ext = findFileExtension( inputFileName ); if ( ext != 0 && strcmp( ext, ".rh" ) == 0 ) return fileNameFromStem( inputFileName, ".h" ); else return fileNameFromStem( inputFileName, ".c" ); } HostType hostTypesC[] = { { "char", 0, "char", true, true, false, CHAR_MIN, CHAR_MAX, 0, 0, sizeof(char) }, { "signed", "char", "char", true, true, false, CHAR_MIN, CHAR_MAX, 0, 0, sizeof(char) }, { "unsigned", "char", "uchar", false, true, false, 0, 0, 0, UCHAR_MAX, sizeof(unsigned char) }, { "short", 0, "short", true, true, false, SHRT_MIN, SHRT_MAX, 0, 0, sizeof(short) }, { "signed", "short", "short", true, true, false, SHRT_MIN, SHRT_MAX, 0, 0, sizeof(short) }, { "unsigned", "short", "ushort", false, true, false, 0, 0, 0, USHRT_MAX, sizeof(unsigned short) }, { "int", 0, "int", true, true, false, INT_MIN, INT_MAX, 0, 0, sizeof(int) }, { "signed", "int", "int", true, true, false, INT_MIN, INT_MAX, 0, 0, sizeof(int) }, { "unsigned", "int", "uint", false, true, false, 0, 0, 0, UINT_MAX, sizeof(unsigned int) }, { "long", 0, "long", true, true, false, LONG_MIN, LONG_MAX, 0, 0, sizeof(long) }, { "signed", "long", "long", true, true, false, LONG_MIN, LONG_MAX, 0, 0, sizeof(long) }, { "unsigned", "long", "ulong", false, true, false, 0, 0, 0, ULONG_MAX, sizeof(unsigned long) }, }; const HostLang hostLangC = { hostTypesC, 12, 0, true, false, /* loopLabels */ Direct, GotoFeature, &makeCodeGen, &defaultOutFnC, &genLineDirectiveC }; /* * ASM */ const char *defaultOutFnAsm( const char *inputFileName ) { return fileNameFromStem( inputFileName, ".s" ); } HostType hostTypesAsm[] = { { "char", 0, "char", true, true, false, CHAR_MIN, CHAR_MAX, 0, 0, sizeof(char) }, { "unsigned", "char", "uchar", false, true, false, 0, 0, 0, UCHAR_MAX, sizeof(unsigned char) }, { "short", 0, "short", true, true, false, SHRT_MIN, SHRT_MAX, 0, 0, sizeof(short) }, { "unsigned", "short", "ushort", false, true, false, 0, 0, 0, USHRT_MAX, sizeof(unsigned short) }, { "int", 0, "int", true, true, false, INT_MIN, INT_MAX, 0, 0, sizeof(int) }, { "unsigned", "int", "uint", false, true, false, 0, 0, 0, UINT_MAX, sizeof(unsigned int) }, { "long", 0, "long", true, true, false, LONG_MIN, LONG_MAX, 0, 0, sizeof(long) }, { "unsigned", "long", "ulong", false, true, false, 0, 0, 0, ULONG_MAX, sizeof(unsigned long) }, }; const HostLang hostLangAsm = { hostTypesAsm, 8, 0, true, false, /* loopLabels */ Direct, GotoFeature, &makeCodeGenAsm, &defaultOutFnC, &genLineDirectiveAsm }; HostType *findAlphType( const HostLang *hostLang, const char *s1 ) { for ( int i = 0; i < hostLang->numHostTypes; i++ ) { if ( strcmp( s1, hostLang->hostTypes[i].data1 ) == 0 && hostLang->hostTypes[i].data2 == 0 ) { return hostLang->hostTypes + i; } } return 0; } HostType *findAlphType( const HostLang *hostLang, const char *s1, const char *s2 ) { for ( int i = 0; i < hostLang->numHostTypes; i++ ) { if ( strcmp( s1, hostLang->hostTypes[i].data1 ) == 0 && hostLang->hostTypes[i].data2 != 0 && strcmp( s2, hostLang->hostTypes[i].data2 ) == 0 ) { return hostLang->hostTypes + i; } } return 0; } HostType *findAlphTypeInternal( const HostLang *hostLang, const char *s1 ) { for ( int i = 0; i < hostLang->numHostTypes; i++ ) { if ( strcmp( s1, hostLang->hostTypes[i].internalName ) == 0 ) return hostLang->hostTypes + i; } return 0; } /* Construct a new parameter checker with for paramSpec. */ ParamCheck::ParamCheck( const char *paramSpec, int argc, const char **argv ) : state(noparam), argOffset(0), curArg(0), iCurArg(1), paramSpec(paramSpec), argc(argc), argv(argv) { } /* Check a single option. Returns the index of the next parameter. Sets p to * the arg character if valid, 0 otherwise. Sets parg to the parameter arg if * there is one, NULL otherwise. */ bool ParamCheck::check() { bool requiresParam; if ( iCurArg >= argc ) { /* Off the end of the arg list. */ state = noparam; return false; } if ( argOffset != 0 && *argOffset == 0 ) { /* We are at the end of an arg string. */ iCurArg += 1; if ( iCurArg >= argc ) { state = noparam; return false; } argOffset = 0; } if ( argOffset == 0 ) { /* Set the current arg. */ curArg = argv[iCurArg]; /* We are at the beginning of an arg string. */ if ( argv[iCurArg] == 0 || /* Argv[iCurArg] is null. */ argv[iCurArg][0] != '-' || /* Not a param. */ argv[iCurArg][1] == 0 ) { /* Only a dash. */ parameter = 0; paramArg = 0; iCurArg += 1; state = noparam; return true; } argOffset = argv[iCurArg] + 1; } /* Get the arg char. */ char argChar = *argOffset; /* Loop over all the parms and look for a match. */ const char *pSpec = paramSpec; while ( *pSpec != 0 ) { char pSpecChar = *pSpec; /* If there is a ':' following the char then * it requires a parm. If a parm is required * then move ahead two in the parmspec. Otherwise * move ahead one in the parm spec. */ if ( pSpec[1] == ':' ) { requiresParam = true; pSpec += 2; } else { requiresParam = false; pSpec += 1; } /* Do we have a match. */ if ( argChar == pSpecChar ) { if ( requiresParam ) { if ( argOffset[1] == 0 ) { /* The param must follow. */ if ( iCurArg + 1 == argc ) { /* We are the last arg so there * cannot be a parameter to it. */ parameter = argChar; paramArg = 0; iCurArg += 1; argOffset = 0; state = invalid; return true; } else { /* the parameter to the arg is the next arg. */ parameter = pSpecChar; paramArg = argv[iCurArg + 1]; iCurArg += 2; argOffset = 0; state = match; return true; } } else { /* The param for the arg is built in. */ parameter = pSpecChar; paramArg = argOffset + 1; iCurArg += 1; argOffset = 0; state = match; return true; } } else { /* Good, we matched the parm and no * arg is required. */ parameter = pSpecChar; paramArg = 0; argOffset += 1; state = match; return true; } } } /* We did not find a match. Bad Argument. */ parameter = argChar; paramArg = 0; argOffset += 1; state = invalid; return true; } std::streamsize output_filter::countAndWrite( const char *s, std::streamsize n ) { for ( int i = 0; i < n; i++ ) { switch ( s[i] ) { case '\n': line += 1; break; case '{': /* If we detec an open block then eliminate the single-indent * addition, which is to account for single statements. */ singleIndent = false; level += 1; break; case '}': level -= 1; break; } } return std::filebuf::xsputn( s, n ); } bool openSingleIndent( const char *s, int n ) { if ( n >= 3 && memcmp( s, "if ", 3 ) == 0 ) return true; if ( n >= 8 && memcmp( s, "else if ", 8 ) == 0 ) return true; if ( n >= 5 && memcmp( s, "else\n", 4 ) == 0 ) return true; return false; } /* Counts newlines before sending sync. */ int output_filter::sync( ) { line += 1; return std::filebuf::sync(); } /* Counts newlines before sending data out to file. */ std::streamsize output_filter::xsputn( const char *s, std::streamsize n ) { std::streamsize ret = n; int l; restart: if ( indent ) { /* Consume mode Looking for the first non-whitespace. */ while ( n > 0 && ( *s == ' ' || *s == '\t' ) ) { s += 1; n -= 1; } if ( n > 0 ) { int tabs = level + ( singleIndent ? 1 : 0 ); if ( *s == '}' ) { /* If the next char is de-dent, then reduce the tabs. This is * not a stream state change. The level reduction will be * computed in write. */ tabs -= 1; } /* Note that the count and write will eliminate this if it detects * an open block. */ if ( openSingleIndent( s, n ) ) singleIndent = true; else singleIndent = false; if ( *s != '#' ) { /* Found some data, print the indentation and turn off indentation * mode. */ for ( l = 0; l < tabs; l++ ) countAndWrite( "\t", 1 ); } indent = 0; goto restart; } } else { char *nl; if ( (nl = (char*)memchr( s, '\n', n )) ) { /* Print up to and including the newline. */ int wl = nl - s + 1; countAndWrite( s, wl ); /* Go into consume state. If we see more non-indentation chars we * will generate the appropriate indentation level. */ s += wl; n -= wl; indent = true; goto restart; } else { /* Indentation off, or no indent trigger (newline). */ countAndWrite( s, n ); } } // What to do here? return ret; } /* Scans a string looking for the file extension. If there is a file * extension then pointer returned points to inside the string * passed in. Otherwise returns null. */ const char *findFileExtension( const char *stemFile ) { const char *ppos = stemFile + strlen(stemFile) - 1; /* Scan backwards from the end looking for the first dot. * If we encounter a '/' before the first dot, then stop the scan. */ while ( 1 ) { /* If we found a dot or got to the beginning of the string then * we are done. */ if ( ppos == stemFile || *ppos == '.' ) break; /* If we hit a / then there is no extension. Done. */ if ( *ppos == '/' ) { ppos = stemFile; break; } ppos--; } /* If we got to the front of the string then bail we * did not find an extension */ if ( ppos == stemFile ) ppos = 0; return ppos; } /* Make a file name from a stem. Removes the old filename suffix and * replaces it with a new one. Returns a newed up string. */ const char *fileNameFromStem( const char *stemFile, const char *suffix ) { long len = strlen( stemFile ); assert( len > 0 ); /* Get the extension. */ const char *ppos = findFileExtension( stemFile ); /* If an extension was found, then shorten what we think the len is. */ if ( ppos != 0 ) len = ppos - stemFile; /* Make the return string from the stem and the suffix. */ char *retVal = new char[ len + strlen( suffix ) + 1 ]; strncpy( retVal, stemFile, len ); strcpy( retVal + len, suffix ); return retVal; } exit_object endp; void operator<<( std::ostream &out, exit_object & ) { out << std::endl; throw AbortCompile( 1 ); } void genLineDirectiveC( std::ostream &out, bool lineDirectives, int line, const char *fileName ) { if ( !lineDirectives ) out << "/* "; out << "#line " << line << " \""; for ( const char *pc = fileName; *pc != 0; pc++ ) { if ( *pc == '\\' ) out << "\\\\"; else if ( *pc == '"' ) out << "\\\""; else out << *pc; } out << '"'; if ( !lineDirectives ) out << " */"; out << '\n'; } void genLineDirectiveAsm( std::ostream &out, bool lineDirectives, int line, const char *fileName ) { out << "/* #line " << line << " \""; for ( const char *pc = fileName; *pc != 0; pc++ ) { if ( *pc == '\\' ) out << "\\\\"; else if ( *pc == '"' ) out << "\\\""; else out << *pc; } out << '"'; out << " */\n"; } void genLineDirectiveTrans( std::ostream &out, bool lineDirectives, int line, const char *fileName ) { }