summaryrefslogtreecommitdiff
path: root/scope.c
diff options
context:
space:
mode:
authorPaul "LeoNerd" Evans <leonerd@leonerd.org.uk>2020-06-27 18:10:47 +0100
committerTony Cook <tony@develop-help.com>2020-12-09 12:00:56 +1100
commit8462890b4373d480bea3ff2c274614e9a78cbdc4 (patch)
treecdc6b9140665fcb04be0c3c45a31a929f0d125b0 /scope.c
parentc07463d862e4832cc6a94200a77a3170ef2dca18 (diff)
downloadperl-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.c20
1 files changed, 17 insertions, 3 deletions
diff --git a/scope.c b/scope.c
index cea1500e6a..19281d12a7 100644
--- a/scope.c
+++ b/scope.c
@@ -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;