mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Fix intra-query memory leak when a SRF returns zero rows.
When looping around after finding that the set-returning function returned zero rows for the current input tuple, ExecProjectSet neglected to reset either of the two memory contexts it's responsible for cleaning out. Typically this wouldn't cause much problem, because once the SRF does return at least one row, the contexts would get reset on the next call. However, if the SRF returns no rows for many input tuples in succession, quite a lot of memory could be transiently consumed. To fix, make sure we reset both contexts while looping around. Per bug #18172 from Sergei Kornilov. Back-patch to all supported branches. Discussion: https://postgr.es/m/18172-9b8c5fc1d676ded3@postgresql.org
This commit is contained in:
@@ -72,20 +72,22 @@ ExecProjectSet(PlanState *pstate)
|
|||||||
return resultSlot;
|
return resultSlot;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Reset argument context to free any expression evaluation storage
|
|
||||||
* allocated in the previous tuple cycle. Note this can't happen until
|
|
||||||
* we're done projecting out tuples from a scan tuple, as ValuePerCall
|
|
||||||
* functions are allowed to reference the arguments for each returned
|
|
||||||
* tuple.
|
|
||||||
*/
|
|
||||||
MemoryContextReset(node->argcontext);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Get another input tuple and project SRFs from it.
|
* Get another input tuple and project SRFs from it.
|
||||||
*/
|
*/
|
||||||
for (;;)
|
for (;;)
|
||||||
{
|
{
|
||||||
|
/*
|
||||||
|
* Reset argument context to free any expression evaluation storage
|
||||||
|
* allocated in the previous tuple cycle. Note this can't happen
|
||||||
|
* until we're done projecting out tuples from a scan tuple, as
|
||||||
|
* ValuePerCall functions are allowed to reference the arguments for
|
||||||
|
* each returned tuple. However, if we loop around after finding that
|
||||||
|
* no rows are produced from a scan tuple, we should reset, to avoid
|
||||||
|
* leaking memory when many successive scan tuples produce no rows.
|
||||||
|
*/
|
||||||
|
MemoryContextReset(node->argcontext);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Retrieve tuples from the outer plan until there are no more.
|
* Retrieve tuples from the outer plan until there are no more.
|
||||||
*/
|
*/
|
||||||
@@ -111,6 +113,12 @@ ExecProjectSet(PlanState *pstate)
|
|||||||
*/
|
*/
|
||||||
if (resultSlot)
|
if (resultSlot)
|
||||||
return resultSlot;
|
return resultSlot;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* When we do loop back, we'd better reset the econtext again, just in
|
||||||
|
* case the SRF leaked some memory there.
|
||||||
|
*/
|
||||||
|
ResetExprContext(econtext);
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
|
Reference in New Issue
Block a user