diff options
author | Paul "LeoNerd" Evans <leonerd@leonerd.org.uk> | 2020-06-27 18:10:47 +0100 |
---|---|---|
committer | Tony Cook <tony@develop-help.com> | 2020-12-09 12:00:56 +1100 |
commit | 8462890b4373d480bea3ff2c274614e9a78cbdc4 (patch) | |
tree | cdc6b9140665fcb04be0c3c45a31a929f0d125b0 /scope.c | |
parent | c07463d862e4832cc6a94200a77a3170ef2dca18 (diff) | |
download | perl-8462890b4373d480bea3ff2c274614e9a78cbdc4.tar.gz |
Implement SAVEt_STRLEN_SMALL
Most uses of SAVEt_STRLEN actually store small values; often zero.
Rather than using an entire U64-sized element for these values, it saves
space to use the same "SMALL" mechanism as other numerical values, like
SAVEt_INT_SMALL.
Diffstat (limited to 'scope.c')
-rw-r--r-- | scope.c | 20 |
1 files changed, 17 insertions, 3 deletions
@@ -512,14 +512,22 @@ Perl_save_I32(pTHX_ I32 *intp) void Perl_save_strlen(pTHX_ STRLEN *ptr) { + const IV i = *ptr; + UV type = ((I32)((U32)i << SAVE_TIGHT_SHIFT) | SAVEt_STRLEN_SMALL); + int size = 2; dSS_ADD; PERL_ARGS_ASSERT_SAVE_STRLEN; - SS_ADD_IV(*ptr); + if (UNLIKELY((I32)(type >> SAVE_TIGHT_SHIFT) != i)) { + SS_ADD_IV(*ptr); + type = SAVEt_STRLEN; + size++; + } + SS_ADD_PTR(ptr); - SS_ADD_UV(SAVEt_STRLEN); - SS_ADD_END(3); + SS_ADD_UV(type); + SS_ADD_END(size); } void @@ -840,6 +848,7 @@ static const U8 arg_counts[] = { 1, /* SAVEt_STACK_POS */ 1, /* SAVEt_READONLY_OFF */ 1, /* SAVEt_FREEPADNAME */ + 1, /* SAVEt_STRLEN_SMALL */ 2, /* SAVEt_AV */ 2, /* SAVEt_DESTRUCTOR */ 2, /* SAVEt_DESTRUCTOR_X */ @@ -1045,6 +1054,11 @@ Perl_leave_scope(pTHX_ I32 base) *(int*)a1.any_ptr = (int)a0.any_i32; break; + case SAVEt_STRLEN_SMALL: + a0 = ap[0]; + *(STRLEN*)a0.any_ptr = (STRLEN)(uv >> SAVE_TIGHT_SHIFT); + break; + case SAVEt_STRLEN: /* STRLEN/size_t ref */ a0 = ap[0]; a1 = ap[1]; *(STRLEN*)a1.any_ptr = (STRLEN)a0.any_iv; |