%s CODE HERE SSTRING DSTRING RD SUBST %{ /* * $Header: /usr/build/vile/vile/filters/RCS/rb-filt.l,v 1.8 2003/05/20 20:38:41 tom Exp $ * * Filter to add vile "attribution" sequences to a Ruby script * based on the Python filter. * * By Paul van Tilburg * * TODO: %(q|Q|x) char .* char (string2) to STRING * ` .* ` (command) to STRING * / .* / (regexp) to STRING * %r char .* char (regexp) to STRING * # (varsubst) to Identifier in DSTRING & HERE */ #include #include DefineFilter("rb"); #define isIdent(ch) (isalnum(ch) || ch == '_') static char *Comment_attr; static char *Error_attr; static char *Ident_attr; static char *Ident2_attr; static char *Keyword_attr; static char *Number_attr; static char *String_attr; static char *here_tag; static char *here_match; static unsigned here_len; static int in_here; static int save_here(char *text, int len); static void pop_quote(char *text, int length); %} CHAR \?(\\M-)?(\\C-)?. VAR [a-z_][a-zA-Z0-9_]* CONST [A-Z][a-zA-Z0-9_]* INSTANCE @+({CONST}|{VAR}) GLOBAL \$([-_.\/,"\\#%=~|\$?&`'+*\[\];!@<>():]|{CONST}|{VAR}) VARSUBST #({INSTANCE}|{GLOBAL}) SUBSTL #\{ SUBSTR \} COMMENT #.*\n BLANK [ \t]* SSTRING \'(\\.|[^']|\n)*\' DSTRING \"(\\.|[^"]|\n)*\" KEYWORD [a-zA-Z_][a-zA-Z0-9_]* QIDENT ({SSTRING}|{DSTRING}|{KEYWORD}|"`"{KEYWORD}"`") SIGN [-+] DECIMAL [0-9_]+ OCTAL 0[0-7_]+ HEXADECIMAL 0x[0-9a-fA-F_]+ REAL [-+]?([0-9_]*\.[0-9][0-9_]*)([eE][+-]?[0-9_]+)? NUMBER {SIGN}?({DECIMAL}|{OCTAL}|{HEXADECIMAL})[L]?|{REAL} %% {CHAR} | {NUMBER} { WriteToken(Number_attr); } {CONST} { char *attr = keyword_attr(yytext); if (attr == 0) attr = Ident2_attr; WriteToken(attr); } {GLOBAL} { WriteToken(Ident2_attr); } {COMMENT} { WriteToken(Comment_attr); } {KEYWORD} { WriteToken(keyword_attr(yytext)); } \' { PushQuote(SSTRING, String_attr); } \" { PushQuote(DSTRING, String_attr); } ^=begin { WriteToken(Comment_attr); new_state(RD); } <<-?{QIDENT} { if (save_here(yytext, yyleng)) { PushQuote(HERE, String_attr); } else { WriteToken(Error_attr); } } ^{BLANK}{QIDENT} { /* check for 'here'-document delimiter, skip possible spaces first */ here_match = strstr(yytext, here_tag); if (here_match != NULL && !strcmp(here_tag, here_match)) { in_here = 0; /* ok, not in here-doc anymore */ pop_quote(yytext, yyleng); } else { flt_bfr_append(yytext, yyleng); } } {SUBSTL} { if (in_here) { flt_bfr_finish(); PushQuote(SUBST, Ident2_attr); } else { flt_bfr_append(yytext, yyleng); } } (\\.|.) | \n { flt_bfr_append(yytext, yyleng); } ^=end { WriteToken(Comment_attr); new_state(CODE); } .* { WriteToken(Comment_attr); } \' { pop_quote(yytext, yyleng); } (\\.|.) | [\n] { flt_bfr_append(yytext, yyleng); } \" { pop_quote(yytext, yyleng); } {VARSUBST} { flt_bfr_embed(yytext, yyleng, Ident2_attr); } {SUBSTL} { flt_bfr_finish(); PushQuote(SUBST, Ident2_attr); } (\\.|.) | [\n] { flt_bfr_append(yytext, yyleng); } {SUBSTR} { pop_quote(yytext, yyleng); } (\\.|.) | [\n] { flt_bfr_append(yytext, yyleng); } \\. | . { ECHO; } %% static int save_here(char *text, int len) { char *s = here_tag = do_alloc(here_tag, len, &here_len); text += 2; /* skip over "<<" */ if (*text == '-') ++text; while (len-- > 0) { if (isIdent(CharOf(*text))) { *s++ = *text; } else if (*text == '\'' || *text == '\"') { ; } else { break; } text++; } *s = 0; return (s - here_tag) != 0; } /* * Replaces PopQuote(). */ static void pop_quote(char *text, int length) { char *attr = 0; flt_bfr_append(text, length); flt_bfr_finish(); pop_state(); switch (stk_state[stk_level].state) { case HERE: case SSTRING: case DSTRING: attr = String_attr; break; } if (attr != 0) flt_bfr_begin(String_attr); } static void init_filter(int before GCC_UNUSED) { } static void do_filter(FILE *inputs) { yyin = inputs; Comment_attr = class_attr(NAME_COMMENT); Error_attr = class_attr(NAME_ERROR); Ident_attr = class_attr(NAME_IDENT); Ident2_attr = class_attr(NAME_IDENT2); Keyword_attr = class_attr(NAME_KEYWORD); Number_attr = class_attr(NAME_NUMBER); String_attr = class_attr(NAME_LITERAL); begin_state(CODE); while (yylex() > 0) { } flt_bfr_finish(); end_state(); }