From 0e924c007dbb74f8f7dbdb75810c9b9a8ed6d3ec Mon Sep 17 00:00:00 2001
From: Tom Lane
Date: Mon, 8 Oct 2012 21:12:27 -0400
Subject: Fix lo_read, lo_write, lo_truncate to cope with "size_t" length
parameters.
libpq defines these functions as accepting "size_t" lengths ... but the
underlying backend functions expect signed int32 length parameters, and so
will misinterpret any value exceeding INT_MAX. Fix the libpq side to throw
error rather than possibly doing something unexpected.
This is a bug of long standing, but I doubt it's worth back-patching. The
problem is really pretty academic anyway with lo_read/lo_write, since any
caller expecting sane behavior would have to have provided a multi-gigabyte
buffer. It's slightly more pressing with lo_truncate, but still we haven't
supported large objects over 2GB until now.
---
doc/src/sgml/lobj.sgml | 56 ++++++++++++++++++++++++++++++++++++++++----------
1 file changed, 45 insertions(+), 11 deletions(-)
(limited to 'doc/src')
diff --git a/doc/src/sgml/lobj.sgml b/doc/src/sgml/lobj.sgml
index ba5674cff37..db5bc100f8f 100644
--- a/doc/src/sgml/lobj.sgml
+++ b/doc/src/sgml/lobj.sgml
@@ -69,6 +69,17 @@
access reads and writes.
+
+ The chunks stored for a large object do not have to be contiguous.
+ For example, if an application opens a new large object, seeks to offset
+ 1000000, and writes a few bytes there, this does not result in allocation
+ of 1000000 bytes worth of storage; only of chunks covering the range of
+ data bytes actually written. A read operation will, however, read out
+ zeroes for any unallocated locations preceding the last existing chunk.
+ This corresponds to the common behavior of sparsely allocated>
+ files in Unix file systems.
+
+
As of PostgreSQL> 9.0, large objects have an owner
and a set of access permissions, which can be managed using
@@ -299,11 +310,19 @@ inv_fd = lo_open(conn, inv_oid, INV_READ|INV_WRITE);
int lo_write(PGconn *conn, int fd, const char *buf, size_t len);
writes len bytes from buf
- to large object descriptor fd>. The fd
- argument must have been returned by a previous
- lo_open. The number of bytes actually
- written is returned. In the event of an error, the return value
- is -1.
+ (which must be of size len) to large object
+ descriptor fd>. The fd argument must
+ have been returned by a previous lo_open. The
+ number of bytes actually written is returned (in the current
+ implementation, this will always equal len unless
+ there is an error). In the event of an error, the return value is -1.
+
+
+
+ Although the len parameter is declared as
+ size_t>, this function will reject length values larger than
+ INT_MAX>. In practice, it's best to transfer data in chunks
+ of at most a few megabytes anyway.
@@ -316,13 +335,22 @@ int lo_write(PGconn *conn, int fd, const char *buf, size_t len);
int lo_read(PGconn *conn, int fd, char *buf, size_t len);
- reads len bytes from large object descriptor
- fd into buf. The
- fd argument must have been returned by a
- previous lo_open. The number of bytes
- actually read is returned. In the event of an error, the return
+ reads up to len bytes from large object descriptor
+ fd into buf (which must be
+ of size len). The fd
+ argument must have been returned by a previous
+ lo_open. The number of bytes actually read is
+ returned; this will be less than len if the end of
+ the large object is reached first. In the event of an error, the return
value is -1.
+
+
+ Although the len parameter is declared as
+ size_t>, this function will reject length values larger than
+ INT_MAX>. In practice, it's best to transfer data in chunks
+ of at most a few megabytes anyway.
+
@@ -416,7 +444,7 @@ int lo_truncate(PGcon *conn, int fd, size_t len);
fd argument must have been returned by a
previous lo_open. If len> is
greater than the large object's current length, the large object
- is extended with null bytes ('\0').
+ is extended to the specified length with null bytes ('\0').
On success, lo_truncate returns
zero. On error, the return value is -1.
@@ -426,6 +454,12 @@ int lo_truncate(PGcon *conn, int fd, size_t len);
fd is not changed.
+
+ Although the len parameter is declared as
+ size_t>, lo_truncate will reject length
+ values larger than INT_MAX>.
+
+
lo_truncate64>>
When dealing with large objects that might exceed 2GB in size,
--
cgit v1.2.3