/********************************************************************************************************* * 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 for radius_gw extension. * */ %{ #include "rgw.h" #include "rgw_conf.tab.h" /* Update the column information */ #define YY_USER_ACTION { \ yylloc->first_column = yylloc->last_column + 1; \ yylloc->last_column = yylloc->first_column + yyleng - 1; \ } /* %option noinput ? */ #define YY_NO_INPUT %} %option bison-bridge bison-locations %option noyywrap %option nounput /* Use the following start condition to parse an URI */ %x IN_PLG %x IN_CLI1 %x IN_CLI2 %x EXPECT_IP4 %x EXPECT_IP6 %x EXPECT_DECINT /* Quoted string. Multilines do not match. */ qstring \"[^\"\n]*\" /* Used to match IP, IP6, and port */ IP4 [0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3} IP6 [[:xdigit:]]*:[[:xdigit:]]*:[[:xdigit:].:]* BR_PORT [[][0-9]+[]] %% /* All sections */ <*>\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 */ <*>{qstring} { /* First copy the string without the quotes for use in the yacc parser */ yylval->string = strdup(yytext+1); if (yylval->string == NULL) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); return LEX_ERROR; /* trig an error in yacc parser */ } yylval->string[yyleng-2] = '\0'; /* the yacc parser will check the string is valid */ return QSTRING; } /* Extension section */ (?i:"RGWX") { BEGIN(IN_PLG); return PLG_PREFIX; } (?i:"auth") { return AUTH; } (?i:"acct") { return ACCT; } [[:xdigit:]]+ { /* Convert this to an integer value */ int ret = sscanf(yytext, "%x", &yylval->integer); if (ret != 1) { /* No matching: an error occurred */ fd_log_debug("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; } [:] { return yytext[0]; } /* Client section */ (?i:"nas"|"cli") { BEGIN(IN_CLI1); yylval->integer=RGW_CLI_NAS; return NAS_OR_PXY; } (?i:"pxy") { BEGIN(IN_CLI1); yylval->integer=RGW_CLI_PXY; return NAS_OR_PXY; } /* Match an IP (4 or 6) and optional port */ ({IP4}|{IP6}){BR_PORT}? { char * work; char * port; unsigned short p = 0; work = strdup(yytext); if ( work == NULL ) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); return LEX_ERROR; /* trig an error in yacc parser */ } if ((port = strchr(work, '[')) != NULL) { *port = '\0'; port++; if (sscanf(port, "%hu]", &p) != 1) { fd_log_debug("'%s' is not a valid port: %s", port, strerror(errno)); free(work); return LEX_ERROR; /* trig an error in yacc parser */ } } /* Do we have an IP or IPv6? Let's check if we have ':' char somewhere in the beginning */ if (memchr(work, ':', 5) != NULL) { struct sockaddr_in6 * sin6 = NULL; sin6 = malloc(sizeof(struct sockaddr_in6)); if (sin6 == NULL) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); free(work); return LEX_ERROR; /* trig an error in yacc parser */ } memset(sin6, 0, sizeof(struct sockaddr_in6)); sin6->sin6_family = AF_INET6; if (inet_pton(AF_INET6, work, &sin6->sin6_addr) != 1) { fd_log_debug("'%s' is not a valid IPv6 address: %s", work, strerror(errno)); free(work); free(sin6); return LEX_ERROR; /* trig an error in yacc parser */ } sin6->sin6_port = htons(p); yylval->ss = (struct sockaddr *)sin6; } else { struct sockaddr_in * sin = NULL; sin = malloc(sizeof(struct sockaddr_in)); if (sin == NULL) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); free(work); return LEX_ERROR; /* trig an error in yacc parser */ } memset(sin, 0, sizeof(struct sockaddr_in)); sin->sin_family = AF_INET; if (inet_pton(AF_INET, work, &sin->sin_addr) != 1) { fd_log_debug("'%s' is not a valid IP address: %s", work, strerror(errno)); free(work); free(sin); return LEX_ERROR; /* trig an error in yacc parser */ } sin->sin_port = htons(p); yylval->ss = (struct sockaddr *)sin; } free(work); return IP; } "/" { BEGIN(IN_CLI2); return '/'; } /* Servers section */ (?i:"auth_server_enable") { BEGIN(EXPECT_DECINT); return AUTH_ENABLE; } (?i:"auth_server_port") { BEGIN(EXPECT_DECINT); return AUTH_PORT; } (?i:"auth_server_ip4") { BEGIN(EXPECT_IP4); return AUTH_IP4; } (?i:"auth_server_ip6") { BEGIN(EXPECT_IP6); return AUTH_IP6; } (?i:"acct_server_enable") { BEGIN(EXPECT_DECINT); return ACCT_ENABLE; } (?i:"acct_server_port") { BEGIN(EXPECT_DECINT); return ACCT_PORT; } (?i:"acct_server_ip4") { BEGIN(EXPECT_IP4); return ACCT_IP4; } (?i:"acct_server_ip6") { BEGIN(EXPECT_IP6); return ACCT_IP6; } [[:digit:]]+ { /* Match an integer (not hexa) */ int ret = sscanf(yytext, "%d", &yylval->integer); if (ret != 1) { /* No matching: an error occurred */ fd_log_debug("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; } (?i:"disable") { return DISABLED; } {IP4} { struct sockaddr_in * sin = NULL; sin = malloc(sizeof(struct sockaddr_in)); if (sin == NULL) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); return LEX_ERROR; /* trig an error in yacc parser */ } memset(sin, 0, sizeof(struct sockaddr_in)); sin->sin_family = AF_INET; if (inet_pton(AF_INET, yytext, &sin->sin_addr) != 1) { fd_log_debug("'%s' is not a valid IP address: %s", yytext, strerror(errno)); free(sin); return LEX_ERROR; /* trig an error in yacc parser */ } yylval->ss = (struct sockaddr *)sin; return IP; } {IP6} { struct sockaddr_in6 * sin6 = NULL; sin6 = malloc(sizeof(struct sockaddr_in6)); if (sin6 == NULL) { fd_log_debug("Unable to allocate memory: %s", strerror(errno)); return LEX_ERROR; /* trig an error in yacc parser */ } memset(sin6, 0, sizeof(struct sockaddr_in6)); sin6->sin6_family = AF_INET6; if (inet_pton(AF_INET6, yytext, &sin6->sin6_addr) != 1) { fd_log_debug("'%s' is not a valid IPv6 address: %s", yytext, strerror(errno)); free(sin6); return LEX_ERROR; /* trig an error in yacc parser */ } yylval->ss = (struct sockaddr *)sin6; return IP; } /* Valid single characters for yyparse in all contexts */ <*>[=] { return yytext[0]; } <*>[;] { BEGIN(INITIAL); return yytext[0]; } /* Unrecognized token */ <*>[[:alnum:]]+ | /* This rule is only useful to print a complete token in error messages */ /* Unrecognized character */ <*>. { fd_log_debug("Unrecognized text on line %d col %d: '%s'.", yylloc->first_line, yylloc->first_column, yytext); return LEX_ERROR; } %%