From f3dda5be890dfdd6a3fae6ff21737307fcc6c0a8 Mon Sep 17 00:00:00 2001 From: Michael Meskes <meskes@postgresql.org> Date: Tue, 17 Jan 2006 19:49:23 +0000 Subject: [PATCH] Data transferred binary is now put into the variables verbatim. Also added a test case for a binary cursor. --- src/interfaces/ecpg/ChangeLog | 9 ++ src/interfaces/ecpg/ecpglib/Makefile | 4 +- src/interfaces/ecpg/ecpglib/data.c | 56 ++++++++++- src/interfaces/ecpg/preproc/Makefile | 4 +- src/interfaces/ecpg/test/test5.pgc | 138 +++++++++++++++++---------- 5 files changed, 153 insertions(+), 58 deletions(-) diff --git a/src/interfaces/ecpg/ChangeLog b/src/interfaces/ecpg/ChangeLog index b7765568dbf..6e7a25c6abc 100644 --- a/src/interfaces/ecpg/ChangeLog +++ b/src/interfaces/ecpg/ChangeLog @@ -1962,6 +1962,15 @@ Fri Dec 2 16:00:10 CET 2005 - Added special handling of CONNECTION variable that is used by ECPG instead of given to the backend. + +Fr Jan 13 17:29:30 CET 2006 + + - Fixed a fixed size buffer in preproc.y to be variable size since an + overflow could occur on the fixed one. + +Tu Jan 17 18:53:03 CET 2006 + + - Data transferred binary is now put into the variables verbatim. - Set ecpg library version to 5.2. - Set ecpg version to 4.2.1. diff --git a/src/interfaces/ecpg/ecpglib/Makefile b/src/interfaces/ecpg/ecpglib/Makefile index ef41c8e6940..a5ce3b39c08 100644 --- a/src/interfaces/ecpg/ecpglib/Makefile +++ b/src/interfaces/ecpg/ecpglib/Makefile @@ -4,7 +4,7 @@ # # Copyright (c) 1994, Regents of the University of California # -# $PostgreSQL: pgsql/src/interfaces/ecpg/ecpglib/Makefile,v 1.37 2005/12/09 21:19:35 petere Exp $ +# $PostgreSQL: pgsql/src/interfaces/ecpg/ecpglib/Makefile,v 1.38 2006/01/17 19:49:23 meskes Exp $ # #------------------------------------------------------------------------- @@ -14,7 +14,7 @@ include $(top_builddir)/src/Makefile.global NAME= ecpg SO_MAJOR_VERSION= 5 -SO_MINOR_VERSION= 3 +SO_MINOR_VERSION= 2 DLTYPE= library override CPPFLAGS := -DFRONTEND -I$(top_srcdir)/src/interfaces/ecpg/include \ diff --git a/src/interfaces/ecpg/ecpglib/data.c b/src/interfaces/ecpg/ecpglib/data.c index 90111db4dae..d88859b72c2 100644 --- a/src/interfaces/ecpg/ecpglib/data.c +++ b/src/interfaces/ecpg/ecpglib/data.c @@ -1,4 +1,4 @@ -/* $PostgreSQL: pgsql/src/interfaces/ecpg/ecpglib/data.c,v 1.29 2005/10/15 02:49:47 momjian Exp $ */ +/* $PostgreSQL: pgsql/src/interfaces/ecpg/ecpglib/data.c,v 1.30 2006/01/17 19:49:23 meskes Exp $ */ #define POSTGRES_ECPG_INTERNAL #include "postgres_fe.h" @@ -45,11 +45,12 @@ ECPGget_data(const PGresult *results, int act_tuple, int act_field, int lineno, long ind_offset, enum ARRAY_TYPE isarray, enum COMPAT_MODE compat, bool force_indicator) { struct sqlca_t *sqlca = ECPGget_sqlca(); - char *pval = (char *) PQgetvalue(results, act_tuple, act_field); - int value_for_indicator = 0; - - ECPGlog("ECPGget_data line %d: RESULT: %s offset: %ld array: %s\n", lineno, pval ? pval : "", offset, isarray ? "Yes" : "No"); + char *pval = (char *) PQgetvalue(results, act_tuple, act_field); + int binary = PQfformat(results, act_field); + int size = PQgetlength(results, act_tuple, act_field); + int value_for_indicator = 0; + ECPGlog("ECPGget_data line %d: RESULT: %s offset: %ld array: %s\n", lineno, pval ? (binary ? "BINARY" : pval) : "EMPTY", offset, isarray ? "Yes" : "No"); /* We will have to decode the value */ /* @@ -131,6 +132,50 @@ ECPGget_data(const PGresult *results, int act_tuple, int act_field, int lineno, do { + if (binary) + { + if (pval) + { + if (varcharsize == 0 || varcharsize*offset >= size) + memcpy((char *) ((long) var + offset * act_tuple), pval, size); + else + { + memcpy((char *) ((long) var + offset * act_tuple), pval, varcharsize*offset); + + if (varcharsize*offset < size) + { + /* truncation */ + switch (ind_type) + { + case ECPGt_short: + case ECPGt_unsigned_short: + *((short *) (ind + ind_offset * act_tuple)) = size; + break; + case ECPGt_int: + case ECPGt_unsigned_int: + *((int *) (ind + ind_offset * act_tuple)) = size; + break; + case ECPGt_long: + case ECPGt_unsigned_long: + *((long *) (ind + ind_offset * act_tuple)) = size; + break; +#ifdef HAVE_LONG_LONG_INT_64 + case ECPGt_long_long: + case ECPGt_unsigned_long_long: + *((long long int *) (ind + ind_offset * act_tuple)) = size; + break; +#endif /* HAVE_LONG_LONG_INT_64 */ + default: + break; + } + sqlca->sqlwarn[0] = sqlca->sqlwarn[1] = 'W'; + } + } + pval += size; + } + } + else + { switch (type) { long res; @@ -619,6 +664,7 @@ ECPGget_data(const PGresult *results, int act_tuple, int act_field, int lineno, if (*pval == ' ') ++pval; } + } } while (*pval != '\0' && ((isarray == ECPG_ARRAY_ARRAY && *pval != '}') || isarray == ECPG_ARRAY_VECTOR)); return (true); diff --git a/src/interfaces/ecpg/preproc/Makefile b/src/interfaces/ecpg/preproc/Makefile index 9572f8dff48..4fe98604b6e 100644 --- a/src/interfaces/ecpg/preproc/Makefile +++ b/src/interfaces/ecpg/preproc/Makefile @@ -4,7 +4,7 @@ # # Copyright (c) 1998-2005, PostgreSQL Global Development Group # -# $PostgreSQL: pgsql/src/interfaces/ecpg/preproc/Makefile,v 1.117 2005/12/09 21:19:36 petere Exp $ +# $PostgreSQL: pgsql/src/interfaces/ecpg/preproc/Makefile,v 1.118 2006/01/17 19:49:23 meskes Exp $ # #------------------------------------------------------------------------- @@ -14,7 +14,7 @@ top_builddir = ../../../.. include $(top_builddir)/src/Makefile.global MAJOR_VERSION= 4 -MINOR_VERSION= 3 +MINOR_VERSION= 2 PATCHLEVEL=1 override CPPFLAGS := -I$(srcdir)/../include -I$(srcdir) $(CPPFLAGS) \ diff --git a/src/interfaces/ecpg/test/test5.pgc b/src/interfaces/ecpg/test/test5.pgc index 18e5b48971f..08fcef3e0e7 100644 --- a/src/interfaces/ecpg/test/test5.pgc +++ b/src/interfaces/ecpg/test/test5.pgc @@ -7,60 +7,100 @@ EXEC SQL typedef short mmSmallInt; EXEC SQL BEGIN DECLARE SECTION; struct TBempl - { - mmInteger idnum; - mmChar name[21]; - mmSmallInt accs; - }; +{ + mmInteger idnum; + mmChar name[21]; + mmSmallInt accs; + mmChar byte[20]; +}; EXEC SQL END DECLARE SECTION; -int main(void) +int +main (void) { -EXEC SQL BEGIN DECLARE SECTION; -struct TBempl empl; -EXEC SQL END DECLARE SECTION; -FILE *dbgs; + EXEC SQL BEGIN DECLARE SECTION; + struct TBempl empl; + char *data = "\\001\\155\\000\\212"; + union + { + mmSmallInt accs; + char t[2]; + } a; + EXEC SQL END DECLARE SECTION; + FILE *dbgs; + int i; + + if ((dbgs = fopen ("log", "w")) != NULL) + ECPGdebug (1, dbgs); + + empl.idnum = 1; + EXEC SQL connect to mm; + if (sqlca.sqlcode) + { + printf ("connect error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } + + EXEC SQL create table empl + (idnum integer, name char (20), accs smallint, byte bytea); + if (sqlca.sqlcode) + { + printf ("create error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } + + EXEC SQL insert into empl values (1, 'first user', 320,:data); + if (sqlca.sqlcode) + { + printf ("insert error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } + + EXEC SQL select name, accs, byte into:empl.name,:empl.accs,:empl.byte from empl where idnum =:empl.idnum; + if (sqlca.sqlcode) + { + printf ("select error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } + printf ("name=%s, accs=%d byte=%s\n", empl.name, empl.accs, empl.byte); + + EXEC SQL DECLARE C CURSOR FOR select name, accs, byte from empl where idnum =:empl.idnum; + EXEC SQL OPEN C; + EXEC SQL FETCH C INTO:empl.name,:empl.accs,:empl.byte; + if (sqlca.sqlcode) + { + printf ("fetch error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } -if ((dbgs = fopen("log", "w")) != NULL) - ECPGdebug(1, dbgs); - -empl.idnum = 1; -EXEC SQL connect to mm; -if (sqlca.sqlcode) - { - printf("connect error = %ld\n", sqlca.sqlcode); - exit(sqlca.sqlcode); - } + printf ("name=%s, accs=%d byte=%s\n", empl.name, empl.accs, empl.byte); -EXEC SQL create table empl - ( - idnum integer, - name char(20), - accs smallint - ); -if (sqlca.sqlcode) - { - printf("select error = %ld\n", sqlca.sqlcode); - exit(sqlca.sqlcode); - } + memset(empl.name, 0, 21L); + memset(empl.byte, '#', 20L); + EXEC SQL DECLARE B BINARY CURSOR FOR select name, accs, byte from empl where idnum =:empl.idnum; + EXEC SQL OPEN B; + EXEC SQL FETCH B INTO :empl.name,:a.accs,:empl.byte; + if (sqlca.sqlcode) + { + printf ("fetch error = %ld\n", sqlca.sqlcode); + exit (sqlca.sqlcode); + } -EXEC SQL insert into empl values (1, 'first user', 20); -if (sqlca.sqlcode) - { - printf("select error = %ld\n", sqlca.sqlcode); - exit(sqlca.sqlcode); - } + EXEC SQL CLOSE B; -EXEC SQL select name, accs - into :empl.name, :empl.accs - from empl where idnum = :empl.idnum; -if (sqlca.sqlcode) - { - printf("select error = %ld\n", sqlca.sqlcode); - exit(sqlca.sqlcode); - } -printf("name=%s, accs=%d\n", empl.name, empl.accs); -EXEC SQL disconnect; -fclose(dbgs); -exit(0); + i=a.t[0]; + a.t[0]=a.t[1]; + a.t[1]=i; + + printf ("name=%s, accs=%d byte=", empl.name, a.accs); + for (i=0; i<20; i++) + { + if (empl.byte[i] == '#') + break; + printf("(%o)", (unsigned char)empl.byte[i]); + } + printf("\n"); + EXEC SQL disconnect; + fclose (dbgs); + exit (0); } -- GitLab