diff --git a/src/backend/access/transam/xact.c b/src/backend/access/transam/xact.c index fb888640a9951061b300f2a6bf21b10b46f7f0cc..f32a7fb00b5ab37549976da92cc398b5c407e6ea 100644 --- a/src/backend/access/transam/xact.c +++ b/src/backend/access/transam/xact.c @@ -7,7 +7,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/access/transam/xact.c,v 1.20 1998/06/15 19:28:02 momjian Exp $ + * $Header: /cvsroot/pgsql/src/backend/access/transam/xact.c,v 1.21 1998/07/21 04:17:21 momjian Exp $ * * NOTES * Transaction aborts can now occur two ways: @@ -137,6 +137,11 @@ *------------------------------------------------------------------------- */ +/* + * Large object clean up added in CommitTransaction() to prevent buffer leaks. + * [PA, 7/17/98] + * [PA] is Pascal André <andre@via.ecp.fr> + */ #include <postgres.h> #include <access/xact.h> @@ -151,6 +156,9 @@ #include <commands/async.h> #include <commands/sequence.h> +/* included for _lo_commit [PA, 7/17/98] */ +#include <libpq/be-fsstubs.h> + static void AbortTransaction(void); static void AtAbort_Cache(void); static void AtAbort_Locks(void); @@ -889,6 +897,10 @@ CommitTransaction() * do commit processing * ---------------- */ + + /* handle commit for large objects [ PA, 7/17/98 ] */ + _lo_commit(); + CloseSequences(); DestroyTempRels(); AtEOXact_portals(); diff --git a/src/backend/libpq/be-fsstubs.c b/src/backend/libpq/be-fsstubs.c index 8ecf45fe1225d36430ab2087fde7744e3143d151..5267cf40abe2c95a6b3ed73dde805fadc8150cea 100644 --- a/src/backend/libpq/be-fsstubs.c +++ b/src/backend/libpq/be-fsstubs.c @@ -7,7 +7,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/libpq/be-fsstubs.c,v 1.21 1998/06/15 19:28:25 momjian Exp $ + * $Header: /cvsroot/pgsql/src/backend/libpq/be-fsstubs.c,v 1.22 1998/07/21 04:17:23 momjian Exp $ * * NOTES * This should be moved to a more appropriate place. It is here @@ -41,6 +41,8 @@ #include <storage/large_object.h> #include <libpq/be-fsstubs.h> +/* [PA] is Pascal André <andre@via.ecp.fr> */ + /*#define FSDB 1*/ #define MAX_LOBJ_FDS 256 @@ -52,7 +54,6 @@ static GlobalMemory fscxt = NULL; static int newLOfd(LargeObjectDesc *lobjCookie); static void deleteLOfd(int fd); - /***************************************************************************** * File Interfaces for Large Objects *****************************************************************************/ @@ -367,6 +368,26 @@ lo_export(Oid lobjId, text *filename) return 1; } +/* + * lo_commit - + * prepares large objects for transaction commit [PA, 7/17/98] + */ +void +_lo_commit(void) +{ + int i; + MemoryContext currentContext; + + currentContext = MemoryContextSwitchTo((MemoryContext) fscxt); + + for (i = 0; i < MAX_LOBJ_FDS; i++) { + if (cookies[i] != NULL) inv_cleanindex(cookies[i]); + } + + MemoryContextSwitchTo(currentContext); + +} + /***************************************************************************** * Support routines for this file diff --git a/src/backend/storage/large_object/inv_api.c b/src/backend/storage/large_object/inv_api.c index c3d63ac36b2cd2baa7783e69bc64cf927ac7159c..e92c75f0a346408ddccd7a9f1a3c860641f69c5e 100644 --- a/src/backend/storage/large_object/inv_api.c +++ b/src/backend/storage/large_object/inv_api.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/storage/large_object/inv_api.c,v 1.30 1998/06/15 19:29:16 momjian Exp $ + * $Header: /cvsroot/pgsql/src/backend/storage/large_object/inv_api.c,v 1.31 1998/07/21 04:17:24 momjian Exp $ * *------------------------------------------------------------------------- */ @@ -60,6 +60,19 @@ * for data. */ +/* + * In order to prevent buffer leak on transaction commit, large object + * scan index handling has been modified. Indexes are persistant inside + * a transaction but may be closed between two calls to this API (when + * transaction is committed while object is opened, or when no + * transaction is active). Scan indexes are thus now reinitialized using + * the object current offset. [PA] + * + * Some cleanup has been also done for non freed memory. + * + * For subsequent notes, [PA] is Pascal André <andre@via.ecp.fr> + */ + #define IFREESPC(p) (PageGetFreeSpace(p) - sizeof(HeapTupleData) - sizeof(struct varlena) - sizeof(int32)) #define IMAXBLK 8092 #define IMINBLK 512 @@ -261,8 +274,11 @@ inv_close(LargeObjectDesc *obj_desc) { Assert(PointerIsValid(obj_desc)); - if (obj_desc->iscan != (IndexScanDesc) NULL) + if (obj_desc->iscan != (IndexScanDesc) NULL) { index_endscan(obj_desc->iscan); + pfree(obj_desc->iscan); + obj_desc->iscan = NULL; + } heap_close(obj_desc->heap_r); index_close(obj_desc->index_r); @@ -546,6 +562,28 @@ inv_write(LargeObjectDesc *obj_desc, char *buf, int nbytes) return (nwritten); } +/* + * inv_cleanindex -- + * Clean opened indexes for large objects, and clears current result. + * This is necessary on transaction commit in order to prevent buffer + * leak. + * This function must be called for each opened large object. + * [ PA, 7/17/98 ] + */ +void +inv_cleanindex(LargeObjectDesc *obj_desc) +{ + Assert(PointerIsValid(obj_desc)); + + if (obj_desc->iscan == (IndexScanDesc) NULL) return; + + index_endscan(obj_desc->iscan); + pfree(obj_desc->iscan); + obj_desc->iscan = (IndexScanDesc) NULL; + + ItemPointerSetInvalid(&(obj_desc->htid)); +} + /* * inv_fetchtup -- Fetch an inversion file system block. * @@ -591,8 +629,13 @@ inv_fetchtup(LargeObjectDesc *obj_desc, Buffer *bufP) { ScanKeyData skey; + /* + * As scan index may be prematurely closed (on commit), + * we must use object current offset (was 0) to + * reinitialize the entry [ PA ]. + */ ScanKeyEntryInitialize(&skey, 0x0, 1, F_INT4GE, - Int32GetDatum(0)); + Int32GetDatum(obj_desc->offset)); obj_desc->iscan = index_beginscan(obj_desc->index_r, (bool) 0, (uint16) 1, @@ -622,7 +665,7 @@ inv_fetchtup(LargeObjectDesc *obj_desc, Buffer *bufP) /* remember this tid -- we may need it for later reads/writes */ ItemPointerCopy(&(res->heap_iptr), &(obj_desc->htid)); - + pfree(res); } else { @@ -1179,6 +1222,7 @@ _inv_getsize(Relation hreln, TupleDesc hdesc, Relation ireln) if (res == (RetrieveIndexResult) NULL) { index_endscan(iscan); + pfree(iscan); return (0); } @@ -1192,11 +1236,13 @@ _inv_getsize(Relation hreln, TupleDesc hdesc, Relation ireln) ReleaseBuffer(buf); htup = heap_fetch(hreln, false, &(res->heap_iptr), &buf); + pfree(res); } while (!HeapTupleIsValid(htup)); /* don't need the index scan anymore */ index_endscan(iscan); + pfree(iscan); /* get olastbyte attribute */ d = heap_getattr(htup, 1, hdesc, &isNull); diff --git a/src/include/libpq/be-fsstubs.h b/src/include/libpq/be-fsstubs.h index 507f2a936778f8f9314a4e25cdf988c27d366621..108869dac955e50652fed317f5bf5f988dc70f90 100644 --- a/src/include/libpq/be-fsstubs.h +++ b/src/include/libpq/be-fsstubs.h @@ -6,7 +6,7 @@ * * Copyright (c) 1994, Regents of the University of California * - * $Id: be-fsstubs.h,v 1.5 1997/09/08 21:52:33 momjian Exp $ + * $Id: be-fsstubs.h,v 1.6 1998/07/21 04:17:26 momjian Exp $ * *------------------------------------------------------------------------- */ @@ -37,4 +37,9 @@ extern int lo_unlink(Oid lobjId); extern struct varlena *loread(int fd, int len); extern int lowrite(int fd, struct varlena * wbuf); +/* + * Added for buffer leak prevention [ Pascal André <andre@via.ecp.fr> ] + */ +extern void _lo_commit(void); + #endif /* BE_FSSTUBS_H */ diff --git a/src/include/storage/large_object.h b/src/include/storage/large_object.h index 276792e19e209d668790f7e0ea2df4e010849719..3d35dba907af27d513ba4e64c2b8c3b6a4777a23 100644 --- a/src/include/storage/large_object.h +++ b/src/include/storage/large_object.h @@ -7,7 +7,7 @@ * * Copyright (c) 1994, Regents of the University of California * - * $Id: large_object.h,v 1.7 1997/09/08 21:54:29 momjian Exp $ + * $Id: large_object.h,v 1.8 1998/07/21 04:17:30 momjian Exp $ * *------------------------------------------------------------------------- */ @@ -54,4 +54,7 @@ extern int inv_tell(LargeObjectDesc *obj_desc); extern int inv_read(LargeObjectDesc *obj_desc, char *buf, int nbytes); extern int inv_write(LargeObjectDesc *obj_desc, char *buf, int nbytes); +/* added for buffer leak prevention [ PA ] */ +extern void inv_cleanindex(LargeObjectDesc *obj_desc); + #endif /* LARGE_OBJECT_H */