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): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef DSQL_PARSER_H
|
|
|
|
#define DSQL_PARSER_H
|
|
|
|
|
|
|
|
#include "../dsql/dsql.h"
|
2008-05-19 15:47:48 +02:00
|
|
|
#include "../dsql/DdlNodes.h"
|
2010-09-17 05:15:32 +02:00
|
|
|
#include "../dsql/BoolNodes.h"
|
2010-02-13 21:29:29 +01:00
|
|
|
#include "../dsql/ExprNodes.h"
|
|
|
|
#include "../dsql/AggNodes.h"
|
2010-02-14 20:08:22 +01:00
|
|
|
#include "../dsql/WinNodes.h"
|
2009-10-21 02:42:38 +02:00
|
|
|
#include "../dsql/PackageNodes.h"
|
2008-05-19 15:47:48 +02:00
|
|
|
#include "../dsql/StmtNodes.h"
|
2011-01-09 22:58:56 +01:00
|
|
|
#include "../jrd/RecordSourceNodes.h"
|
2010-07-29 02:20:53 +02:00
|
|
|
#include "../common/classes/Nullable.h"
|
2008-04-15 21:45:19 +02:00
|
|
|
#include "../common/classes/stack.h"
|
2008-02-28 14:48:16 +01:00
|
|
|
|
2010-10-12 10:02:57 +02:00
|
|
|
#include "gen/parse.h"
|
2009-10-21 02:42:38 +02:00
|
|
|
|
2008-02-28 14:48:16 +01:00
|
|
|
namespace Jrd {
|
|
|
|
|
2008-04-15 04:18:38 +02:00
|
|
|
class Parser : public Firebird::PermanentStorage
|
2008-02-03 20:16:12 +01:00
|
|
|
{
|
|
|
|
private:
|
2011-03-07 19:40:04 +01:00
|
|
|
// User-defined text position type.
|
|
|
|
struct Position
|
|
|
|
{
|
2013-01-25 13:34:19 +01:00
|
|
|
ULONG firstLine;
|
|
|
|
ULONG firstColumn;
|
|
|
|
ULONG lastLine;
|
|
|
|
ULONG lastColumn;
|
2011-03-07 19:40:04 +01:00
|
|
|
const char* firstPos;
|
|
|
|
const char* lastPos;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef Position YYPOSN;
|
2008-02-03 20:16:12 +01:00
|
|
|
typedef int Yshort;
|
|
|
|
|
|
|
|
struct yyparsestate
|
|
|
|
{
|
|
|
|
yyparsestate* save; // Previously saved parser state
|
|
|
|
int state;
|
|
|
|
int errflag;
|
|
|
|
Yshort* ssp; // state stack pointer
|
|
|
|
YYSTYPE* vsp; // value stack pointer
|
|
|
|
YYPOSN* psp; // position stack pointer
|
|
|
|
YYSTYPE val; // value as returned by actions
|
|
|
|
YYPOSN pos; // position as returned by universal action
|
|
|
|
Yshort* ss; // state stack base
|
|
|
|
YYSTYPE* vs; // values stack base
|
|
|
|
YYPOSN* ps; // position stack base
|
|
|
|
int lexeme; // index of the conflict lexeme in the lexical queue
|
|
|
|
unsigned int stacksize; // current maximum stack size
|
|
|
|
Yshort ctry; // index in yyctable[] for this conflict
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LexerState
|
|
|
|
{
|
|
|
|
// This is, in fact, parser state. Not used in lexer itself
|
|
|
|
int dsql_debug;
|
2008-12-05 01:56:15 +01:00
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
// Actual lexer state begins from here
|
2008-04-15 21:45:19 +02:00
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
const TEXT* ptr;
|
|
|
|
const TEXT* end;
|
|
|
|
const TEXT* last_token;
|
2009-08-30 04:26:50 +02:00
|
|
|
const TEXT* start;
|
2008-02-03 20:16:12 +01:00
|
|
|
const TEXT* line_start;
|
|
|
|
const TEXT* last_token_bk;
|
|
|
|
const TEXT* line_start_bk;
|
2013-01-25 13:34:19 +01:00
|
|
|
SSHORT att_charset;
|
|
|
|
SLONG lines, lines_bk;
|
2008-02-03 20:16:12 +01:00
|
|
|
int prev_keyword;
|
|
|
|
USHORT param_number;
|
|
|
|
};
|
|
|
|
|
2009-12-19 02:32:00 +01:00
|
|
|
struct StrMark
|
2009-08-30 04:26:50 +02:00
|
|
|
{
|
2009-12-19 02:32:00 +01:00
|
|
|
StrMark()
|
|
|
|
: introduced(false),
|
|
|
|
pos(0),
|
|
|
|
length(0),
|
|
|
|
str(NULL)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator >(const StrMark& o) const
|
|
|
|
{
|
|
|
|
return pos > o.pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool introduced;
|
2009-08-30 04:26:50 +02:00
|
|
|
unsigned pos;
|
|
|
|
unsigned length;
|
2012-12-25 18:34:50 +01:00
|
|
|
IntlString* str;
|
2009-08-30 04:26:50 +02:00
|
|
|
};
|
|
|
|
|
2012-12-10 15:42:56 +01:00
|
|
|
public:
|
|
|
|
static const int MAX_TOKEN_LEN = 256;
|
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
public:
|
2012-12-25 18:34:50 +01:00
|
|
|
Parser(MemoryPool& pool, DsqlCompilerScratch* aScratch, USHORT aClientDialect,
|
|
|
|
USHORT aDbDialect, USHORT aParserVersion, const TEXT* string, size_t length,
|
|
|
|
SSHORT characterSet);
|
2008-02-03 20:16:12 +01:00
|
|
|
~Parser();
|
|
|
|
|
|
|
|
public:
|
2012-02-25 20:56:37 +01:00
|
|
|
dsql_req* parse();
|
2008-02-03 20:16:12 +01:00
|
|
|
|
2009-09-02 06:23:02 +02:00
|
|
|
const Firebird::string& getTransformedString() const
|
2009-08-30 04:26:50 +02:00
|
|
|
{
|
|
|
|
return transformedString;
|
|
|
|
}
|
|
|
|
|
2009-01-08 10:26:06 +01:00
|
|
|
bool isStmtAmbiguous() const
|
2008-02-03 20:16:12 +01:00
|
|
|
{
|
|
|
|
return stmt_ambiguous;
|
|
|
|
}
|
|
|
|
|
2012-12-25 18:34:50 +01:00
|
|
|
Firebird::string* newString(const Firebird::string& s)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return FB_NEW_POOL(getPool()) Firebird::string(getPool(), s);
|
2012-12-25 18:34:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
IntlString* newIntlString(const Firebird::string& s, const char* charSet = NULL)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return FB_NEW_POOL(getPool()) IntlString(getPool(), s, charSet);
|
2012-12-25 18:34:50 +01:00
|
|
|
}
|
|
|
|
|
2010-10-30 20:57:53 +02:00
|
|
|
// newNode overloads
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
T* newNode()
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool()));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename T1>
|
|
|
|
T* newNode(T1 a1)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool(), a1));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename T1, typename T2>
|
|
|
|
T* newNode(T1 a1, T2 a2)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool(), a1, a2));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename T1, typename T2, typename T3>
|
|
|
|
T* newNode(T1 a1, T2 a2, T3 a3)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool(), a1, a2, a3));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename T1, typename T2, typename T3, typename T4>
|
|
|
|
T* newNode(T1 a1, T2 a2, T3 a3, T4 a4)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool(), a1, a2, a3, a4));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename T1, typename T2, typename T3, typename T4, typename T5>
|
|
|
|
T* newNode(T1 a1, T2 a2, T3 a3, T4 a4, T5 a5)
|
|
|
|
{
|
2015-10-12 16:26:00 +02:00
|
|
|
return setupNode<T>(FB_NEW_POOL(getPool()) T(getPool(), a1, a2, a3, a4, a5));
|
2010-10-30 20:57:53 +02:00
|
|
|
}
|
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
private:
|
2012-03-25 03:08:55 +02:00
|
|
|
template <typename T> T* setupNode(Node* node)
|
|
|
|
{
|
2013-01-25 13:34:19 +01:00
|
|
|
node->line = (ULONG) lex.lines_bk;
|
|
|
|
node->column = (ULONG) (lex.last_token_bk - lex.line_start_bk + 1);
|
2012-03-25 03:08:55 +02:00
|
|
|
return static_cast<T*>(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overload to not make LineColumnNode data wrong after its construction.
|
|
|
|
template <typename T> T* setupNode(LineColumnNode* node)
|
|
|
|
{
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overload for non-Node classes constructed with newNode.
|
|
|
|
template <typename T> T* setupNode(void* node)
|
|
|
|
{
|
|
|
|
return static_cast<T*>(node);
|
|
|
|
}
|
|
|
|
|
2013-08-14 04:20:17 +02:00
|
|
|
BoolExprNode* valueToBool(ValueExprNode* value)
|
|
|
|
{
|
|
|
|
BoolAsValueNode* node = value->as<BoolAsValueNode>();
|
|
|
|
if (node)
|
|
|
|
return node->boolean;
|
2013-12-29 01:35:50 +01:00
|
|
|
|
|
|
|
ComparativeBoolNode* cmpNode = newNode<ComparativeBoolNode>(
|
|
|
|
blr_eql, value, MAKE_constant("1", CONSTANT_BOOLEAN));
|
2016-04-04 19:49:44 +02:00
|
|
|
cmpNode->dsqlCheckBoolean = true;
|
|
|
|
|
2013-12-29 01:35:50 +01:00
|
|
|
return cmpNode;
|
2013-08-14 04:20:17 +02:00
|
|
|
}
|
|
|
|
|
2012-12-10 15:42:56 +01:00
|
|
|
void yyReducePosn(YYPOSN& ret, YYPOSN* termPosns, YYSTYPE* termVals,
|
|
|
|
int termNo, int stkPos, int yychar, YYPOSN& yyposn, void*);
|
|
|
|
|
|
|
|
int yylex();
|
2014-04-04 18:05:20 +02:00
|
|
|
bool yylexSkipSpaces();
|
2012-12-10 15:42:56 +01:00
|
|
|
int yylexAux();
|
|
|
|
|
|
|
|
void yyerror(const TEXT* error_string);
|
|
|
|
void yyerror_detailed(const TEXT* error_string, int yychar, YYSTYPE&, YYPOSN&);
|
2014-04-12 08:35:36 +02:00
|
|
|
void yyerrorIncompleteCmd();
|
2012-12-10 15:42:56 +01:00
|
|
|
|
|
|
|
void check_bound(const char* const to, const char* const string);
|
|
|
|
void check_copy_incr(char*& to, const char ch, const char* const string);
|
|
|
|
|
|
|
|
void yyabandon(SLONG, ISC_STATUS);
|
|
|
|
|
2012-12-25 18:34:50 +01:00
|
|
|
Firebird::MetaName optName(Firebird::MetaName* name)
|
|
|
|
{
|
|
|
|
return (name ? *name : Firebird::MetaName());
|
|
|
|
}
|
2012-12-10 15:42:56 +01:00
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
void transformString(const char* start, unsigned length, Firebird::string& dest);
|
2012-12-25 18:34:50 +01:00
|
|
|
Firebird::string makeParseStr(const Position& p1, const Position& p2);
|
2012-12-10 15:42:56 +01:00
|
|
|
ParameterNode* make_parameter();
|
2009-08-30 04:26:50 +02:00
|
|
|
|
2010-07-06 02:49:33 +02:00
|
|
|
// Set the value of a clause, checking if it was already specified.
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
void setClause(T& clause, const char* duplicateMsg, const T& value)
|
|
|
|
{
|
2012-04-25 03:42:47 +02:00
|
|
|
checkDuplicateClause(clause, duplicateMsg);
|
2010-07-06 02:49:33 +02:00
|
|
|
clause = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, typename Delete>
|
|
|
|
void setClause(Firebird::AutoPtr<T, Delete>& clause, const char* duplicateMsg, T* value)
|
|
|
|
{
|
2012-04-25 03:42:47 +02:00
|
|
|
checkDuplicateClause(clause, duplicateMsg);
|
2010-07-06 02:49:33 +02:00
|
|
|
clause = value;
|
|
|
|
}
|
|
|
|
|
2010-07-28 18:14:20 +02:00
|
|
|
template <typename T>
|
2010-07-29 02:20:53 +02:00
|
|
|
void setClause(Nullable<T>& clause, const char* duplicateMsg, const T& value)
|
2012-04-25 03:42:47 +02:00
|
|
|
{
|
|
|
|
checkDuplicateClause(clause, duplicateMsg);
|
|
|
|
clause = value;
|
|
|
|
}
|
|
|
|
|
2012-05-03 18:43:29 +02:00
|
|
|
template <typename T1, typename T2>
|
|
|
|
void setClause(NestConst<T1>& clause, const char* duplicateMsg, const T2& value)
|
|
|
|
{
|
|
|
|
setClause(*clause.getAddress(), duplicateMsg, value);
|
|
|
|
}
|
|
|
|
|
2012-04-25 03:42:47 +02:00
|
|
|
void setClause(bool& clause, const char* duplicateMsg)
|
|
|
|
{
|
|
|
|
setClause(clause, duplicateMsg, true);
|
|
|
|
}
|
|
|
|
|
2015-12-09 14:47:03 +01:00
|
|
|
void setClauseFlag(unsigned& clause, const unsigned flag, const char* duplicateMsg)
|
|
|
|
{
|
|
|
|
using namespace Firebird;
|
|
|
|
if (clause & flag)
|
2015-12-13 02:02:02 +01:00
|
|
|
{
|
2015-12-09 14:47:03 +01:00
|
|
|
status_exception::raise(
|
|
|
|
Arg::Gds(isc_sqlerr) << Arg::Num(-637) <<
|
|
|
|
Arg::Gds(isc_dsql_duplicate_spec) << duplicateMsg);
|
2015-12-13 02:02:02 +01:00
|
|
|
}
|
2015-12-09 14:47:03 +01:00
|
|
|
clause |= flag;
|
|
|
|
}
|
|
|
|
|
2012-04-25 03:42:47 +02:00
|
|
|
template <typename T>
|
|
|
|
void checkDuplicateClause(const T& clause, const char* duplicateMsg)
|
2010-07-28 18:14:20 +02:00
|
|
|
{
|
|
|
|
using namespace Firebird;
|
2012-04-25 03:42:47 +02:00
|
|
|
if (isDuplicateClause(clause))
|
2010-07-28 18:14:20 +02:00
|
|
|
{
|
|
|
|
status_exception::raise(
|
|
|
|
Arg::Gds(isc_sqlerr) << Arg::Num(-637) <<
|
|
|
|
Arg::Gds(isc_dsql_duplicate_spec) << duplicateMsg);
|
|
|
|
}
|
2012-04-25 03:42:47 +02:00
|
|
|
}
|
2010-07-28 18:14:20 +02:00
|
|
|
|
2012-04-25 03:42:47 +02:00
|
|
|
template <typename T>
|
|
|
|
bool isDuplicateClause(const T& clause)
|
|
|
|
{
|
|
|
|
return clause != 0;
|
2010-07-28 18:14:20 +02:00
|
|
|
}
|
|
|
|
|
2012-04-25 03:42:47 +02:00
|
|
|
bool isDuplicateClause(const Firebird::MetaName& clause)
|
2010-07-06 02:49:33 +02:00
|
|
|
{
|
2012-04-25 03:42:47 +02:00
|
|
|
return clause.hasData();
|
2010-07-06 02:49:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
2012-04-25 03:42:47 +02:00
|
|
|
bool isDuplicateClause(const Nullable<T>& clause)
|
2010-07-06 02:49:33 +02:00
|
|
|
{
|
2012-04-25 03:42:47 +02:00
|
|
|
return clause.specified;
|
2010-07-06 02:49:33 +02:00
|
|
|
}
|
|
|
|
|
2012-04-25 03:42:47 +02:00
|
|
|
template <typename T>
|
|
|
|
bool isDuplicateClause(const Firebird::Array<T>& clause)
|
2010-07-06 02:49:33 +02:00
|
|
|
{
|
|
|
|
return clause.hasData();
|
|
|
|
}
|
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
// start - defined in btyacc_fb.ske
|
|
|
|
private:
|
|
|
|
static void yySCopy(YYSTYPE* to, YYSTYPE* from, int size);
|
|
|
|
static void yyPCopy(YYPOSN* to, YYPOSN* from, int size);
|
|
|
|
static void yyFreeState(yyparsestate* p);
|
|
|
|
|
2015-10-13 03:49:11 +02:00
|
|
|
void yyMoreStack(yyparsestate* yyps);
|
|
|
|
yyparsestate* yyNewState(int size);
|
|
|
|
|
2008-02-03 20:16:12 +01:00
|
|
|
private:
|
|
|
|
int parseAux();
|
|
|
|
int yylex1();
|
|
|
|
int yyexpand();
|
|
|
|
// end - defined in btyacc_fb.ske
|
|
|
|
|
|
|
|
private:
|
2012-12-25 18:34:50 +01:00
|
|
|
DsqlCompilerScratch* scratch;
|
2008-02-03 20:16:12 +01:00
|
|
|
USHORT client_dialect;
|
|
|
|
USHORT db_dialect;
|
|
|
|
USHORT parser_version;
|
|
|
|
|
2009-08-30 04:26:50 +02:00
|
|
|
Firebird::string transformedString;
|
2012-12-25 18:34:50 +01:00
|
|
|
Firebird::GenericMap<Firebird::NonPooled<IntlString*, StrMark> > strMarks;
|
2008-02-03 20:16:12 +01:00
|
|
|
bool stmt_ambiguous;
|
2012-02-25 20:56:37 +01:00
|
|
|
dsql_req* DSQL_parse;
|
2008-02-03 20:16:12 +01:00
|
|
|
|
|
|
|
// These value/posn are taken from the lexer
|
|
|
|
YYSTYPE yylval;
|
|
|
|
YYPOSN yyposn;
|
|
|
|
|
|
|
|
// These value/posn of the root non-terminal are returned to the caller
|
|
|
|
YYSTYPE yyretlval;
|
2011-03-07 19:40:04 +01:00
|
|
|
Position yyretposn;
|
2008-02-03 20:16:12 +01:00
|
|
|
|
|
|
|
int yynerrs;
|
|
|
|
|
|
|
|
// Current parser state
|
|
|
|
yyparsestate* yyps;
|
|
|
|
// yypath!=NULL: do the full parse, starting at *yypath parser state.
|
|
|
|
yyparsestate* yypath;
|
|
|
|
// Base of the lexical value queue
|
|
|
|
YYSTYPE* yylvals;
|
|
|
|
// Current posistion at lexical value queue
|
|
|
|
YYSTYPE* yylvp;
|
|
|
|
// End position of lexical value queue
|
|
|
|
YYSTYPE* yylve;
|
|
|
|
// The last allocated position at the lexical value queue
|
|
|
|
YYSTYPE* yylvlim;
|
|
|
|
// Base of the lexical position queue
|
2011-03-07 19:40:04 +01:00
|
|
|
Position* yylpsns;
|
2008-02-03 20:16:12 +01:00
|
|
|
// Current posistion at lexical position queue
|
2011-03-07 19:40:04 +01:00
|
|
|
Position* yylpp;
|
2008-02-03 20:16:12 +01:00
|
|
|
// End position of lexical position queue
|
2011-03-07 19:40:04 +01:00
|
|
|
Position* yylpe;
|
2008-02-03 20:16:12 +01:00
|
|
|
// The last allocated position at the lexical position queue
|
2011-03-07 19:40:04 +01:00
|
|
|
Position* yylplim;
|
2008-02-03 20:16:12 +01:00
|
|
|
// Current position at lexical token queue
|
|
|
|
Yshort* yylexp;
|
|
|
|
Yshort* yylexemes;
|
|
|
|
|
|
|
|
public:
|
|
|
|
LexerState lex;
|
|
|
|
};
|
|
|
|
|
2008-02-29 09:45:02 +01:00
|
|
|
} // namespace
|
2008-02-03 20:16:12 +01:00
|
|
|
|
|
|
|
#endif // DSQL_PARSER_H
|