Files
GoldenCheetah/src/FileIO/JsonRideFile.l
Eric Christoffersen c13b24251e Speedup RideCache Load Time
Implement regex/hash based string substitution object to perform multiple
substitutions in 2 passes. Speeds up athlete data load by 2x.
Use QStringRef to avoid copy
Fixes #3234
2019-12-02 19:54:30 -03:00

175 lines
6.2 KiB
Plaintext

%{
/*
* Copyright (c) 2010 Mark Liversedge (liversedge@gmail.com)
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the Free
* Software Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc., 51
* Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "JsonRideFile.h"
// we use stdio for reading from FILE *JsonRideFilein
// because thats what lex likes to do, and since we're
// reading files that seems ok anyway
#include <stdio.h>
// The parser defines the token values for us
// so lets include them before declaring the
// token patterns
#include "JsonRideFile_yacc.h"/* generated by the scanner */
// the options below tell flex to no bother with
// yywrap since we only ever read a single file
// anyway. And yyunput() isn't needed for our
// parser, we read in one pass with no swanky
// interactions
#define YYSTYPE QString
// Un-Escape special characters (JSON compliance)
static QString unprotect(char *string)
{
// sending UTF-8 to FLEX demands symetric conversion back to QString
QString string2 = QString::fromUtf8(string);
// this is a lexer string so it will be enclosed
// in quotes. Lets strip those first
QStringRef r = string2.midRef(1,string2.length()-2);
// does it end with a space (to avoid token conflict) ?
if (r.endsWith(" ")) r = r.mid(0, r.length()-1);
QString s = Utils::RidefileUnEscape(r);
return s;
}
// we reimplement these to remove compiler warnings
// about unused parameter (scanner) in the default
// implementations, which may freak out developers
void *JsonRideFilealloc (yy_size_t size , yyscan_t /*scanner*/)
{
return (void *) malloc( size );
}
void *JsonRideFilerealloc (void * ptr, yy_size_t size , yyscan_t /*scanner*/)
{
/* The cast to (char *) in the following accommodates both
* implementations that use char* generic pointers, and those
* that use void* generic pointers. It works with the latter
* because both ANSI C and C++ allow castless assignment from
* any pointer type to void*, and deal with argument conversions
* as though doing an assignment.
*/
return (void *) realloc( (char *) ptr, size );
}
void JsonRideFilefree (void * ptr , yyscan_t /*scanner*/)
{
free( (char *) ptr ); /* see JsonRideFilerealloc() for (char *) cast */
}
// replace this too, as a) it exits (!!)
// cannot shutup compiler warning on yy_fatal_error function tho :(
#define YY_FATAL_ERROR(msg) qDebug()<<msg;
%}
%option prefix="JsonRideFile"
%option never-interactive
%option noyyalloc
%option noyyrealloc
%option noyyfree
%option noyywrap
%option nounput
%option noinput
%option reentrant
%option bison-bridge
%%
\"RIDE\" return RIDE;
\"STARTTIME\" return STARTTIME;
\"RECINTSECS\" return RECINTSECS;
\"DEVICETYPE\" return DEVICETYPE;
\"IDENTIFIER\" return IDENTIFIER;
\"OVERRIDES\" return OVERRIDES;
\"TAGS\" return TAGS;
\"INTERVALS\" return INTERVALS;
\"NAME\" return NAME;
\"START\" return START;
\"STOP\" return STOP;
\"PTEST\" return TEST; /* bool is a performance test */
\"COLOR\" return COLOR;
\"CALIBRATIONS\" return CALIBRATIONS;
\"VALUE\" return VALUE;
\"VALUES\" return VALUES;
\"UNIT\" return UNIT;
\"UNITS\" return UNITS;
\"XDATA\" return XDATA;
\"REFERENCES\" return REFERENCES;
\"SAMPLES\" return SAMPLES;
\"SECS\" return SECS;
\"KM\" return KM;
\"WATTS\" return WATTS;
\"NM\" return NM;
\"CAD\" return CAD;
\"KPH\" return KPH;
\"HR\" return HR;
\"ALT\" return ALTITUDE; // ALT clashes with qtnamespace.h:46
\"LAT\" return LAT;
\"LON\" return LON;
\"HEADWIND\" return HEADWIND;
\"SLOPE\" return SLOPE;
\"TEMP\" return TEMP;
\"LRBALANCE\" return LRBALANCE;
\"LTE\" return LTE;
\"RTE\" return RTE;
\"LPS\" return LPS;
\"RPS\" return RPS;
\"LPCO\" return LPCO;
\"RPCO\" return RPCO;
\"LPPB\" return LPPB;
\"RPPB\" return RPPB;
\"LPPE\" return LPPE;
\"RPPE\" return RPPE;
\"LPPPB\" return LPPPB;
\"RPPPB\" return RPPPB;
\"LPPPE\" return LPPPE;
\"RPPPE\" return RPPPE;
\"SMO2\" return SMO2;
\"THB\" return THB;
\"RCON\" return RCON;
\"RVERT\" return RVERT;
\"RCAD\" return RCAD;
[-+]?[0-9]+ { *yylval = QString::fromUtf8(yytext); return JS_INTEGER; }
[-+]?[0-9]+e-[0-9]+ { *yylval = QString::fromUtf8(yytext); return JS_FLOAT; }
[-+]?[0-9]+\.[-+e0-9]* { *yylval = QString::fromUtf8(yytext); return JS_FLOAT; }
\"([^\"]|\\\")*\" { *yylval = unprotect(yytext); return JS_STRING; } /* contains non-quotes or escaped-quotes */
[ \n\t\r] ; /* we just ignore whitespace */
. return yytext[0]; /* any other character, typically :, { or } */
%%
// Older versions of flex (prior to 2.5.9) do not have the destroy function
// Or We're not using GNU flex then we also won't have a destroy function
#if !defined(FLEX_SCANNER) || (YY_FLEX_MINOR_VERSION < 6 && YY_FLEX_SUBMINOR_VERSION < 9)
int JsonRideFilelex_destroy(void*) { return 0; }
#endif
void JsonRideFile_setString(QString p, void *scanner)
{
// internally work with UTF-8 encoding
// this works for FLEX, since the multi-byte characters only appear WITHIN a "String",
// but not as part of the grammar - this is important since a char in UTF-8 can have up to 4 bytes
JsonRideFile_scan_string(p.toUtf8().data(), scanner);
}