diff --git a/src/backend/parser/Makefile b/src/backend/parser/Makefile
index 043852f3de66de0a0e1a09bbf461f4fc5fa59dad..d5d4f4372d8780b4ffc3a8e34571107418201f1d 100644
--- a/src/backend/parser/Makefile
+++ b/src/backend/parser/Makefile
@@ -2,7 +2,7 @@
 #
 # Makefile for parser
 #
-# $Header: /cvsroot/pgsql/src/backend/parser/Makefile,v 1.36 2002/03/08 07:12:11 tgl Exp $
+# $Header: /cvsroot/pgsql/src/backend/parser/Makefile,v 1.37 2002/04/20 21:56:14 petere Exp $
 #
 #-------------------------------------------------------------------------
 
@@ -14,6 +14,8 @@ OBJS= analyze.o gram.o keywords.o parser.o parse_agg.o parse_clause.o \
       parse_expr.o parse_func.o parse_node.o parse_oper.o parse_relation.o \
       parse_type.o parse_coerce.o parse_target.o scan.o scansup.o
 
+FLEXFLAGS = -CF
+
 
 all: SUBSYS.o
 
@@ -42,7 +44,7 @@ endif
 
 $(srcdir)/scan.c: scan.l
 ifdef FLEX
-	$(FLEX) $(FLEXFLAGS) -Pbase_yy -o'$@' $<
+	$(FLEX) $(FLEXFLAGS) -o'$@' $<
 else
 	@$(missing) flex $< $@
 endif
@@ -59,13 +61,3 @@ clean:
 	rm -f SUBSYS.o $(OBJS)
 # And the garbage that might have been left behind by partial build:
 	@rm -f y.tab.c y.tab.h lex.yy.c
-
-
-# This is unusual:  We actually have to build some of the parts before
-# we know what the header file dependencies are.  
-dep depend: gram.c scan.c
-	$(CC) -MM $(CFLAGS) *.c >depend
-
-ifeq (depend,$(wildcard depend))
-include depend
-endif
diff --git a/src/backend/parser/parse_type.c b/src/backend/parser/parse_type.c
index a6ba9661951a8a33a90d8db1e7919ac9e69abd5c..0901164389d84d64e3561063d5beb4a090af1984 100644
--- a/src/backend/parser/parse_type.c
+++ b/src/backend/parser/parse_type.c
@@ -8,7 +8,7 @@
  *
  *
  * IDENTIFICATION
- *	  $Header: /cvsroot/pgsql/src/backend/parser/parse_type.c,v 1.39 2002/03/30 01:02:41 tgl Exp $
+ *	  $Header: /cvsroot/pgsql/src/backend/parser/parse_type.c,v 1.40 2002/04/20 21:56:14 petere Exp $
  *
  *-------------------------------------------------------------------------
  */
@@ -473,17 +473,17 @@ typeidTypeRelid(Oid type_id)
 void
 parseTypeString(const char *str, Oid *type_id, int32 *typmod)
 {
-	char	   *buf;
+	StringInfoData buf;
 	List	   *raw_parsetree_list;
 	SelectStmt *stmt;
 	ResTarget  *restarget;
 	A_Const    *aconst;
 	TypeName   *typename;
 
-	buf = (char *) palloc(strlen(str) + 16);
-	sprintf(buf, "SELECT (NULL::%s)", str);
+	initStringInfo(&buf);
+	appendStringInfo(&buf, "SELECT (NULL::%s)", str);
 
-	raw_parsetree_list = parser(buf, NULL, 0);
+	raw_parsetree_list = parser(&buf, NULL, 0);
 
 	/*
 	 * Make sure we got back exactly what we expected and no more;
@@ -528,5 +528,5 @@ parseTypeString(const char *str, Oid *type_id, int32 *typmod)
 	*type_id = typenameTypeId(typename);
 	*typmod = typename->typmod;
 
-	pfree(buf);
+	pfree(buf.data);
 }
diff --git a/src/backend/parser/parser.c b/src/backend/parser/parser.c
index f83d04e69ccca0488473b2f8cb2c12d16e5350e7..4eadbef5f44e64c211946f073cdde8e2afa93834 100644
--- a/src/backend/parser/parser.c
+++ b/src/backend/parser/parser.c
@@ -14,7 +14,7 @@
  * Portions Copyright (c) 1994, Regents of the University of California
  *
  * IDENTIFICATION
- *	  $Header: /cvsroot/pgsql/src/backend/parser/parser.c,v 1.51 2001/11/05 17:46:26 momjian Exp $
+ *	  $Header: /cvsroot/pgsql/src/backend/parser/parser.c,v 1.52 2002/04/20 21:56:14 petere Exp $
  *
  *-------------------------------------------------------------------------
  */
@@ -28,12 +28,6 @@
 #include "parser/parse_expr.h"
 
 
-#if defined(FLEX_SCANNER)
-extern void DeleteBuffer(void);
-#endif   /* FLEX_SCANNER */
-
-char	   *parseString;		/* the char* which holds the string to be
-								 * parsed */
 List	   *parsetree;			/* result of parsing is left here */
 
 static int	lookahead_token;	/* one-token lookahead */
@@ -48,24 +42,20 @@ static bool have_lookahead;		/* lookahead_token set? */
  * Returns a list of raw (un-analyzed) parse trees.
  */
 List *
-parser(char *str, Oid *typev, int nargs)
+parser(StringInfo str, Oid *typev, int nargs)
 {
 	int			yyresult;
 
-	parseString = str;
 	parsetree = NIL;			/* in case parser forgets to set it */
 	have_lookahead = false;
 
-	scanner_init();
+	scanner_init(str);
 	parser_init(typev, nargs);
 	parse_expr_init();
 
 	yyresult = yyparse();
 
-#if defined(FLEX_SCANNER)
-	DeleteBuffer();
-#endif   /* FLEX_SCANNER */
-
+	scanner_finish();
 	clearerr(stdin);
 
 	if (yyresult)				/* error */
diff --git a/src/backend/parser/scan.l b/src/backend/parser/scan.l
index e8b43ec0d76516c5d2baee7e8441cbce07f6790e..cb8610c87ac384a3e0730eb67e33283ff4af7365 100644
--- a/src/backend/parser/scan.l
+++ b/src/backend/parser/scan.l
@@ -9,7 +9,7 @@
  *
  *
  * IDENTIFICATION
- *	  $Header: /cvsroot/pgsql/src/backend/parser/scan.l,v 1.91 2002/03/06 06:09:56 momjian Exp $
+ *	  $Header: /cvsroot/pgsql/src/backend/parser/scan.l,v 1.92 2002/04/20 21:56:14 petere Exp $
  *
  *-------------------------------------------------------------------------
  */
@@ -17,9 +17,6 @@
 
 #include <ctype.h>
 #include <unistd.h>
-#ifndef __linux__
-#include <math.h>
-#endif
 #include <errno.h>
 
 #include "miscadmin.h"
@@ -28,45 +25,18 @@
 #include "parser/gramparse.h"
 #include "parser/keywords.h"
 #include "parser/parse.h"
-#include "parser/scansup.h"
 #include "utils/builtins.h"
 
 #ifdef MULTIBYTE
 #include "mb/pg_wchar.h"
 #endif
 
-extern char *parseString;
-static char *parseCh;
-
-/* some versions of lex define this as a macro */
-#if defined(yywrap)
-#undef yywrap
-#endif /* yywrap */
-
-/* set up my input handler --- need one flavor for flex, one for lex */
-#if defined(FLEX_SCANNER)
-
-#define YY_NEVER_INTERACTIVE 1
-#define YY_NO_UNPUT
-static int myinput(char* buf, int max);
-#undef YY_INPUT
-#define YY_INPUT(buf,result,max) {result = myinput(buf,max);}
-
-/* No reason to constrain amount of data slurped per myinput() call. */
+/* No reason to constrain amount of data slurped */
 #define YY_READ_BUF_SIZE 16777216
 
 /* Avoid exit() on fatal scanner errors (a bit ugly -- see yy_fatal_error) */
 #define fprintf(file, fmt, msg)  elog(FATAL, "%s", (msg))
 
-#else /* !FLEX_SCANNER */
-
-#undef input
-int input();
-#undef unput
-void unput(char);
-
-#endif /* FLEX_SCANNER */
-
 extern YYSTYPE yylval;
 
 static int		xcdepth = 0;	/* depth of nesting in slash-star comments */
@@ -83,8 +53,23 @@ static int		literalalloc;	/* current allocated buffer size */
 
 #define startlit()  (literalbuf[0] = '\0', literallen = 0)
 static void addlit(char *ytext, int yleng);
+static void addlitchar(unsigned char ychar);
+static char *litbufdup(void);
+
+/* Handles to the buffer that the lexer uses internally */
+static YY_BUFFER_STATE scanbufhandle;
+static char *scanbuf;
+
+unsigned char unescape_single_char(unsigned char c);
 
 %}
+
+%option 8bit
+%option never-interactive
+%option nounput
+%option noyywrap
+%option prefix="base_yy"
+
 /*
  * OK, here is a short description of lex/flex rules behavior.
  * The longest pattern which matches an input string is always chosen.
@@ -126,18 +111,17 @@ xhcat			{quote}{whitespace_with_newline}{quote}
 /* Extended quote
  * xqdouble implements SQL92 embedded quote
  * xqcat allows strings to cross input lines
- * Note: reduction of '' and \ sequences to output text is done in scanstr(),
- * not by rules here.  But we do get rid of xqcat sequences here.
  */
 quote			'
 xqstart			{quote}
 xqstop			{quote}
 xqdouble		{quote}{quote}
 xqinside		[^\\']+
-xqliteral		[\\](.|\n)
+xqescape		[\\][^0-7]
+xqoctesc		[\\][0-7]{1,3}
 xqcat			{quote}{whitespace_with_newline}{quote}
 
-/* Delimited quote
+/* Double quote
  * Allows embedded spaces and other special characters into identifiers.
  */
 dquote			\"
@@ -238,10 +222,7 @@ whitespace_with_newline	({horiz_whitespace}*{newline}{whitespace}*)
 
 other			.
 
-/* DO NOT PUT ANY COMMENTS IN THE FOLLOWING SECTION.
- * AT&T lex does not properly handle C-style comments in this second lex block.
- * So, put comments here. thomas - 1997-09-08
- *
+/*
  * Quoted strings must allow some special characters such as single-quote
  *  and newline.
  * Embedded single-quotes are implemented both in the SQL92-standard
@@ -285,14 +266,14 @@ other			.
 {xbitstart}		{
 					BEGIN(xbit);
 					startlit();
-					addlit("b", 1);
+					addlitchar('b');
 				}
 <xbit>{xbitstop}	{
 					BEGIN(INITIAL);
 					if (literalbuf[strspn(literalbuf + 1, "01") + 1] != '\0')
 						elog(ERROR, "invalid bit string input: '%s'",
 							 literalbuf);
-					yylval.str = pstrdup(literalbuf);
+					yylval.str = litbufdup();
 					return BITCONST;
 				}
 <xh>{xhinside}	|
@@ -335,14 +316,22 @@ other			.
 				}
 <xq>{xqstop}	{
 					BEGIN(INITIAL);
-					yylval.str = scanstr(literalbuf);
+					yylval.str = litbufdup();
 					return SCONST;
 				}
-<xq>{xqdouble}	|
-<xq>{xqinside}	|
-<xq>{xqliteral} {
+<xq>{xqdouble}  {
+					addlitchar('\'');
+				}
+<xq>{xqinside}  {
 					addlit(yytext, yyleng);
 				}
+<xq>{xqescape}  {
+					addlitchar(unescape_single_char(yytext[1]));
+				}
+<xq>{xqoctesc}  {
+					unsigned char c = strtoul(yytext+1, NULL, 8);
+					addlitchar(c);
+				}
 <xq>{xqcat}		{
 					/* ignore */
 				}
@@ -371,11 +360,11 @@ other			.
 						literalbuf[NAMEDATALEN-1] = '\0';
 #endif
 					}
-					yylval.str = pstrdup(literalbuf);
+					yylval.str = litbufdup();
 					return IDENT;
 				}
 <xd>{xddouble} {
-					addlit(yytext, yyleng-1);
+					addlitchar('"');
 				}
 <xd>{xdinside}	{
 					addlit(yytext, yyleng);
@@ -540,36 +529,44 @@ yyerror(const char *message)
 	elog(ERROR, "parser: %s at or near \"%s\"", message, yytext);
 }
 
-int
-yywrap(void)
-{
-	return(1);
-}
 
 /*
- scanner_init:
-	called by postgres before any actual parsing is done
-*/
+ * Called before any actual parsing is done
+ */
 void
-scanner_init(void)
+scanner_init(StringInfo str)
 {
-	/* it's important to set this to NULL
-	   because input()/myinput() checks the non-nullness of parseCh
-	   to know when to pass the string to lex/flex */
-	parseCh = NULL;
+	/*
+	 * Might be left over after elog()
+	 */
+	if (YY_CURRENT_BUFFER)
+		yy_delete_buffer(YY_CURRENT_BUFFER);
+
+	scanbuf = palloc(str->len + 2);
+	memcpy(scanbuf, str->data, str->len);
+	scanbuf[str->len] = scanbuf[str->len + 1] = YY_END_OF_BUFFER_CHAR;
+	scanbufhandle = yy_scan_buffer(scanbuf, str->len + 2);
 
 	/* initialize literal buffer to a reasonable but expansible size */
 	literalalloc = 128;
 	literalbuf = (char *) palloc(literalalloc);
 	startlit();
 
-#if defined(FLEX_SCANNER)
-	if (YY_CURRENT_BUFFER)
-		yy_flush_buffer(YY_CURRENT_BUFFER);
-#endif /* FLEX_SCANNER */
-	BEGIN INITIAL;
+	BEGIN(INITIAL);
 }
 
+
+/*
+ * Called after parsing is done to clean up after scanner_init()
+ */
+void
+scanner_finish(void)
+{
+	yy_delete_buffer(scanbufhandle);
+	pfree(scanbuf);
+}
+
+
 static void
 addlit(char *ytext, int yleng)
 {
@@ -587,54 +584,55 @@ addlit(char *ytext, int yleng)
 	literalbuf[literallen] = '\0';
 }
 
-#if !defined(FLEX_SCANNER)
 
-/* get lex input from a string instead of from stdin */
-int
-input()
+static void
+addlitchar(unsigned char ychar)
 {
-	if (parseCh == NULL)
-		parseCh = parseString;
-	if (*parseCh == '\0')
-		return(0);
-	else
-		return(*parseCh++);
+	/* enlarge buffer if needed */
+	if ((literallen+1) >= literalalloc)
+	{
+		literalalloc *= 2;
+		literalbuf = (char *) repalloc(literalbuf, literalalloc);
+	}
+	/* append new data, add trailing null */
+	literalbuf[literallen] = ychar;
+	literallen += 1;
+	literalbuf[literallen] = '\0';
 }
 
-/* undo lex input from a string instead of from stdin */
-void
-unput(char c)
+
+/*
+ * One might be tempted to write pstrdup(literalbuf) instead of this,
+ * but for long literals this is much faster because the length is
+ * already known.
+ */
+static char *
+litbufdup(void)
 {
-	if (parseCh == NULL)
-		elog(FATAL, "Unput() failed.\n");
-	else if (c != 0)
-		*--parseCh = c;
-}
+	char *new;
 
-#endif /* !defined(FLEX_SCANNER) */
+	new = palloc(literallen + 1);
+	memcpy(new, literalbuf, literallen+1);
+	return new;
+}
 
-#ifdef FLEX_SCANNER
 
-/* input routine for flex to read input from a string instead of a file */
-static int
-myinput(char* buf, int max)
+unsigned char
+unescape_single_char(unsigned char c)
 {
-	int len;
-
-	if (parseCh == NULL)
-		parseCh = parseString;
-	len = strlen(parseCh);		/* remaining data available */
-	/* Note: this code used to think that flex wants a null-terminated
-	 * string.  It does NOT, and returning 1 less character than it asks
-	 * for will cause failure under the right boundary conditions.  So
-	 * shut up and fill the buffer to the limit, you hear?
-	 */
-	if (len > max)
-		len = max;
-	if (len > 0)
-		memcpy(buf, parseCh, len);
-	parseCh += len;
-	return len;
+	switch (c)
+	{
+		case 'b':
+			return '\b';
+		case 'f':
+			return '\f';
+		case 'n':
+			return '\n';
+		case 'r':
+			return '\r';
+		case 't':
+			return '\t';
+		default:
+			return c;
+	}
 }
-
-#endif /* FLEX_SCANNER */
diff --git a/src/backend/tcop/postgres.c b/src/backend/tcop/postgres.c
index d8c2065258b0ef4bbe151ccb367cf0da6a720388..5305f1b2f754b9eede4103bcdf8416ae02c0fd88 100644
--- a/src/backend/tcop/postgres.c
+++ b/src/backend/tcop/postgres.c
@@ -8,7 +8,7 @@
  *
  *
  * IDENTIFICATION
- *	  $Header: /cvsroot/pgsql/src/backend/tcop/postgres.c,v 1.261 2002/04/18 20:01:09 tgl Exp $
+ *	  $Header: /cvsroot/pgsql/src/backend/tcop/postgres.c,v 1.262 2002/04/20 21:56:15 petere Exp $
  *
  * NOTES
  *	  this is the "main" module of the postgres backend and
@@ -120,7 +120,7 @@ int			XfuncMode = 0;
 static int	InteractiveBackend(StringInfo inBuf);
 static int	SocketBackend(StringInfo inBuf);
 static int	ReadCommand(StringInfo inBuf);
-static List *pg_parse_query(char *query_string, Oid *typev, int nargs);
+static List *pg_parse_query(StringInfo query_string, Oid *typev, int nargs);
 static List *pg_analyze_and_rewrite(Node *parsetree);
 static void start_xact_command(void);
 static void finish_xact_command(void);
@@ -330,11 +330,15 @@ pg_parse_and_rewrite(char *query_string,		/* string to execute */
 	List	   *raw_parsetree_list;
 	List	   *querytree_list;
 	List	   *list_item;
+	StringInfoData stri;
+
+	initStringInfo(&stri);
+	appendStringInfo(&stri, "%s", query_string);
 
 	/*
 	 * (1) parse the request string into a list of raw parse trees.
 	 */
-	raw_parsetree_list = pg_parse_query(query_string, typev, nargs);
+	raw_parsetree_list = pg_parse_query(&stri, typev, nargs);
 
 	/*
 	 * (2) Do parse analysis and rule rewrite.
@@ -365,12 +369,12 @@ pg_parse_and_rewrite(char *query_string,		/* string to execute */
  * commands are not processed any further than the raw parse stage.
  */
 static List *
-pg_parse_query(char *query_string, Oid *typev, int nargs)
+pg_parse_query(StringInfo query_string, Oid *typev, int nargs)
 {
 	List	   *raw_parsetree_list;
 
 	if (Debug_print_query)
-		elog(LOG, "query: %s", query_string);
+		elog(LOG, "query: %s", query_string->data);
 
 	if (Show_parser_stats)
 		ResetUsage();
@@ -549,7 +553,7 @@ pg_plan_query(Query *querytree)
  */
 
 void
-pg_exec_query_string(char *query_string,		/* string to execute */
+pg_exec_query_string(StringInfo query_string,		/* string to execute */
 					 CommandDest dest,	/* where results should go */
 					 MemoryContext parse_context)		/* context for
 														 * parsetrees */
@@ -559,7 +563,7 @@ pg_exec_query_string(char *query_string,		/* string to execute */
 	List	   *parsetree_list,
 			   *parsetree_item;
 
-	debug_query_string = query_string;	/* used by pgmonitor */
+	debug_query_string = query_string->data;	/* used by pgmonitor */
 
 	/*
 	 * Start up a transaction command.	All queries generated by the
@@ -725,7 +729,7 @@ pg_exec_query_string(char *query_string,		/* string to execute */
 				 * process utility functions (create, destroy, etc..)
 				 */
 				if (Debug_print_query)
-					elog(LOG, "ProcessUtility: %s", query_string);
+					elog(LOG, "ProcessUtility: %s", query_string->data);
 				else elog(DEBUG2, "ProcessUtility");
 
 				if (querytree->originalQuery)
@@ -1688,7 +1692,7 @@ PostgresMain(int argc, char *argv[], const char *username)
 	if (!IsUnderPostmaster)
 	{
 		puts("\nPOSTGRES backend interactive interface ");
-		puts("$Revision: 1.261 $ $Date: 2002/04/18 20:01:09 $\n");
+		puts("$Revision: 1.262 $ $Date: 2002/04/20 21:56:15 $\n");
 	}
 
 	/*
@@ -1913,7 +1917,7 @@ PostgresMain(int argc, char *argv[], const char *username)
 
 					pgstat_report_activity(parser_input->data);
 
-					pg_exec_query_string(parser_input->data,
+					pg_exec_query_string(parser_input,
 										 whereToSendOutput,
 										 QueryContext);
 
diff --git a/src/include/parser/gramparse.h b/src/include/parser/gramparse.h
index 42ae42d9e63b084e5d3db3fb6f2821d71d197e3a..3a1c9353bbfb84420ee168c77c0d5c72360e6756 100644
--- a/src/include/parser/gramparse.h
+++ b/src/include/parser/gramparse.h
@@ -7,7 +7,7 @@
  * Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
  * Portions Copyright (c) 1994, Regents of the University of California
  *
- * $Id: gramparse.h,v 1.20 2002/04/09 20:35:55 tgl Exp $
+ * $Id: gramparse.h,v 1.21 2002/04/20 21:56:15 petere Exp $
  *
  *-------------------------------------------------------------------------
  */
@@ -15,11 +15,14 @@
 #ifndef GRAMPARSE_H
 #define GRAMPARSE_H
 
+#include "lib/stringinfo.h"
+
 /* from parser.c */
 extern int	yylex(void);
 
 /* from scan.l */
-extern void scanner_init(void);
+extern void scanner_init(StringInfo str);
+extern void scanner_finish(void);
 extern int	base_yylex(void);
 extern void yyerror(const char *message);
 
diff --git a/src/include/parser/parser.h b/src/include/parser/parser.h
index 8547a66cc14b313ca70b7bfe5361bb830b20131c..30d9a11fe460ac3f92231c27948f56efe28d005c 100644
--- a/src/include/parser/parser.h
+++ b/src/include/parser/parser.h
@@ -7,15 +7,16 @@
  * Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
  * Portions Copyright (c) 1994, Regents of the University of California
  *
- * $Id: parser.h,v 1.11 2001/11/05 17:46:35 momjian Exp $
+ * $Id: parser.h,v 1.12 2002/04/20 21:56:15 petere Exp $
  *
  *-------------------------------------------------------------------------
  */
 #ifndef PARSER_H
 #define PARSER_H
 
+#include "lib/stringinfo.h"
 #include "parser/parse_node.h"
 
-extern List *parser(char *str, Oid *typev, int nargs);
+extern List *parser(StringInfo str, Oid *typev, int nargs);
 
 #endif   /* PARSER_H */
diff --git a/src/include/tcop/tcopprot.h b/src/include/tcop/tcopprot.h
index 28e45e7b5e98068b53cc444f1381ba3243219513..0f79e8ab9d6d1760a25114a7162266093c7ce3d8 100644
--- a/src/include/tcop/tcopprot.h
+++ b/src/include/tcop/tcopprot.h
@@ -7,7 +7,7 @@
  * Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
  * Portions Copyright (c) 1994, Regents of the University of California
  *
- * $Id: tcopprot.h,v 1.47 2001/11/10 23:51:14 tgl Exp $
+ * $Id: tcopprot.h,v 1.48 2002/04/20 21:56:15 petere Exp $
  *
  * OLD COMMENTS
  *	  This file was created so that other c files could get the two
@@ -22,6 +22,7 @@
 #include <setjmp.h>
 
 #include "executor/execdesc.h"
+#include "lib/stringinfo.h"
 #include "tcop/dest.h"
 
 
@@ -37,7 +38,7 @@ extern bool ShowPortNumber;
 extern List *pg_parse_and_rewrite(char *query_string,
 					 Oid *typev, int nargs);
 extern Plan *pg_plan_query(Query *querytree);
-extern void pg_exec_query_string(char *query_string,
+extern void pg_exec_query_string(StringInfo query_string,
 					 CommandDest dest,
 					 MemoryContext parse_context);
 #endif   /* BOOTSTRAP_INCLUDE */