2008-02-03 20:16:12 +01:00
|
|
|
/*
|
|
|
|
* 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"
|
2008-04-15 04:18:38 +02:00
|
|
|
#include "../jrd/jrd.h"
|
2008-02-03 20:16:12 +01:00
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
using namespace Firebird;
|
2008-02-28 14:48:16 +01:00
|
|
|
using namespace Jrd;
|
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
|
2008-04-15 04:18:38 +02:00
|
|
|
Parser::Parser(MemoryPool& pool, USHORT aClientDialect, USHORT aDbDialect, USHORT aParserVersion,
|
2008-02-03 20:16:12 +01:00
|
|
|
const TEXT* string, USHORT length, SSHORT characterSet)
|
2008-04-15 04:18:38 +02:00
|
|
|
: PermanentStorage(pool),
|
2009-10-21 02:42:38 +02:00
|
|
|
compilingText(pool, string, length),
|
2008-04-15 04:18:38 +02:00
|
|
|
client_dialect(aClientDialect),
|
2008-02-03 20:16:12 +01:00
|
|
|
db_dialect(aDbDialect),
|
|
|
|
parser_version(aParserVersion),
|
2009-08-30 04:26:50 +02:00
|
|
|
transformedString(pool),
|
|
|
|
introducerMarks(pool),
|
2008-02-03 20:16:12 +01:00
|
|
|
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;
|
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
lex.start = string;
|
2008-02-03 20:16:12 +01:00
|
|
|
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;
|
2008-02-03 20:16:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-21 02:42:38 +02:00
|
|
|
dsql_nod* Parser::parse()
|
2008-02-03 20:16:12 +01:00
|
|
|
{
|
|
|
|
if (parseAux() != 0)
|
|
|
|
return NULL;
|
2009-01-08 10:26:06 +01:00
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
transformString(lex.start, lex.end - lex.start, transformedString);
|
|
|
|
|
2009-01-08 10:26:06 +01:00
|
|
|
return DSQL_parse;
|
2008-02-03 20:16:12 +01:00
|
|
|
}
|
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
|
|
|
|
// 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'};
|
|
|
|
|
2009-09-02 06:23:02 +02:00
|
|
|
const unsigned fromBegin = start - lex.start;
|
2009-08-30 04:26:50 +02:00
|
|
|
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;
|
2009-09-02 06:23:02 +02:00
|
|
|
|
|
|
|
if (mark.pos >= fromBegin + length)
|
2009-08-30 04:26:50 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
const char* s = lex.start + mark.pos;
|
|
|
|
buffer.add(pos, s - pos);
|
|
|
|
|
2009-09-02 06:23:02 +02:00
|
|
|
const size_t count = buffer.getCount();
|
|
|
|
const size_t newSize = count + 2 + mark.textLength * 2 + 1;
|
|
|
|
buffer.grow(newSize);
|
2009-08-30 04:26:50 +02:00
|
|
|
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);
|
2009-08-30 04:26:50 +02:00
|
|
|
|
|
|
|
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()));
|
|
|
|
}
|