mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-12 18:34:36 +08:00
Fix minor memory leaks in pg_dump.
I found these by running pg_dump under "valgrind --leak-check=full". The changes in flagInhIndexes() and getIndexes() replace allocation of an array of which we use only some elements by individual allocations of just the actually-needed objects. The previous coding wasted some memory, but more importantly it confused valgrind's leak tracking. collectComments() and collectSecLabels() remain major blots on the valgrind report, because they don't PQclear their query results, in order to avoid a lot of strdup's. That's a dubious tradeoff, but I'll leave it alone here; an upcoming patch will modify those functions enough to justify changing the tradeoff.
This commit is contained in:
parent
b3b4d8e68a
commit
70bef49400
@ -260,6 +260,8 @@ getSchemaData(Archive *fout, int *numTablesPtr)
|
|||||||
pg_log_info("reading subscriptions");
|
pg_log_info("reading subscriptions");
|
||||||
getSubscriptions(fout);
|
getSubscriptions(fout);
|
||||||
|
|
||||||
|
free(inhinfo); /* not needed any longer */
|
||||||
|
|
||||||
*numTablesPtr = numTables;
|
*numTablesPtr = numTables;
|
||||||
return tblinfo;
|
return tblinfo;
|
||||||
}
|
}
|
||||||
@ -373,24 +375,20 @@ static void
|
|||||||
flagInhIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
flagInhIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
||||||
{
|
{
|
||||||
int i,
|
int i,
|
||||||
j,
|
j;
|
||||||
k;
|
|
||||||
|
|
||||||
for (i = 0; i < numTables; i++)
|
for (i = 0; i < numTables; i++)
|
||||||
{
|
{
|
||||||
IndexAttachInfo *attachinfo;
|
|
||||||
|
|
||||||
if (!tblinfo[i].ispartition || tblinfo[i].numParents == 0)
|
if (!tblinfo[i].ispartition || tblinfo[i].numParents == 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
Assert(tblinfo[i].numParents == 1);
|
Assert(tblinfo[i].numParents == 1);
|
||||||
|
|
||||||
attachinfo = (IndexAttachInfo *)
|
for (j = 0; j < tblinfo[i].numIndexes; j++)
|
||||||
pg_malloc0(tblinfo[i].numIndexes * sizeof(IndexAttachInfo));
|
|
||||||
for (j = 0, k = 0; j < tblinfo[i].numIndexes; j++)
|
|
||||||
{
|
{
|
||||||
IndxInfo *index = &(tblinfo[i].indexes[j]);
|
IndxInfo *index = &(tblinfo[i].indexes[j]);
|
||||||
IndxInfo *parentidx;
|
IndxInfo *parentidx;
|
||||||
|
IndexAttachInfo *attachinfo;
|
||||||
|
|
||||||
if (index->parentidx == 0)
|
if (index->parentidx == 0)
|
||||||
continue;
|
continue;
|
||||||
@ -399,14 +397,16 @@ flagInhIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
|||||||
if (parentidx == NULL)
|
if (parentidx == NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
attachinfo[k].dobj.objType = DO_INDEX_ATTACH;
|
attachinfo = (IndexAttachInfo *) pg_malloc(sizeof(IndexAttachInfo));
|
||||||
attachinfo[k].dobj.catId.tableoid = 0;
|
|
||||||
attachinfo[k].dobj.catId.oid = 0;
|
attachinfo->dobj.objType = DO_INDEX_ATTACH;
|
||||||
AssignDumpId(&attachinfo[k].dobj);
|
attachinfo->dobj.catId.tableoid = 0;
|
||||||
attachinfo[k].dobj.name = pg_strdup(index->dobj.name);
|
attachinfo->dobj.catId.oid = 0;
|
||||||
attachinfo[k].dobj.namespace = index->indextable->dobj.namespace;
|
AssignDumpId(&attachinfo->dobj);
|
||||||
attachinfo[k].parentIdx = parentidx;
|
attachinfo->dobj.name = pg_strdup(index->dobj.name);
|
||||||
attachinfo[k].partitionIdx = index;
|
attachinfo->dobj.namespace = index->indextable->dobj.namespace;
|
||||||
|
attachinfo->parentIdx = parentidx;
|
||||||
|
attachinfo->partitionIdx = index;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We must state the DO_INDEX_ATTACH object's dependencies
|
* We must state the DO_INDEX_ATTACH object's dependencies
|
||||||
@ -423,17 +423,15 @@ flagInhIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
|||||||
* will not try to run the ATTACH concurrently with other
|
* will not try to run the ATTACH concurrently with other
|
||||||
* operations on those tables.
|
* operations on those tables.
|
||||||
*/
|
*/
|
||||||
addObjectDependency(&attachinfo[k].dobj, index->dobj.dumpId);
|
addObjectDependency(&attachinfo->dobj, index->dobj.dumpId);
|
||||||
addObjectDependency(&attachinfo[k].dobj, parentidx->dobj.dumpId);
|
addObjectDependency(&attachinfo->dobj, parentidx->dobj.dumpId);
|
||||||
addObjectDependency(&attachinfo[k].dobj,
|
addObjectDependency(&attachinfo->dobj,
|
||||||
index->indextable->dobj.dumpId);
|
index->indextable->dobj.dumpId);
|
||||||
addObjectDependency(&attachinfo[k].dobj,
|
addObjectDependency(&attachinfo->dobj,
|
||||||
parentidx->indextable->dobj.dumpId);
|
parentidx->indextable->dobj.dumpId);
|
||||||
|
|
||||||
/* keep track of the list of partitions in the parent index */
|
/* keep track of the list of partitions in the parent index */
|
||||||
simple_ptr_list_append(&parentidx->partattaches, &attachinfo[k].dobj);
|
simple_ptr_list_append(&parentidx->partattaches, &attachinfo->dobj);
|
||||||
|
|
||||||
k++;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3374,6 +3374,8 @@ _selectTableAccessMethod(ArchiveHandle *AH, const char *tableam)
|
|||||||
|
|
||||||
destroyPQExpBuffer(cmd);
|
destroyPQExpBuffer(cmd);
|
||||||
|
|
||||||
|
if (AH->currTableAm)
|
||||||
|
free(AH->currTableAm);
|
||||||
AH->currTableAm = pg_strdup(want);
|
AH->currTableAm = pg_strdup(want);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6869,7 +6869,6 @@ getIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
|||||||
PQExpBuffer query = createPQExpBuffer();
|
PQExpBuffer query = createPQExpBuffer();
|
||||||
PGresult *res;
|
PGresult *res;
|
||||||
IndxInfo *indxinfo;
|
IndxInfo *indxinfo;
|
||||||
ConstraintInfo *constrinfo;
|
|
||||||
int i_tableoid,
|
int i_tableoid,
|
||||||
i_oid,
|
i_oid,
|
||||||
i_indexname,
|
i_indexname,
|
||||||
@ -7133,7 +7132,6 @@ getIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
|||||||
|
|
||||||
tbinfo->indexes = indxinfo =
|
tbinfo->indexes = indxinfo =
|
||||||
(IndxInfo *) pg_malloc(ntups * sizeof(IndxInfo));
|
(IndxInfo *) pg_malloc(ntups * sizeof(IndxInfo));
|
||||||
constrinfo = (ConstraintInfo *) pg_malloc(ntups * sizeof(ConstraintInfo));
|
|
||||||
tbinfo->numIndexes = ntups;
|
tbinfo->numIndexes = ntups;
|
||||||
|
|
||||||
for (j = 0; j < ntups; j++)
|
for (j = 0; j < ntups; j++)
|
||||||
@ -7173,28 +7171,31 @@ getIndexes(Archive *fout, TableInfo tblinfo[], int numTables)
|
|||||||
* If we found a constraint matching the index, create an
|
* If we found a constraint matching the index, create an
|
||||||
* entry for it.
|
* entry for it.
|
||||||
*/
|
*/
|
||||||
constrinfo[j].dobj.objType = DO_CONSTRAINT;
|
ConstraintInfo *constrinfo;
|
||||||
constrinfo[j].dobj.catId.tableoid = atooid(PQgetvalue(res, j, i_contableoid));
|
|
||||||
constrinfo[j].dobj.catId.oid = atooid(PQgetvalue(res, j, i_conoid));
|
|
||||||
AssignDumpId(&constrinfo[j].dobj);
|
|
||||||
constrinfo[j].dobj.dump = tbinfo->dobj.dump;
|
|
||||||
constrinfo[j].dobj.name = pg_strdup(PQgetvalue(res, j, i_conname));
|
|
||||||
constrinfo[j].dobj.namespace = tbinfo->dobj.namespace;
|
|
||||||
constrinfo[j].contable = tbinfo;
|
|
||||||
constrinfo[j].condomain = NULL;
|
|
||||||
constrinfo[j].contype = contype;
|
|
||||||
if (contype == 'x')
|
|
||||||
constrinfo[j].condef = pg_strdup(PQgetvalue(res, j, i_condef));
|
|
||||||
else
|
|
||||||
constrinfo[j].condef = NULL;
|
|
||||||
constrinfo[j].confrelid = InvalidOid;
|
|
||||||
constrinfo[j].conindex = indxinfo[j].dobj.dumpId;
|
|
||||||
constrinfo[j].condeferrable = *(PQgetvalue(res, j, i_condeferrable)) == 't';
|
|
||||||
constrinfo[j].condeferred = *(PQgetvalue(res, j, i_condeferred)) == 't';
|
|
||||||
constrinfo[j].conislocal = true;
|
|
||||||
constrinfo[j].separate = true;
|
|
||||||
|
|
||||||
indxinfo[j].indexconstraint = constrinfo[j].dobj.dumpId;
|
constrinfo = (ConstraintInfo *) pg_malloc(sizeof(ConstraintInfo));
|
||||||
|
constrinfo->dobj.objType = DO_CONSTRAINT;
|
||||||
|
constrinfo->dobj.catId.tableoid = atooid(PQgetvalue(res, j, i_contableoid));
|
||||||
|
constrinfo->dobj.catId.oid = atooid(PQgetvalue(res, j, i_conoid));
|
||||||
|
AssignDumpId(&constrinfo->dobj);
|
||||||
|
constrinfo->dobj.dump = tbinfo->dobj.dump;
|
||||||
|
constrinfo->dobj.name = pg_strdup(PQgetvalue(res, j, i_conname));
|
||||||
|
constrinfo->dobj.namespace = tbinfo->dobj.namespace;
|
||||||
|
constrinfo->contable = tbinfo;
|
||||||
|
constrinfo->condomain = NULL;
|
||||||
|
constrinfo->contype = contype;
|
||||||
|
if (contype == 'x')
|
||||||
|
constrinfo->condef = pg_strdup(PQgetvalue(res, j, i_condef));
|
||||||
|
else
|
||||||
|
constrinfo->condef = NULL;
|
||||||
|
constrinfo->confrelid = InvalidOid;
|
||||||
|
constrinfo->conindex = indxinfo[j].dobj.dumpId;
|
||||||
|
constrinfo->condeferrable = *(PQgetvalue(res, j, i_condeferrable)) == 't';
|
||||||
|
constrinfo->condeferred = *(PQgetvalue(res, j, i_condeferred)) == 't';
|
||||||
|
constrinfo->conislocal = true;
|
||||||
|
constrinfo->separate = true;
|
||||||
|
|
||||||
|
indxinfo[j].indexconstraint = constrinfo->dobj.dumpId;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -11878,6 +11879,7 @@ dumpFunc(Archive *fout, const FuncInfo *finfo)
|
|||||||
char *funcsig; /* identity signature */
|
char *funcsig; /* identity signature */
|
||||||
char *funcfullsig = NULL; /* full signature */
|
char *funcfullsig = NULL; /* full signature */
|
||||||
char *funcsig_tag;
|
char *funcsig_tag;
|
||||||
|
char *qual_funcsig;
|
||||||
char *proretset;
|
char *proretset;
|
||||||
char *prosrc;
|
char *prosrc;
|
||||||
char *probin;
|
char *probin;
|
||||||
@ -12168,15 +12170,17 @@ dumpFunc(Archive *fout, const FuncInfo *finfo)
|
|||||||
|
|
||||||
funcsig_tag = format_function_signature(fout, finfo, false);
|
funcsig_tag = format_function_signature(fout, finfo, false);
|
||||||
|
|
||||||
|
qual_funcsig = psprintf("%s.%s",
|
||||||
|
fmtId(finfo->dobj.namespace->dobj.name),
|
||||||
|
funcsig);
|
||||||
|
|
||||||
if (prokind[0] == PROKIND_PROCEDURE)
|
if (prokind[0] == PROKIND_PROCEDURE)
|
||||||
keyword = "PROCEDURE";
|
keyword = "PROCEDURE";
|
||||||
else
|
else
|
||||||
keyword = "FUNCTION"; /* works for window functions too */
|
keyword = "FUNCTION"; /* works for window functions too */
|
||||||
|
|
||||||
appendPQExpBuffer(delqry, "DROP %s %s.%s;\n",
|
appendPQExpBuffer(delqry, "DROP %s %s;\n",
|
||||||
keyword,
|
keyword, qual_funcsig);
|
||||||
fmtId(finfo->dobj.namespace->dobj.name),
|
|
||||||
funcsig);
|
|
||||||
|
|
||||||
appendPQExpBuffer(q, "CREATE %s %s.%s",
|
appendPQExpBuffer(q, "CREATE %s %s.%s",
|
||||||
keyword,
|
keyword,
|
||||||
@ -12329,9 +12333,7 @@ dumpFunc(Archive *fout, const FuncInfo *finfo)
|
|||||||
|
|
||||||
append_depends_on_extension(fout, q, &finfo->dobj,
|
append_depends_on_extension(fout, q, &finfo->dobj,
|
||||||
"pg_catalog.pg_proc", keyword,
|
"pg_catalog.pg_proc", keyword,
|
||||||
psprintf("%s.%s",
|
qual_funcsig);
|
||||||
fmtId(finfo->dobj.namespace->dobj.name),
|
|
||||||
funcsig));
|
|
||||||
|
|
||||||
if (dopt->binary_upgrade)
|
if (dopt->binary_upgrade)
|
||||||
binary_upgrade_extension_member(q, &finfo->dobj,
|
binary_upgrade_extension_member(q, &finfo->dobj,
|
||||||
@ -12376,6 +12378,7 @@ dumpFunc(Archive *fout, const FuncInfo *finfo)
|
|||||||
if (funcfullsig)
|
if (funcfullsig)
|
||||||
free(funcfullsig);
|
free(funcfullsig);
|
||||||
free(funcsig_tag);
|
free(funcsig_tag);
|
||||||
|
free(qual_funcsig);
|
||||||
if (allargtypes)
|
if (allargtypes)
|
||||||
free(allargtypes);
|
free(allargtypes);
|
||||||
if (argmodes)
|
if (argmodes)
|
||||||
@ -14768,6 +14771,8 @@ dumpForeignServer(Archive *fout, const ForeignServerInfo *srvinfo)
|
|||||||
srvinfo->rolname,
|
srvinfo->rolname,
|
||||||
srvinfo->dobj.catId, srvinfo->dobj.dumpId);
|
srvinfo->dobj.catId, srvinfo->dobj.dumpId);
|
||||||
|
|
||||||
|
PQclear(res);
|
||||||
|
|
||||||
free(qsrvname);
|
free(qsrvname);
|
||||||
|
|
||||||
destroyPQExpBuffer(q);
|
destroyPQExpBuffer(q);
|
||||||
|
Loading…
Reference in New Issue
Block a user