Skip to content
This repository has been archived by the owner on Jun 1, 2023. It is now read-only.

Commit

Permalink
add Internals::gc()
Browse files Browse the repository at this point in the history
free empty SV arenas and empty body roots.
Note that is usually doesn't reclaim system memory, very often
it even increases memory usage.

opslab arenas should be deleted by cv_undef() already.

See GH #336
  • Loading branch information
rurban committed Nov 2, 2018
1 parent 963ae96 commit d5176ae
Show file tree
Hide file tree
Showing 11 changed files with 339 additions and 9 deletions.
1 change: 1 addition & 0 deletions MANIFEST
Original file line number Diff line number Diff line change
Expand Up @@ -6345,6 +6345,7 @@ t/bigmem/vec.t Check vec() handles large offsets
t/charset_tools.pl To aid in portable testing across platforms with different character sets
t/cmd/elsif.t See if else-if works
t/cmd/for.t See if for loops work
t/cmd/gc.t See if Internals::gc() works
t/cmd/mod.t See if statement modifiers work
t/cmd/subval.t See if subroutine values work
t/cmd/switch.t See if switch optimizations work
Expand Down
5 changes: 5 additions & 0 deletions embed.fnc
Original file line number Diff line number Diff line change
Expand Up @@ -1718,6 +1718,11 @@ Apd |void |sv_free |NULLOK SV *const sv
poMX |void |sv_free2 |NN SV *const sv|const U32 refcnt
: Used only in perl.c
pd |void |sv_free_arenas
#ifdef PERL_CORE
pd |void |opslab_gc |NN OPSLAB *slab
#endif
Apd |void |op_gc_arenas
Apd |void |sv_gc_arenas
Apd |char* |sv_gets |NN SV *const sv|NN PerlIO *const fp|STRLEN append
Apd |char* |sv_grow |NN SV *const sv|STRLEN newlen
Apd |void |sv_inc |NULLOK SV *const sv
Expand Down
3 changes: 3 additions & 0 deletions embed.h
Original file line number Diff line number Diff line change
Expand Up @@ -599,6 +599,7 @@
#define op_dump(a) Perl_op_dump(aTHX_ a)
#define op_dump_cv(a,b) Perl_op_dump_cv(aTHX_ a,b)
#define op_free(a) Perl_op_free(aTHX_ a)
#define op_gc_arenas() Perl_op_gc_arenas(aTHX)
#define op_linklist(a) Perl_op_linklist(aTHX_ a)
#define op_lvalue_flags(a,b,c) Perl_op_lvalue_flags(aTHX_ a,b,c)
#define op_null(a) Perl_op_null(aTHX_ a)
Expand Down Expand Up @@ -804,6 +805,7 @@
#define sv_eq_flags(a,b,c) Perl_sv_eq_flags(aTHX_ a,b,c)
#define sv_force_normal_flags(a,b) Perl_sv_force_normal_flags(aTHX_ a,b)
#define sv_free(a) Perl_sv_free(aTHX_ a)
#define sv_gc_arenas() Perl_sv_gc_arenas(aTHX)
#define sv_get_backrefs Perl_sv_get_backrefs
#define sv_gets(a,b,c) Perl_sv_gets(aTHX_ a,b,c)
#define sv_grow(a,b) Perl_sv_grow(aTHX_ a,b)
Expand Down Expand Up @@ -1830,6 +1832,7 @@
#define opslab_force_free(a) Perl_opslab_force_free(aTHX_ a)
#define opslab_free(a) Perl_opslab_free(aTHX_ a)
#define opslab_free_nopad(a) Perl_opslab_free_nopad(aTHX_ a)
#define opslab_gc(a) Perl_opslab_gc(aTHX_ a)
#define parser_free_nexttoke_ops(a,b) Perl_parser_free_nexttoke_ops(aTHX_ a,b)
#define should_warn_nl S_should_warn_nl
#define strip_spaces(a,b) S_strip_spaces(aTHX_ a,b)
Expand Down
11 changes: 11 additions & 0 deletions lib/Internals.pod
Original file line number Diff line number Diff line change
Expand Up @@ -65,11 +65,22 @@ Clear any placeholders from a locked hash. Should not be used directly.
You should use the wrapper functions providewd by Hash::Util instead.
As of 5.25 also available as C< Hash::Util::_clear_placeholders(%hash) >

=item gc()

Free any empty SV arena and empty body roots, and free all block
temporaries, thus should be only called at the end of blocks. Note
that this usually doesn't reclaim much system memory, and in certain
configurations it might even cause crashes.
With C<PERL_POISON> it is not needed.

Experimental, unstable and might go away soon.

=back

=head1 AUTHOR

Perl core development team.
cperl development team.

=head1 SEE ALSO

Expand Down
55 changes: 55 additions & 0 deletions op.c
Original file line number Diff line number Diff line change
Expand Up @@ -779,6 +779,61 @@ Perl_opslab_force_free(pTHX_ OPSLAB *slab)
opslab_free(slab);
}

/*
=for apidoc Apd|void |op_gc_arenas

Deallocate memory used by freed obslab arenas. Note that if one slab
contains at least one live OP the whole arena stays live.

=cut
*/
void
Perl_op_gc_arenas(pTHX)
{
/* TODO walk all CVs to look for empty arenas */
if (!PL_compcv
|| CvROOT(PL_compcv)
|| (CvSTART(PL_compcv) && !CvSLABBED(PL_compcv)))
{
return; /* not slabbed */
} else {
opslab_gc((OPSLAB *)CvSTART(PL_compcv));
}
}

/*
=for apidoc pd|void |opslab_gc

Deallocate memory used by a freed obslab arena. Note that if the slab
contains at least one live OP the whole arena stays live.

=cut
*/
void
Perl_opslab_gc(pTHX_ OPSLAB *slab)
{
OPSLAB *slab2 = slab;
PERL_ARGS_ASSERT_OPSLAB_GC;
DEBUG_S_warn((aTHX_ "gc slab %p", (void*)slab));

do {
OPSLOT *slot;
for (slot = slab2->opslab_first;
slot->opslot_next;
slot = slot->opslot_next)
{
if (slot->opslot_op.op_type != OP_FREED) {
assert(slot->opslot_op.op_slabbed);
goto next_slab;
}
}
if (slab->opslab_refcnt == 1)
opslab_free(slab);
next_slab:
;
} while ((slab2 = slab2->opslab_next));
}

#ifdef PERL_DEBUG_READONLY_OPS
OP *
Perl_op_refcnt_inc(pTHX_ OP *o)
Expand Down
6 changes: 6 additions & 0 deletions pod/perlcdelta.pod
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,12 @@ L</Selected Bug Fixes> section.

[ List each security issue as a =head2 entry ]

=head2 Added experimental Internals::gc()

With the ability to free empty SV arenas. It's currently unstable, and
might be replaced by proper arena memory handling later. See
L<Internals/gc> and L<perlapi/sv_gc_arenas>.

=head1 Incompatible Changes

XXX For a release on a stable branch, this section aspires to be:
Expand Down
11 changes: 11 additions & 0 deletions proto.h
Original file line number Diff line number Diff line change
Expand Up @@ -4444,6 +4444,9 @@ PERL_CALLCONV void Perl_op_dump_cv(pTHX_ const OP *o, const CV *cv)
PERL_CALLCONV void Perl_op_free(pTHX_ OP* arg)
__attribute__global__;

PERL_CALLCONV void Perl_op_gc_arenas(pTHX)
__attribute__global__;

PERL_CALLCONV OP* Perl_op_linklist(pTHX_ OP *o)
__attribute__global__
__attribute__nonnull__(pTHX_1);
Expand Down Expand Up @@ -5997,6 +6000,9 @@ PERL_CALLCONV void Perl_sv_free2(pTHX_ SV *const sv, const U32 refcnt)
assert(sv)

PERL_CALLCONV void Perl_sv_free_arenas(pTHX);
PERL_CALLCONV void Perl_sv_gc_arenas(pTHX)
__attribute__global__;

PERL_CALLCONV SV* Perl_sv_get_backrefs(SV *const sv)
__attribute__global__
__attribute__nonnull__(1);
Expand Down Expand Up @@ -8113,6 +8119,11 @@ PERL_CALLCONV void Perl_opslab_free_nopad(pTHX_ OPSLAB *slab)
#define PERL_ARGS_ASSERT_OPSLAB_FREE_NOPAD \
assert(slab)

PERL_CALLCONV void Perl_opslab_gc(pTHX_ OPSLAB *slab)
__attribute__nonnull__(pTHX_1);
#define PERL_ARGS_ASSERT_OPSLAB_GC \
assert(slab)

PERL_CALLCONV void Perl_parser_free_nexttoke_ops(pTHX_ yy_parser *parser, OPSLAB *slab)
__attribute__nonnull__(pTHX_1)
__attribute__nonnull__(pTHX_2);
Expand Down
3 changes: 2 additions & 1 deletion regcomp.c
Original file line number Diff line number Diff line change
Expand Up @@ -20234,7 +20234,8 @@ Perl_pregfree2(pTHX_ REGEXP *rx)
if (r->mother_re) {
ReREFCNT_dec(r->mother_re);
} else {
CALLREGFREE_PVT(rx); /* free the private data */
if (LIKELY(r->engine)) /* gc() might have deleted this body already */
CALLREGFREE_PVT(rx); /* free the private data */
SvREFCNT_dec(RXp_PAREN_NAMES(r));
}
if (r->substrs) {
Expand Down
Loading

0 comments on commit d5176ae

Please sign in to comment.