Commit 09ac603c authored by Tom Lane's avatar Tom Lane

Work around unportable behavior of malloc(0) and realloc(NULL, 0).

On some platforms these functions return NULL, rather than the more common
practice of returning a pointer to a zero-sized block of memory.  Hack our
various wrapper functions to hide the difference by substituting a size
request of 1.  This is probably not so important for the callers, who
should never touch the block anyway if they asked for size 0 --- but it's
important for the wrapper functions themselves, which mistakenly treated
the NULL result as an out-of-memory failure.  This broke at least pg_dump
for the case of no user-defined aggregates, as per report from
Matthew Carrington.

Back-patch to 9.2 to fix the pg_dump issue.  Given the lack of previous
complaints, it seems likely that there is no live bug in previous releases,
even though some of these functions were in place before that.
parent 2164f9a1
...@@ -51,6 +51,7 @@ struct options ...@@ -51,6 +51,7 @@ struct options
static void help(const char *progname); static void help(const char *progname);
void get_opts(int, char **, struct options *); void get_opts(int, char **, struct options *);
void *pg_malloc(size_t size); void *pg_malloc(size_t size);
void *pg_realloc(void *ptr, size_t size);
char *pg_strdup(const char *str); char *pg_strdup(const char *str);
void add_one_elt(char *eltname, eary *eary); void add_one_elt(char *eltname, eary *eary);
char *get_comma_elts(eary *eary); char *get_comma_elts(eary *eary);
...@@ -203,16 +204,37 @@ help(const char *progname) ...@@ -203,16 +204,37 @@ help(const char *progname)
void * void *
pg_malloc(size_t size) pg_malloc(size_t size)
{ {
void *ptr = malloc(size); void *ptr;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
ptr = malloc(size);
if (!ptr) if (!ptr)
{ {
fprintf(stderr, "out of memory"); fprintf(stderr, "out of memory\n");
exit(1); exit(1);
} }
return ptr; return ptr;
} }
void *
pg_realloc(void *ptr, size_t size)
{
void *result;
/* Avoid unportable behavior of realloc(NULL, 0) */
if (ptr == NULL && size == 0)
size = 1;
result = realloc(ptr, size);
if (!result)
{
fprintf(stderr, "out of memory\n");
exit(1);
}
return result;
}
char * char *
pg_strdup(const char *str) pg_strdup(const char *str)
{ {
...@@ -220,7 +242,7 @@ pg_strdup(const char *str) ...@@ -220,7 +242,7 @@ pg_strdup(const char *str)
if (!result) if (!result)
{ {
fprintf(stderr, "out of memory"); fprintf(stderr, "out of memory\n");
exit(1); exit(1);
} }
return result; return result;
...@@ -242,14 +264,8 @@ add_one_elt(char *eltname, eary *eary) ...@@ -242,14 +264,8 @@ add_one_elt(char *eltname, eary *eary)
else if (eary->num >= eary->alloc) else if (eary->num >= eary->alloc)
{ {
eary ->alloc *= 2; eary ->alloc *= 2;
eary ->array = (char **) eary ->array = (char **) pg_realloc(eary->array,
realloc(eary->array, eary->alloc * sizeof(char *)); eary->alloc * sizeof(char *));
if (!eary->array)
{
fprintf(stderr, "out of memory");
exit(1);
}
} }
eary ->array[eary->num] = pg_strdup(eltname); eary ->array[eary->num] = pg_strdup(eltname);
......
...@@ -192,33 +192,39 @@ get_user_info(char **user_name) ...@@ -192,33 +192,39 @@ get_user_info(char **user_name)
void * void *
pg_malloc(size_t n) pg_malloc(size_t size)
{ {
void *p = malloc(n); void *p;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
p = malloc(size);
if (p == NULL) if (p == NULL)
pg_log(PG_FATAL, "%s: out of memory\n", os_info.progname); pg_log(PG_FATAL, "%s: out of memory\n", os_info.progname);
return p; return p;
} }
void * void *
pg_realloc(void *ptr, size_t n) pg_realloc(void *ptr, size_t size)
{ {
void *p = realloc(ptr, n); void *p;
/* Avoid unportable behavior of realloc(NULL, 0) */
if (ptr == NULL && size == 0)
size = 1;
p = realloc(ptr, size);
if (p == NULL) if (p == NULL)
pg_log(PG_FATAL, "%s: out of memory\n", os_info.progname); pg_log(PG_FATAL, "%s: out of memory\n", os_info.progname);
return p; return p;
} }
void void
pg_free(void *p) pg_free(void *ptr)
{ {
if (p != NULL) if (ptr != NULL)
free(p); free(ptr);
} }
......
...@@ -299,6 +299,9 @@ pg_malloc(size_t size) ...@@ -299,6 +299,9 @@ pg_malloc(size_t size)
{ {
void *result; void *result;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
result = malloc(size); result = malloc(size);
if (!result) if (!result)
{ {
...@@ -313,6 +316,9 @@ pg_realloc(void *ptr, size_t size) ...@@ -313,6 +316,9 @@ pg_realloc(void *ptr, size_t size)
{ {
void *result; void *result;
/* Avoid unportable behavior of realloc(NULL, 0) */
if (ptr == NULL && size == 0)
size = 1;
result = realloc(ptr, size); result = realloc(ptr, size);
if (!result) if (!result)
{ {
......
...@@ -3358,6 +3358,9 @@ guc_malloc(int elevel, size_t size) ...@@ -3358,6 +3358,9 @@ guc_malloc(int elevel, size_t size)
{ {
void *data; void *data;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
data = malloc(size); data = malloc(size);
if (data == NULL) if (data == NULL)
ereport(elevel, ereport(elevel,
...@@ -3371,6 +3374,9 @@ guc_realloc(int elevel, void *old, size_t size) ...@@ -3371,6 +3374,9 @@ guc_realloc(int elevel, void *old, size_t size)
{ {
void *data; void *data;
/* Avoid unportable behavior of realloc(NULL, 0) */
if (old == NULL && size == 0)
size = 1;
data = realloc(old, size); data = realloc(old, size);
if (data == NULL) if (data == NULL)
ereport(elevel, ereport(elevel,
......
...@@ -294,6 +294,9 @@ pg_malloc(size_t size) ...@@ -294,6 +294,9 @@ pg_malloc(size_t size)
{ {
void *result; void *result;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
result = malloc(size); result = malloc(size);
if (!result) if (!result)
{ {
......
...@@ -54,6 +54,9 @@ pg_malloc0(size_t size) ...@@ -54,6 +54,9 @@ pg_malloc0(size_t size)
{ {
void *result; void *result;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
result = malloc(size); result = malloc(size);
if (!result) if (!result)
{ {
......
...@@ -233,6 +233,9 @@ pg_malloc(size_t size) ...@@ -233,6 +233,9 @@ pg_malloc(size_t size)
{ {
void *result; void *result;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
result = malloc(size); result = malloc(size);
if (!result) if (!result)
{ {
......
...@@ -42,6 +42,9 @@ pg_malloc(size_t size) ...@@ -42,6 +42,9 @@ pg_malloc(size_t size)
{ {
void *tmp; void *tmp;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
tmp = malloc(size); tmp = malloc(size);
if (!tmp) if (!tmp)
exit_horribly(NULL, "out of memory\n"); exit_horribly(NULL, "out of memory\n");
...@@ -63,6 +66,9 @@ pg_realloc(void *ptr, size_t size) ...@@ -63,6 +66,9 @@ pg_realloc(void *ptr, size_t size)
{ {
void *tmp; void *tmp;
/* Avoid unportable behavior of realloc(NULL, 0) */
if (ptr == NULL && size == 0)
size = 1;
tmp = realloc(ptr, size); tmp = realloc(ptr, size);
if (!tmp) if (!tmp)
exit_horribly(NULL, "out of memory\n"); exit_horribly(NULL, "out of memory\n");
......
...@@ -60,6 +60,9 @@ pg_malloc(size_t size) ...@@ -60,6 +60,9 @@ pg_malloc(size_t size)
{ {
void *tmp; void *tmp;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
tmp = malloc(size); tmp = malloc(size);
if (!tmp) if (!tmp)
{ {
......
...@@ -304,6 +304,9 @@ pg_malloc(size_t size) ...@@ -304,6 +304,9 @@ pg_malloc(size_t size)
{ {
void *tmp; void *tmp;
/* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
tmp = malloc(size); tmp = malloc(size);
if (!tmp) if (!tmp)
{ {
......
...@@ -70,7 +70,11 @@ fe_palloc(Size size) ...@@ -70,7 +70,11 @@ fe_palloc(Size size)
{ {
void *res; void *res;
if ((res = malloc(size)) == NULL) /* Avoid unportable behavior of malloc(0) */
if (size == 0)
size = 1;
res = malloc(size);
if (res == NULL)
{ {
fprintf(stderr, _("out of memory\n")); fprintf(stderr, _("out of memory\n"));
exit(1); exit(1);
...@@ -96,7 +100,11 @@ fe_repalloc(void *pointer, Size size) ...@@ -96,7 +100,11 @@ fe_repalloc(void *pointer, Size size)
{ {
void *res; void *res;
if ((res = realloc(pointer, size)) == NULL) /* Avoid unportable behavior of realloc(NULL, 0) */
if (pointer == NULL && size == 0)
size = 1;
res = realloc(pointer, size);
if (res == NULL)
{ {
fprintf(stderr, _("out of memory\n")); fprintf(stderr, _("out of memory\n"));
exit(1); exit(1);
......
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