8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 05:23:02 +01:00
firebird-mirror/src/dsql/Parser.cpp

154 lines
3.6 KiB
C++
Raw Normal View History

/*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.0 (the "License");
* you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Adriano dos Santos Fernandes
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2008 Adriano dos Santos Fernandes <adrianosf@uol.com.br>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#include "firebird.h"
#include "../dsql/Parser.h"
#include "../jrd/jrd.h"
using namespace Firebird;
2008-02-28 14:48:16 +01:00
using namespace Jrd;
Parser::Parser(MemoryPool& pool, USHORT aClientDialect, USHORT aDbDialect, USHORT aParserVersion,
const TEXT* string, USHORT length, SSHORT characterSet)
: PermanentStorage(pool),
compilingText(pool, string, length),
client_dialect(aClientDialect),
db_dialect(aDbDialect),
parser_version(aParserVersion),
transformedString(pool),
introducerMarks(pool),
stmt_ambiguous(false)
{
yyps = 0;
yypath = 0;
yylvals = 0;
yylvp = 0;
yylve = 0;
yylvlim = 0;
yylpsns = 0;
yylpp = 0;
yylpe = 0;
yylplim = 0;
yylexp = 0;
yylexemes = 0;
lex.start = string;
lex.line_start = lex.ptr = string;
lex.end = string + length;
lex.lines = 1;
lex.att_charset = characterSet;
lex.line_start_bk = lex.line_start;
lex.lines_bk = lex.lines;
lex.param_number = 1;
lex.prev_keyword = -1;
#ifdef DSQL_DEBUG
if (DSQL_debug & 32)
dsql_trace("Source DSQL string:\n%.*s", (int) length, string);
#endif
}
Parser::~Parser()
{
while (yyps)
{
yyparsestate* p = yyps;
yyps = p->save;
yyFreeState(p);
}
while (yypath)
{
yyparsestate* p = yypath;
yypath = p->save;
yyFreeState(p);
}
2008-12-21 04:39:32 +01:00
delete[] yylvals;
delete[] yylpsns;
delete[] yylexemes;
}
dsql_nod* Parser::parse()
{
if (parseAux() != 0)
return NULL;
2009-01-08 10:26:06 +01:00
transformString(lex.start, lex.end - lex.start, transformedString);
2009-01-08 10:26:06 +01:00
return DSQL_parse;
}
// Transform strings (or substrings) prefixed with introducer (_charset) to ASCII equivalent.
void Parser::transformString(const char* start, unsigned length, string& dest)
{
const static char HEX_DIGITS[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'A', 'B', 'C', 'D', 'E', 'F'};
const unsigned fromBegin = start - lex.start;
HalfStaticArray<char, 256> buffer;
const char* pos = start;
for (size_t i = 0; i < introducerMarks.getCount(); ++i)
{
const IntroducerMark& mark = introducerMarks[i];
if (mark.pos < fromBegin)
continue;
if (mark.pos >= fromBegin + length)
break;
const char* s = lex.start + mark.pos;
buffer.add(pos, s - pos);
const size_t count = buffer.getCount();
const size_t newSize = count + 2 + mark.textLength * 2 + 1;
buffer.grow(newSize);
char* p = buffer.begin() + count;
*p++ = 'X';
*p++ = '\'';
const char* s2 = lex.start + mark.textPos;
for (const char* end = lex.start + mark.textPos + mark.textLength; s2 < end; ++s2)
{
*p++ = HEX_DIGITS[UCHAR(*s2) >> 4];
*p++ = HEX_DIGITS[UCHAR(*s2) & 0xF];
}
*p = '\'';
2009-09-03 03:28:54 +02:00
fb_assert(p < buffer.begin() + newSize);
pos = s + mark.length;
}
fb_assert(start + length - pos >= 0);
buffer.add(pos, start + length - pos);
dest = string(buffer.begin(), MIN(string::max_length(), buffer.getCount()));
}