Commit eb121ba2 authored by Thomas G. Lockhart's avatar Thomas G. Lockhart

Implement IS OF and IS NOT OF type predicate.

 Can now do queries of the form: SELECT value IS OF (integer, float8);
Define macros for handling typmod manipulation for date/time types.
 Should be more robust than all of that brute-force inline code.
Rename macros for masking and typmod manipulation to put TIMESTAMP_
 or INTERVAL_ in front of the macro name, to reduce the possibility
 of name space collisions.
Allow bit string constants without fully-specified length declaration.
Try implementing CREATE TABLE/OF as a mapping to inheritance.
 May be appropriate, or may be replace later with something more exactly
 like one might expect from databases without the feature.
parent 7c1e67bd
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/parser/gram.y,v 2.353 2002/08/04 04:31:44 momjian Exp $ * $Header: /cvsroot/pgsql/src/backend/parser/gram.y,v 2.354 2002/08/04 06:51:23 thomas Exp $
* *
* HISTORY * HISTORY
* AUTHOR DATE MAJOR EVENT * AUTHOR DATE MAJOR EVENT
...@@ -70,7 +70,6 @@ static bool QueryIsRule = FALSE; ...@@ -70,7 +70,6 @@ static bool QueryIsRule = FALSE;
static Oid *param_type_info; static Oid *param_type_info;
static int pfunc_num_args; static int pfunc_num_args;
/* /*
* If you need access to certain yacc-generated variables and find that * If you need access to certain yacc-generated variables and find that
* they're static by default, uncomment the next line. (this is not a * they're static by default, uncomment the next line. (this is not a
...@@ -97,8 +96,6 @@ static Node *makeSetOp(SetOperation op, bool all, Node *larg, Node *rarg); ...@@ -97,8 +96,6 @@ static Node *makeSetOp(SetOperation op, bool all, Node *larg, Node *rarg);
static Node *doNegate(Node *n); static Node *doNegate(Node *n);
static void doNegateFloat(Value *v); static void doNegateFloat(Value *v);
#define MASK(b) (1 << (b))
%} %}
...@@ -265,7 +262,7 @@ static void doNegateFloat(Value *v); ...@@ -265,7 +262,7 @@ static void doNegateFloat(Value *v);
%type <node> def_arg, columnElem, where_clause, insert_column_item, %type <node> def_arg, columnElem, where_clause, insert_column_item,
a_expr, b_expr, c_expr, r_expr, AexprConst, a_expr, b_expr, c_expr, r_expr, AexprConst,
in_expr, having_clause, func_table in_expr, having_clause, func_table
%type <list> row, row_descriptor, row_list, in_expr_nodes %type <list> row, row_descriptor, row_list, in_expr_nodes, type_list
%type <node> case_expr, case_arg, when_clause, case_default %type <node> case_expr, case_arg, when_clause, case_default
%type <list> when_clause_list %type <list> when_clause_list
%type <ival> sub_type %type <ival> sub_type
...@@ -282,8 +279,11 @@ static void doNegateFloat(Value *v); ...@@ -282,8 +279,11 @@ static void doNegateFloat(Value *v);
%type <target> target_el, insert_target_el, update_target_el %type <target> target_el, insert_target_el, update_target_el
%type <typnam> Typename, SimpleTypename, ConstTypename, %type <typnam> Typename, SimpleTypename, ConstTypename,
GenericType, Numeric, opt_float, Character, GenericType, Numeric, opt_float,
ConstDatetime, ConstInterval, Bit Character, ConstCharacter,
CharacterWithLength, CharacterWithoutLength,
ConstDatetime, ConstInterval,
Bit, ConstBit, BitWithLength, BitWithoutLength
%type <str> character %type <str> character
%type <str> extract_arg %type <str> extract_arg
%type <str> opt_charset, opt_collate %type <str> opt_charset, opt_collate
...@@ -404,7 +404,7 @@ static void doNegateFloat(Value *v); ...@@ -404,7 +404,7 @@ static void doNegateFloat(Value *v);
%token UNIONJOIN %token UNIONJOIN
/* Special keywords, not in the query language - see the "lex" file */ /* Special keywords, not in the query language - see the "lex" file */
%token <str> IDENT, FCONST, SCONST, BITCONST, Op %token <str> IDENT, FCONST, SCONST, NCONST, BCONST, XCONST, Op
%token <ival> ICONST, PARAM %token <ival> ICONST, PARAM
/* these are not real. they are here so that they get generated as #define's*/ /* these are not real. they are here so that they get generated as #define's*/
...@@ -943,12 +943,13 @@ zone_value: ...@@ -943,12 +943,13 @@ zone_value:
| ConstInterval Sconst opt_interval | ConstInterval Sconst opt_interval
{ {
A_Const *n = (A_Const *) makeStringConst($2, $1); A_Const *n = (A_Const *) makeStringConst($2, $1);
if ($3 != -1) if ($3 != INTERVAL_FULL_RANGE)
{ {
if (($3 & ~(MASK(HOUR) | MASK(MINUTE))) != 0) if (($3 & ~(INTERVAL_MASK(HOUR) | INTERVAL_MASK(MINUTE))) != 0)
elog(ERROR, elog(ERROR,
"Time zone interval must be HOUR or HOUR TO MINUTE"); "Time zone interval"
n->typename->typmod = ((($3 & 0x7FFF) << 16) | 0xFFFF); " must be HOUR or HOUR TO MINUTE");
n->typename->typmod = INTERVAL_TYPMOD(INTERVAL_FULL_PRECISION, $3);
} }
$$ = (Node *)n; $$ = (Node *)n;
} }
...@@ -959,17 +960,14 @@ zone_value: ...@@ -959,17 +960,14 @@ zone_value:
elog(ERROR, elog(ERROR,
"INTERVAL(%d) precision must be between %d and %d", "INTERVAL(%d) precision must be between %d and %d",
$3, 0, MAX_INTERVAL_PRECISION); $3, 0, MAX_INTERVAL_PRECISION);
if ($6 != -1)
{ if (($6 != INTERVAL_FULL_RANGE)
if (($6 & ~(MASK(HOUR) | MASK(MINUTE))) != 0) && (($6 & ~(INTERVAL_MASK(HOUR) | INTERVAL_MASK(MINUTE))) != 0))
elog(ERROR, elog(ERROR,
"Time zone interval must be HOUR or HOUR TO MINUTE"); "Time zone interval"
n->typename->typmod = ((($6 & 0x7FFF) << 16) | $3); " must be HOUR or HOUR TO MINUTE");
}
else n->typename->typmod = INTERVAL_TYPMOD($3, $6);
{
n->typename->typmod = ((0x7FFF << 16) | $3);
}
$$ = (Node *)n; $$ = (Node *)n;
} }
...@@ -1388,6 +1386,21 @@ CreateStmt: CREATE OptTemp TABLE qualified_name '(' OptTableElementList ')' ...@@ -1388,6 +1386,21 @@ CreateStmt: CREATE OptTemp TABLE qualified_name '(' OptTableElementList ')'
n->hasoids = $9; n->hasoids = $9;
$$ = (Node *)n; $$ = (Node *)n;
} }
| CREATE OptTemp TABLE qualified_name OF qualified_name
'(' OptTableElementList ')' OptWithOids
{
/* SQL99 CREATE TABLE OF <UDT> (cols) seems to be satisfied
* by our inheritance capabilities. Let's try it...
*/
CreateStmt *n = makeNode(CreateStmt);
$4->istemp = $2;
n->relation = $4;
n->tableElts = $8;
n->inhRelations = makeList1($6);
n->constraints = NIL;
n->hasoids = $10;
$$ = (Node *)n;
}
; ;
/* /*
...@@ -2100,7 +2113,8 @@ TriggerFuncArg: ...@@ -2100,7 +2113,8 @@ TriggerFuncArg:
} }
| FCONST { $$ = makeString($1); } | FCONST { $$ = makeString($1); }
| Sconst { $$ = makeString($1); } | Sconst { $$ = makeString($1); }
| BITCONST { $$ = makeString($1); } | BCONST { $$ = makeString($1); }
| XCONST { $$ = makeString($1); }
| ColId { $$ = makeString($1); } | ColId { $$ = makeString($1); }
; ;
...@@ -4651,6 +4665,13 @@ relation_expr: ...@@ -4651,6 +4665,13 @@ relation_expr:
$$->inhOpt = INH_NO; $$->inhOpt = INH_NO;
$$->alias = NULL; $$->alias = NULL;
} }
| ONLY '(' qualified_name ')'
{
/* no inheritance, SQL99-style syntax */
$$ = $3;
$$->inhOpt = INH_NO;
$$->alias = NULL;
}
; ;
...@@ -4722,12 +4743,16 @@ opt_array_bounds: ...@@ -4722,12 +4743,16 @@ opt_array_bounds:
* for. FIXME later. * for. FIXME later.
*/ */
SimpleTypename: SimpleTypename:
ConstTypename { $$ = $1; } GenericType { $$ = $1; }
| Numeric { $$ = $1; }
| Bit { $$ = $1; }
| Character { $$ = $1; }
| ConstDatetime { $$ = $1; }
| ConstInterval opt_interval | ConstInterval opt_interval
{ {
$$ = $1; $$ = $1;
if ($2 != -1) if ($2 != INTERVAL_FULL_RANGE)
$$->typmod = ((($2 & 0x7FFF) << 16) | 0xFFFF); $$->typmod = INTERVAL_TYPMOD(INTERVAL_FULL_PRECISION, $2);
} }
| ConstInterval '(' Iconst ')' opt_interval | ConstInterval '(' Iconst ')' opt_interval
{ {
...@@ -4736,7 +4761,7 @@ SimpleTypename: ...@@ -4736,7 +4761,7 @@ SimpleTypename:
elog(ERROR, elog(ERROR,
"INTERVAL(%d) precision must be between %d and %d", "INTERVAL(%d) precision must be between %d and %d",
$3, 0, MAX_INTERVAL_PRECISION); $3, 0, MAX_INTERVAL_PRECISION);
$$->typmod = ((($5 & 0x7FFF) << 16) | $3); $$->typmod = INTERVAL_TYPMOD($3, $5);
} }
| type_name attrs | type_name attrs
{ {
...@@ -4746,12 +4771,21 @@ SimpleTypename: ...@@ -4746,12 +4771,21 @@ SimpleTypename:
} }
; ;
/* We have a separate ConstTypename to allow defaulting fixed-length
* types such as CHAR() and BIT() to an unspecified length.
* SQL9x requires that these default to a length of one, but this
* makes no sense for constructs like CHAR 'hi' and BIT '0101',
* where there is an obvious better choice to make.
* Note that ConstInterval is not included here since it must
* be pushed up higher in the rules to accomodate the postfix
* options (e.g. INTERVAL '1' YEAR).
*/
ConstTypename: ConstTypename:
GenericType { $$ = $1;} GenericType { $$ = $1; }
| Numeric { $$ = $1;} | Numeric { $$ = $1; }
| Bit { $$ = $1;} | ConstBit { $$ = $1; }
| Character { $$ = $1;} | ConstCharacter { $$ = $1; }
| ConstDatetime { $$ = $1;} | ConstDatetime { $$ = $1; }
; ;
GenericType: GenericType:
...@@ -4899,7 +4933,29 @@ opt_decimal: ...@@ -4899,7 +4933,29 @@ opt_decimal:
* SQL92 bit-field data types * SQL92 bit-field data types
* The following implements BIT() and BIT VARYING(). * The following implements BIT() and BIT VARYING().
*/ */
Bit: BIT opt_varying '(' Iconst ')' Bit: BitWithLength
{
$$ = $1;
}
| BitWithoutLength
{
$$ = $1;
}
;
ConstBit: BitWithLength
{
$$ = $1;
}
| BitWithoutLength
{
$$->typmod = -1;
$$ = $1;
}
;
BitWithLength:
BIT opt_varying '(' Iconst ')'
{ {
char *typname; char *typname;
...@@ -4913,7 +4969,10 @@ Bit: BIT opt_varying '(' Iconst ')' ...@@ -4913,7 +4969,10 @@ Bit: BIT opt_varying '(' Iconst ')'
typname, (MaxAttrSize * BITS_PER_BYTE)); typname, (MaxAttrSize * BITS_PER_BYTE));
$$->typmod = $4; $$->typmod = $4;
} }
| BIT opt_varying ;
BitWithoutLength:
BIT opt_varying
{ {
/* bit defaults to bit(1), varbit to no limit */ /* bit defaults to bit(1), varbit to no limit */
if ($2) if ($2)
...@@ -4934,7 +4993,34 @@ Bit: BIT opt_varying '(' Iconst ')' ...@@ -4934,7 +4993,34 @@ Bit: BIT opt_varying '(' Iconst ')'
* SQL92 character data types * SQL92 character data types
* The following implements CHAR() and VARCHAR(). * The following implements CHAR() and VARCHAR().
*/ */
Character: character '(' Iconst ')' opt_charset Character: CharacterWithLength
{
$$ = $1;
}
| CharacterWithoutLength
{
$$ = $1;
}
;
ConstCharacter: CharacterWithLength
{
$$ = $1;
}
| CharacterWithoutLength
{
/* Length was not specified so allow to be unrestricted.
* This handles problems with fixed-length (bpchar) strings
* which in column definitions must default to a length
* of one, but should not be constrained if the length
* was not specified.
*/
$1->typmod = -1;
$$ = $1;
}
;
CharacterWithLength: character '(' Iconst ')' opt_charset
{ {
if (($5 != NULL) && (strcmp($5, "sql_text") != 0)) if (($5 != NULL) && (strcmp($5, "sql_text") != 0))
{ {
...@@ -4963,7 +5049,9 @@ Character: character '(' Iconst ')' opt_charset ...@@ -4963,7 +5049,9 @@ Character: character '(' Iconst ')' opt_charset
*/ */
$$->typmod = VARHDRSZ + $3; $$->typmod = VARHDRSZ + $3;
} }
| character opt_charset ;
CharacterWithoutLength: character opt_charset
{ {
if (($2 != NULL) && (strcmp($2, "sql_text") != 0)) if (($2 != NULL) && (strcmp($2, "sql_text") != 0))
{ {
...@@ -5090,27 +5178,30 @@ opt_timezone: ...@@ -5090,27 +5178,30 @@ opt_timezone:
; ;
opt_interval: opt_interval:
YEAR_P { $$ = MASK(YEAR); } YEAR_P { $$ = INTERVAL_MASK(YEAR); }
| MONTH_P { $$ = MASK(MONTH); } | MONTH_P { $$ = INTERVAL_MASK(MONTH); }
| DAY_P { $$ = MASK(DAY); } | DAY_P { $$ = INTERVAL_MASK(DAY); }
| HOUR_P { $$ = MASK(HOUR); } | HOUR_P { $$ = INTERVAL_MASK(HOUR); }
| MINUTE_P { $$ = MASK(MINUTE); } | MINUTE_P { $$ = INTERVAL_MASK(MINUTE); }
| SECOND_P { $$ = MASK(SECOND); } | SECOND_P { $$ = INTERVAL_MASK(SECOND); }
| YEAR_P TO MONTH_P | YEAR_P TO MONTH_P
{ $$ = MASK(YEAR) | MASK(MONTH); } { $$ = INTERVAL_MASK(YEAR) | INTERVAL_MASK(MONTH); }
| DAY_P TO HOUR_P | DAY_P TO HOUR_P
{ $$ = MASK(DAY) | MASK(HOUR); } { $$ = INTERVAL_MASK(DAY) | INTERVAL_MASK(HOUR); }
| DAY_P TO MINUTE_P | DAY_P TO MINUTE_P
{ $$ = MASK(DAY) | MASK(HOUR) | MASK(MINUTE); } { $$ = INTERVAL_MASK(DAY) | INTERVAL_MASK(HOUR)
| INTERVAL_MASK(MINUTE); }
| DAY_P TO SECOND_P | DAY_P TO SECOND_P
{ $$ = MASK(DAY) | MASK(HOUR) | MASK(MINUTE) | MASK(SECOND); } { $$ = INTERVAL_MASK(DAY) | INTERVAL_MASK(HOUR)
| INTERVAL_MASK(MINUTE) | INTERVAL_MASK(SECOND); }
| HOUR_P TO MINUTE_P | HOUR_P TO MINUTE_P
{ $$ = MASK(HOUR) | MASK(MINUTE); } { $$ = INTERVAL_MASK(HOUR) | INTERVAL_MASK(MINUTE); }
| HOUR_P TO SECOND_P | HOUR_P TO SECOND_P
{ $$ = MASK(HOUR) | MASK(MINUTE) | MASK(SECOND); } { $$ = INTERVAL_MASK(HOUR) | INTERVAL_MASK(MINUTE)
| INTERVAL_MASK(SECOND); }
| MINUTE_P TO SECOND_P | MINUTE_P TO SECOND_P
{ $$ = MASK(MINUTE) | MASK(SECOND); } { $$ = INTERVAL_MASK(MINUTE) | INTERVAL_MASK(SECOND); }
| /*EMPTY*/ { $$ = -1; } | /*EMPTY*/ { $$ = INTERVAL_FULL_RANGE; }
; ;
...@@ -5510,6 +5601,14 @@ a_expr: c_expr { $$ = $1; } ...@@ -5510,6 +5601,14 @@ a_expr: c_expr { $$ = $1; }
} }
| a_expr IS DISTINCT FROM a_expr %prec DISTINCT | a_expr IS DISTINCT FROM a_expr %prec DISTINCT
{ $$ = (Node *) makeSimpleA_Expr(DISTINCT, "=", $1, $5); } { $$ = (Node *) makeSimpleA_Expr(DISTINCT, "=", $1, $5); }
| a_expr IS OF '(' type_list ')'
{
$$ = (Node *) makeSimpleA_Expr(OF, "=", $1, (Node *) $5);
}
| a_expr IS NOT OF '(' type_list ')'
{
$$ = (Node *) makeSimpleA_Expr(OF, "!=", $1, (Node *) $6);
}
| a_expr BETWEEN b_expr AND b_expr %prec BETWEEN | a_expr BETWEEN b_expr AND b_expr %prec BETWEEN
{ {
$$ = (Node *) makeA_Expr(AND, NIL, $$ = (Node *) makeA_Expr(AND, NIL,
...@@ -5588,6 +5687,19 @@ a_expr: c_expr { $$ = $1; } ...@@ -5588,6 +5687,19 @@ a_expr: c_expr { $$ = $1; }
n->subselect = $4; n->subselect = $4;
$$ = (Node *)n; $$ = (Node *)n;
} }
| UNIQUE select_with_parens %prec Op
{
/* Not sure how to get rid of the parentheses
* but there are lots of shift/reduce errors without them.
*
* Should be able to implement this by plopping the entire
* select into a node, then transforming the target expressions
* from whatever they are into count(*), and testing the
* entire result equal to one.
* But, will probably implement a separate node in the executor.
*/
elog(ERROR, "UNIQUE predicate is not yet implemented");
}
| r_expr | r_expr
{ $$ = $1; } { $$ = $1; }
; ;
...@@ -5643,6 +5755,14 @@ b_expr: c_expr ...@@ -5643,6 +5755,14 @@ b_expr: c_expr
{ $$ = (Node *) makeA_Expr(OP, $2, $1, NULL); } { $$ = (Node *) makeA_Expr(OP, $2, $1, NULL); }
| b_expr IS DISTINCT FROM b_expr %prec Op | b_expr IS DISTINCT FROM b_expr %prec Op
{ $$ = (Node *) makeSimpleA_Expr(DISTINCT, "=", $1, $5); } { $$ = (Node *) makeSimpleA_Expr(DISTINCT, "=", $1, $5); }
| b_expr IS OF '(' type_list ')'
{
$$ = (Node *) makeSimpleA_Expr(OF, "=", $1, (Node *) $5);
}
| b_expr IS NOT OF '(' type_list ')'
{
$$ = (Node *) makeSimpleA_Expr(OF, "!=", $1, (Node *) $6);
}
; ;
/* /*
...@@ -6139,6 +6259,16 @@ extract_list: ...@@ -6139,6 +6259,16 @@ extract_list:
| /*EMPTY*/ { $$ = NIL; } | /*EMPTY*/ { $$ = NIL; }
; ;
type_list: type_list ',' Typename
{
$$ = lappend($1, $3);
}
| Typename
{
$$ = makeList1($1);
}
;
/* Allow delimited string SCONST in extract_arg as an SQL extension. /* Allow delimited string SCONST in extract_arg as an SQL extension.
* - thomas 2001-04-12 * - thomas 2001-04-12
*/ */
...@@ -6186,6 +6316,7 @@ position_list: ...@@ -6186,6 +6316,7 @@ position_list:
* SQL9x defines a specific syntax for arguments to SUBSTRING(): * SQL9x defines a specific syntax for arguments to SUBSTRING():
* o substring(text from int for int) * o substring(text from int for int)
* o substring(text from int) get entire string from starting point "int" * o substring(text from int) get entire string from starting point "int"
* o substring(text from pattern) get entire string matching pattern
* o substring(text for int) get first "int" characters of string * o substring(text for int) get first "int" characters of string
* We also want to implement generic substring functions which accept * We also want to implement generic substring functions which accept
* the usual generic list of arguments. So we will accept both styles * the usual generic list of arguments. So we will accept both styles
...@@ -6511,8 +6642,8 @@ func_name: function_name ...@@ -6511,8 +6642,8 @@ func_name: function_name
; ;
/* Constants /*
* Include TRUE/FALSE for SQL3 support. - thomas 1997-10-24 * Constants
*/ */
AexprConst: Iconst AexprConst: Iconst
{ {
...@@ -6535,19 +6666,25 @@ AexprConst: Iconst ...@@ -6535,19 +6666,25 @@ AexprConst: Iconst
n->val.val.str = $1; n->val.val.str = $1;
$$ = (Node *)n; $$ = (Node *)n;
} }
| BITCONST | BCONST
{ {
A_Const *n = makeNode(A_Const); A_Const *n = makeNode(A_Const);
n->val.type = T_BitString; n->val.type = T_BitString;
n->val.val.str = $1; n->val.val.str = $1;
$$ = (Node *)n; $$ = (Node *)n;
} }
/* This rule formerly used Typename, | XCONST
* but that causes reduce conflicts with subscripted column names. {
* Now, separate into ConstTypename and ConstInterval, /* This is a bit constant per SQL99:
* to allow implementing the SQL92 syntax for INTERVAL literals. * Without Feature F511, "BIT data type",
* - thomas 2000-06-24 * a <general literal> shall not be a
* <bit string literal> or a <hex string literal>.
*/ */
A_Const *n = makeNode(A_Const);
n->val.type = T_BitString;
n->val.val.str = $1;
$$ = (Node *)n;
}
| ConstTypename Sconst | ConstTypename Sconst
{ {
A_Const *n = makeNode(A_Const); A_Const *n = makeNode(A_Const);
...@@ -6563,8 +6700,8 @@ AexprConst: Iconst ...@@ -6563,8 +6700,8 @@ AexprConst: Iconst
n->val.type = T_String; n->val.type = T_String;
n->val.val.str = $2; n->val.val.str = $2;
/* precision is not specified, but fields may be... */ /* precision is not specified, but fields may be... */
if ($3 != -1) if ($3 != INTERVAL_FULL_RANGE)
n->typename->typmod = ((($3 & 0x7FFF) << 16) | 0xFFFF); n->typename->typmod = INTERVAL_TYPMOD(INTERVAL_FULL_PRECISION, $3);
$$ = (Node *)n; $$ = (Node *)n;
} }
| ConstInterval '(' Iconst ')' Sconst opt_interval | ConstInterval '(' Iconst ')' Sconst opt_interval
...@@ -6578,7 +6715,7 @@ AexprConst: Iconst ...@@ -6578,7 +6715,7 @@ AexprConst: Iconst
elog(ERROR, elog(ERROR,
"INTERVAL(%d) precision must be between %d and %d", "INTERVAL(%d) precision must be between %d and %d",
$3, 0, MAX_INTERVAL_PRECISION); $3, 0, MAX_INTERVAL_PRECISION);
n->typename->typmod = ((($6 & 0x7FFF) << 16) | $3); n->typename->typmod = INTERVAL_TYPMOD($3, $6);
$$ = (Node *)n; $$ = (Node *)n;
} }
| PARAM opt_indirection | PARAM opt_indirection
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment