On 7/27/22 13:20, demerphq wrote:
On Wed, 27 Jul 2022 at 17:46, Mark Murawski <markm-li...@intellasoft.net> wrote:

    Hi All!

    I'm working on a new feature for plperl in postgresql to populate
    some
    metadata in main::_FN for running plperl functions from postgres sql.

    I've snipped out some extra code that's unrelated to focus on the
    issue
    at hand.   If the section labeled 'Leaking Section' is entirely
    commented out (and of course the related SvREFCNT_dec_current is
    commented as well), then there's no memory issue at all.  If I do use
    this section of code, something is not freed and I'm not sure what
    that
    is, since I'm very new to perl-c



    /*
      * Decrement the refcount of the given SV within the active Perl
    interpreter
      *
      * This is handy because it reloads the active-interpreter
    pointer, saving
      * some notation in callers that switch the active interpreter.
      */
    static inline void
    SvREFCNT_dec_current(SV *sv)
    {
         dTHX;

         SvREFCNT_dec(sv);
    }

    static SV  *
    plperl_call_perl_func(plperl_proc_desc *desc, FunctionCallInfo fcinfo)
    {
         dTHX;
         dSP;

         HV                 *hv;                  // hash
         SV                 *FNsv;                // scalar reference
    to the
    hash
         SV                 *svFN;                // local reference
    to the
    hash?

         ENTER;
         SAVETMPS;

         /* Give functions some metadata about what's going on in $_FN
    (Similar to $_TD for triggers) */

         // Leaking Section {
         FNsv = get_sv("main::_FN", GV_ADD);
         save_item(FNsv);                        /* local $_FN */

         hv = newHV(); // create new hash
         hv_ksplit(hv, 12);                      /* pre-grow the hash */
         hv_store_string(hv, "name", cstr2sv(desc->proname));

         svFN = newRV_noinc((SV *) hv); // reference to the new hash
         sv_setsv(FNsv, svFN);


So at this point FNsv and svFN both are references to the same HV. But you dont free svFN.

Likely a simple SvREFCNT_dec(svFN) would do it.

Basically you are nearly doing this:

for my $FNsv ($main::FN) {
    my %hash;
    my $svFN= \%hash; push @LEAK, \$svFN;
    $FNsv= $svFN;
}

If you put a sv_2mortal(svFN) or an explicit refcount decrement on svFN then I expect the leak would go away. Note the reason i use a for loop here is I am trying to emphasize that $FNsv IS $main::FN, as the topic of a for loop is an alias which is as close as you can get in Perl to the C level operation of copying a pointer to an SV structure. (The difference being that in Perl aliases are refcounted copies and C they are not refcounted). Compare this to $svFN which is clearly a new variable.

What you should be doing is this:

my %hash;
$main::FN= \%hash;

Which would be something like this:


sv_upgrade(FNsv,SVtRV);
SvRV_set(FNsv,(SV*)newHV());
SvROK_on(FNsv);


         // Leaking Section }

         // dostuff

         SvREFCNT_dec_current(hv);

         PUTBACK;
         FREETMPS;
         LEAVE;

         ...snip...
    }


    If anyone would like to see the full context, I've attached the
    entire
    file.  My additions are between the 'New..........' sections

    My question is... the perl-c api docs do not make it clear for which
    allocations or accesses that you need to decrement the ref count.


I would say the docs actually do explain this. Most functions will specify that you need to do the refcount incrementing yourself if it is relevant.

A general rule of thumb however is that any item you create yourself which is not "owned by perl" by being attached to some data structure it exposes is your problem to deal with. So for instance this:

     FNsv = get_sv("main::_FN", GV_ADD);

is getting you a local pointer to the structure representing $main::_FN which is a global var. Thus it is stored in the global stash, and thus Perl knows about it and will clean it up, you don't need to worry about its refcount unless you store it in something else that is refcount managed.

On the other hand

     hv = newHV(); // create new hash

is a pointer to a new HV structure which is not stored in any place that perl knows about. So if you dont arrange for it to be recount decremented it will leak. However you then did this:

     svFN = newRV_noinc((SV *) hv); // reference to the new hash

this is creating a new reference to the hash. The combination of the two basically is equivalent to creating an anonymous hashref. The "noinc" is there because the hv starts off with a refcount of 1, and the new reference is the thing that will "own" that refcount. So at this point you no longer need to manage 'hv' provided you correctly manage svFN.

You then do this:

sv_setsv(FNsv, svFN);

This results in the refcount of 'hv' being incremented, as there are now two RV's pointing at it. You need to free up the temporary, or even better, simply dont use it. You dont need it, you can simply turn FNsv into an RV, and then set its RV field appropriately, the leak will go away and the code will be more efficient.

Another comment is regarding the hv_ksplit() which seems redundant. The initial number of buckets is 8, the new size up is 12 or 16. Setting it to 12 is likely just a waste. Either set it much larger, or dont use it at all.

Yves



--
perl -Mre=debug -e "/just|another|perl|hacker/"


Spectacular.  I'll be reviewing and making changes that you're suggesting and this helps my understanding for sure.

Thanks!


Reply via email to