/********************************************************************************************************* * Software License Agreement (BSD License) * * Author: Sebastien Decugis * * * * Copyright (c) 2013, WIDE Project and NICT * * All rights reserved. * * * * Redistribution and use of this software in source and binary forms, with or without modification, are * * permitted provided that the following conditions are met: * * * * * Redistributions of source code must retain the above * * copyright notice, this list of conditions and the * * following disclaimer. * * * * * Redistributions in binary form must reproduce the above * * copyright notice, this list of conditions and the * * following disclaimer in the documentation and/or other * * materials provided with the distribution. * * * * * Neither the name of the WIDE Project or NICT nor the * * names of its contributors may be used to endorse or * * promote products derived from this software without * * specific prior written permission of WIDE Project and * * NICT. * * * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED * * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A * * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR * * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR * * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * *********************************************************************************************************/ /* Lex configuration parser. * * This file defines the token for parsing the daemon's configuration file * Note that each extension has a separate independant configuration file. * * Note : This module is NOT thread-safe. All processing must be done from one thread only. */ %{ /* Include the daemon's header files */ #include "fdcore-internal.h" /* Include yacc tokens definitions */ #include "fdd.tab.h" /* Update the column information */ #ifdef DEBUG_LEX #define YY_USER_ACTION { \ yylloc->first_column = yylloc->last_column + 1; \ yylloc->last_column = yylloc->first_column + yyleng - 1; \ fd_log_debug( \ "(%d:%d-%d:%d) matched rule %d, length=%d, txt='%s'", \ yylloc->first_line, yylloc->first_column, \ yylloc->last_line, yylloc->last_column, \ yy_act, yyleng, yytext); \ } #else /* DEBUG_LEX */ #define YY_USER_ACTION { \ yylloc->first_column = yylloc->last_column + 1; \ yylloc->last_column = yylloc->first_column + yyleng - 1; \ } #endif /* %option noinput ? */ #define YY_NO_INPUT /* Additional for files inclusion */ #include #include #define MAX_NESTED_CONF_FILES 5 struct nested_conffiles_t { YY_BUFFER_STATE parent_level_state; glob_t filelist; int current_file; } nested_conffiles[MAX_NESTED_CONF_FILES]; int current_nested_level = 0; int globerrfct(const char *epath, int eerrno) { TRACE_ERROR("Failed to scan %s: %s", epath, strerror(eerrno)); return 1; } %} %option bison-bridge bison-locations %option noyywrap %option nounput %x in_include /* Quoted string. Multilines do not match. */ qstring \"[^\"\n]*\" %% <*>\n { /* Update the line count */ yylloc->first_line++; yylloc->last_line++; yylloc->last_column=0; } <*>([[:space:]]{-}[\n])+ ; /* Eat all spaces, not new lines */ <*>#.*$ ; /* Eat all comments */ include BEGIN(in_include); /* Following an "include" keyword */ { {qstring} { /* Name of the file to include. This is directly sent to glob. */ int globerror=0; char * buf = strdup(yytext+1); if (buf[yyleng-2] != '"') { TRACE_ERROR("Unterminated string: %s", yytext); return LEX_ERROR; } buf[yyleng-2] = '\0'; if (current_nested_level >= MAX_NESTED_CONF_FILES) { TRACE_ERROR("Too many recursion levels in configuration files includes"); return LEX_ERROR; } /* glob the include */ globerror = glob(buf, GLOB_ERR, globerrfct, &nested_conffiles[current_nested_level].filelist); if (globerror == GLOB_NOSPACE) { TRACE_ERROR("Not enough memory to parse include directive."); return LEX_ERROR; } if (globerror == GLOB_ABORTED) { TRACE_ERROR("An error was encountered in include directive."); return LEX_ERROR; } if (globerror == GLOB_NOMATCH) { globfree(&nested_conffiles[current_nested_level].filelist); goto nomatch; } if (globerror) { TRACE_ERROR("Unexpected error in glob (%d).", globerror); return LEX_ERROR; } /* We have a list of files to include. */ /* save the current buffer for returning when this include has been parsed */ nested_conffiles[current_nested_level].parent_level_state = YY_CURRENT_BUFFER; /* Start with the first match */ nested_conffiles[current_nested_level].current_file = 0; yyin = fopen( nested_conffiles[current_nested_level].filelist.gl_pathv[0], "r" ); if ( ! yyin ) { TRACE_ERROR("Error in %s: %s", nested_conffiles[current_nested_level].filelist.gl_pathv[0], strerror(errno)); return LEX_ERROR; } yy_switch_to_buffer(yy_create_buffer( yyin, YY_BUF_SIZE )); /* In case of recursive includes */ current_nested_level++; nomatch: BEGIN(INITIAL); } } <> { if (current_nested_level == 0) { /* We are at the end of parsing */ yyterminate(); } /* Otherwise we are doing an include statement */ --current_nested_level; yy_delete_buffer(YY_CURRENT_BUFFER); /* Go to next file, if any */ nested_conffiles[current_nested_level].current_file++; if ( nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file] == NULL ) { /* We have finished with this list of includes */ globfree(&nested_conffiles[current_nested_level].filelist); yy_switch_to_buffer(nested_conffiles[current_nested_level].parent_level_state); } else { /* Proceed to next included file */ yyin = fopen( nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file], "r" ); if ( ! yyin ) { TRACE_ERROR("Error in %s: %s", nested_conffiles[current_nested_level].filelist.gl_pathv[nested_conffiles[current_nested_level].current_file], strerror(errno)); return LEX_ERROR; } yy_switch_to_buffer(yy_create_buffer( yyin, YY_BUF_SIZE )); /* In case of recursive includes */ current_nested_level++; } } {qstring} { /* First copy the string without the quotes for use in the yacc parser */ CHECK_MALLOC_DO( yylval->string = strdup(yytext+1), /* This allocates one useless tail char but... it's easier :D */ return LEX_ERROR );/* on error, trig an error in yacc parser */ yylval->string[yyleng-2] = '\0'; /* the yacc parser will check the string is valid */ return QSTRING; } [[:digit:]]+ { /* Convert this to an integer value */ int ret = sscanf(yytext, "%i", &yylval->integer); if (ret != 1) { /* No matching: an error occurred */ TRACE_ERROR("Unable to convert the value '%s' to a valid number: %s", yytext, strerror(errno)); return LEX_ERROR; /* trig an error in yacc parser */ /* Maybe we could REJECT instead of failing here? */ } return INTEGER; } /* Full words tokens (keywords) */ (?i:"Identity") { return IDENTITY; } (?i:"Realm") { return REALM; } (?i:"Port") { return PORT; } (?i:"SecPort") { return SECPORT; } /* (?i:"SctpSec3436") { return SEC3436; } */ (?i:"No_IPv6") { return NOIP6; } (?i:"No_IP") { return NOIP; } (?i:"No_TCP") { return NOTCP; } (?i:"No_SCTP") { return NOSCTP; } (?i:"Prefer_TCP") { return PREFERTCP; } (?i:"TLS_old_method") { return OLDTLS; } (?i:"SCTP_streams") { return SCTPSTREAMS; } (?i:"AppServThreads") { return APPSERVTHREADS;} (?i:"ListenOn") { return LISTENON; } (?i:"ThreadsPerServer") { return THRPERSRV; } (?i:"TcTimer") { return TCTIMER; } (?i:"TwTimer") { return TWTIMER; } (?i:"NoRelay") { return NORELAY; } (?i:"LoadExtension") { return LOADEXT; } (?i:"ConnectPeer") { return CONNPEER; } (?i:"ConnectTo") { return CONNTO; } (?i:"No_TLS") { return NOTLS; } (?i:"TLS_Cred") { return TLS_CRED; } (?i:"TLS_CA") { return TLS_CA; } (?i:"TLS_CRL") { return TLS_CRL; } (?i:"TLS_Prio") { return TLS_PRIO; } (?i:"TLS_DH_bits") { return TLS_DH_BITS; } (?i:"TLS_DH_file") { return TLS_DH_FILE; } /* Valid single characters for yyparse */ <*>[=,:;{}] { return yytext[0]; } /* Unrecognized token */ <*>[[:alnum:]]+ | /* This rule is only useful to print a complete token in error messages */ /* Unrecognized character */ <*>. { TRACE_ERROR("Unrecognized text on line %d col %d: '%s'.", yylloc->first_line, yylloc->first_column, yytext); return LEX_ERROR; } %%