-
Notifications
You must be signed in to change notification settings - Fork 571
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Replace some sv_2mortal() instances with more direct equivalents #23153
Conversation
@@ -4189,7 +4189,8 @@ Perl_do_readline(pTHX) | |||
} | |||
else { | |||
/* XXX on RC builds, push on stack rather than mortalize ? */ | |||
sv = sv_2mortal(newSV(80)); | |||
sv = newSV_type(SVt_PV); | |||
sv_grow_fresh(sv, 81); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What end the life of the sv head now that mortal is missing?
Why 81 bytes when old was 80?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What end the life of the sv head now that mortal is missing?
Thanks, it was meant to be newSV_type_mortal
, I'll change that now.
Why 81 bytes when old was 80?
It's to keep the allocation size consistent: Perl_newSV
calls sv_grow_fresh(sv, len + 1);
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why 81 bytes when old was 80?
It's to keep the allocation size consistent:
Perl_newSV
callssv_grow_fresh(sv, len + 1);
.
Be very very careful and single step it line by line if your making assumptions about +1
s and perl's PV/NewX API. This class of bugs already is sitting in stable and blead perl. "can't introduce an exploit" "safe than sorry" +1 +1
, another 16 or 32 bytes just disappeared times a couple 1000. All these +1s and the PERL_STR_RNDUP macro are very fragile. Open PR with this class of bug
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
update:
the off by 1 in
goes back to your commit at
64b4056614429bb6fc7d35118e19a220459358fd
I suggest you single step the C code and verify what integer from the interp is entering glibc.so func since you/nobody else (including me) understands this area of the SVPV API anymore. Short made up demo math of the details in #23153
80/8=10, aligned!
(81+7)&(~7)=88
(81+15)&(~15)=96
Atleast use bytes 82-95 on a PP C level if you paided for them from libc, but this 1+1+15 then !0xF mask over and ever. needs to be stopped.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The +1 is done by the normal Perl_sv_grow() for the default COW build as well, sv_grow_fresh
simply duplicates that.
That said, the exact value doesn't matter too much.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This commit does create/leave behind 100% unused bytes (~1 upto ~15 bytes). But probably the old code before this commit also did (an 80 byte null term PV will always be 81 bytes of real VM). So the "fix" is moving the byte length const to 72-1 or 72-2, or 88-1 or 88-2 or 96-1 or 96-2. But count 80 is the magic shell terminal line length since 1970, which might mean something in this particular area of the interp, so 72 or 96 might change some other runtime observable behavior without reading/studying more src code.
Test code to play around with on 5.41.7 winperl64.
Malloc bucket jump happens at 81 and 97. Don't ask me why CRT realloc() / HeapReAlloc() / RtlReallocateHeap() move the memory block 100% redundantly between a pair of constant addresses on each realloc() +1 call.
Spectre security fix by MS ?????
Someone else is free to try out my demo code on Win11 or Glibc or OSX.
perl cppinl.pl 71 71 1
#use Inline ('force', 'noclean');
use Inline C => Config =>
PRE_HEAD => '#define PERL_NO_GET_CONTEXT 1';
use Inline C => Config => BUILD_NOISY => 1;
use Inline C => <<'END_OF_C_CODE';
#define tm(_sv) if(1) { \
int lc=0; \
for( ;lc < 64; lc++) { \
char * p = SvPVX(_sv); \
char *op = p; \
size_t ol = SvLEN(_sv); \
size_t pl = ol+1; \
Renew(p, pl, char); \
SvPV_set(_sv, p); \
SvLEN_set(_sv, pl); \
if(p != op) \
warn("try %3d 0x%x b4 %p af %p\n", pl, pl, p, op); \
}}
SV* do3(int i1, int i2, int dtorsv1) {
dTHX;
SV* s1 = newSV(i1);
warn("**newSV**\n");
sv_dump(s1);
if (dtorsv1) {
warn("**newSV**\n");
tm(s1);
SvREFCNT_dec_NN(s1);
}
SV* s2 = newSV_type(SVt_PV);
sv_grow_fresh(s2, i2);
warn("**grow_fresh**\n");
sv_dump(s2);
if (! dtorsv1) {
warn("**newSV**\n");
tm(s1);
}
warn("**grow_fresh**\n");
tm(s2);
return s2;
}
END_OF_C_CODE
$, = "\n";
use v5.41;
do3($ARGV[0],$ARGV[1],$ARGV[2]);
**newSV**
SV = PV(0x34ac38) at 0x349fa0
REFCNT = 1
FLAGS = ()
PV = 0x3781b8 ""
CUR = 0
LEN = 73
**newSV**
try 74 0x4a b4 378298 af 3781b8
try 75 0x4b b4 3781b8 af 378298
try 76 0x4c b4 378298 af 3781b8
try 77 0x4d b4 3781b8 af 378298
try 78 0x4e b4 378298 af 3781b8
try 79 0x4f b4 3781b8 af 378298
try 80 0x50 b4 378298 af 3781b8
try 81 0x51 b4 3665c8 af 378298
try 82 0x52 b4 3666c8 af 3665c8
try 83 0x53 b4 3665c8 af 3666c8
try 84 0x54 b4 3666c8 af 3665c8
try 85 0x55 b4 3665c8 af 3666c8
try 86 0x56 b4 3666c8 af 3665c8
try 87 0x57 b4 3665c8 af 3666c8
try 88 0x58 b4 3666c8 af 3665c8
try 89 0x59 b4 3665c8 af 3666c8
try 90 0x5a b4 3666c8 af 3665c8
try 91 0x5b b4 3665c8 af 3666c8
try 92 0x5c b4 3666c8 af 3665c8
try 93 0x5d b4 3665c8 af 3666c8
try 94 0x5e b4 3666c8 af 3665c8
try 95 0x5f b4 3665c8 af 3666c8
try 96 0x60 b4 3666c8 af 3665c8
try 97 0x61 b4 2398748 af 3666c8
try 98 0x62 b4 2398868 af 2398748
try 99 0x63 b4 2398748 af 2398868
try 100 0x64 b4 2398868 af 2398748
try 101 0x65 b4 2398748 af 2398868
try 102 0x66 b4 2398868 af 2398748
try 103 0x67 b4 2398748 af 2398868
try 104 0x68 b4 2398868 af 2398748
try 105 0x69 b4 2398748 af 2398868
try 106 0x6a b4 2398868 af 2398748
try 107 0x6b b4 2398748 af 2398868
try 108 0x6c b4 2398868 af 2398748
try 109 0x6d b4 2398748 af 2398868
try 110 0x6e b4 2398868 af 2398748
try 111 0x6f b4 2398748 af 2398868
try 112 0x70 b4 2398868 af 2398748
try 113 0x71 b4 23f32a8 af 2398868
try 114 0x72 b4 23f3168 af 23f32a8
try 115 0x73 b4 23f32a8 af 23f3168
try 116 0x74 b4 23f3168 af 23f32a8
try 117 0x75 b4 23f32a8 af 23f3168
try 118 0x76 b4 23f3168 af 23f32a8
try 119 0x77 b4 23f32a8 af 23f3168
try 120 0x78 b4 23f3168 af 23f32a8
try 121 0x79 b4 23f32a8 af 23f3168
try 122 0x7a b4 23f3168 af 23f32a8
try 123 0x7b b4 23f32a8 af 23f3168
try 124 0x7c b4 23f3168 af 23f32a8
try 125 0x7d b4 23f32a8 af 23f3168
try 126 0x7e b4 23f3168 af 23f32a8
try 127 0x7f b4 23f32a8 af 23f3168
try 128 0x80 b4 23f3168 af 23f32a8
try 129 0x81 b4 390028 af 23f3168
try 130 0x82 b4 390188 af 390028
try 131 0x83 b4 390028 af 390188
try 132 0x84 b4 390188 af 390028
try 133 0x85 b4 390028 af 390188
try 134 0x86 b4 390188 af 390028
try 135 0x87 b4 390028 af 390188
try 136 0x88 b4 390188 af 390028
try 137 0x89 b4 390028 af 390188
**grow_fresh**
SV = PV(0x34ac38) at 0x349fa0
REFCNT = 1
FLAGS = ()
PV = 0x378298 ""
CUR = 0
LEN = 72
**grow_fresh**
try 73 0x49 b4 3781b8 af 378298
try 74 0x4a b4 378298 af 3781b8
try 75 0x4b b4 3781b8 af 378298
try 76 0x4c b4 378298 af 3781b8
try 77 0x4d b4 3781b8 af 378298
try 78 0x4e b4 378298 af 3781b8
try 79 0x4f b4 3781b8 af 378298
try 80 0x50 b4 378298 af 3781b8
try 81 0x51 b4 3666c8 af 378298
try 82 0x52 b4 3665c8 af 3666c8
try 83 0x53 b4 3666c8 af 3665c8
try 84 0x54 b4 3665c8 af 3666c8
try 85 0x55 b4 3666c8 af 3665c8
try 86 0x56 b4 3665c8 af 3666c8
try 87 0x57 b4 3666c8 af 3665c8
try 88 0x58 b4 3665c8 af 3666c8
try 89 0x59 b4 3666c8 af 3665c8
try 90 0x5a b4 3665c8 af 3666c8
try 91 0x5b b4 3666c8 af 3665c8
try 92 0x5c b4 3665c8 af 3666c8
try 93 0x5d b4 3666c8 af 3665c8
try 94 0x5e b4 3665c8 af 3666c8
try 95 0x5f b4 3666c8 af 3665c8
try 96 0x60 b4 3665c8 af 3666c8
try 97 0x61 b4 2398868 af 3665c8
try 98 0x62 b4 2398748 af 2398868
try 99 0x63 b4 2398868 af 2398748
try 100 0x64 b4 2398748 af 2398868
try 101 0x65 b4 2398868 af 2398748
try 102 0x66 b4 2398748 af 2398868
try 103 0x67 b4 2398868 af 2398748
try 104 0x68 b4 2398748 af 2398868
try 105 0x69 b4 2398868 af 2398748
try 106 0x6a b4 2398748 af 2398868
try 107 0x6b b4 2398868 af 2398748
try 108 0x6c b4 2398748 af 2398868
try 109 0x6d b4 2398868 af 2398748
try 110 0x6e b4 2398748 af 2398868
try 111 0x6f b4 2398868 af 2398748
try 112 0x70 b4 2398748 af 2398868
try 113 0x71 b4 23f3168 af 2398748
try 114 0x72 b4 23f32a8 af 23f3168
try 115 0x73 b4 23f3168 af 23f32a8
try 116 0x74 b4 23f32a8 af 23f3168
try 117 0x75 b4 23f3168 af 23f32a8
try 118 0x76 b4 23f32a8 af 23f3168
try 119 0x77 b4 23f3168 af 23f32a8
try 120 0x78 b4 23f32a8 af 23f3168
try 121 0x79 b4 23f3168 af 23f32a8
try 122 0x7a b4 23f32a8 af 23f3168
try 123 0x7b b4 23f3168 af 23f32a8
try 124 0x7c b4 23f32a8 af 23f3168
try 125 0x7d b4 23f3168 af 23f32a8
try 126 0x7e b4 23f32a8 af 23f3168
try 127 0x7f b4 23f3168 af 23f32a8
try 128 0x80 b4 23f32a8 af 23f3168
try 129 0x81 b4 390028 af 23f32a8
try 130 0x82 b4 390188 af 390028
try 131 0x83 b4 390028 af 390188
try 132 0x84 b4 390188 af 390028
try 133 0x85 b4 390028 af 390188
try 134 0x86 b4 390188 af 390028
try 135 0x87 b4 390028 af 390188
try 136 0x88 b4 390188 af 390028
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But count 80 is the magic shell terminal line length since 1970, which might mean something in this particular area of the interp, so 72 or 96 might change some other runtime observable behavior without reading/studying more src code.
Yeah, this was the main motivation for preserving the 80 + 1
behaviour in that commit.
89e85e9
to
981d0f7
Compare
A few commits making this sort of change:
sv_2mortal(newSVpvn_utf8(SvPVX(sv), namelen, do_utf8));
->
newSVpvn_flags(SvPVX(sv), namelen, SVs_TEMP|do_utf8);
newSVpvn_utf8(key, klen, TRUE);
+sv_2mortal()
->
newSVpvn_flags(key, klen, SVf_UTF8|SVs_TEMP);
(AV *) sv_2mortal((SV *) newAV())
->
MUTABLE_AV(newSV_type_mortal(SVt_PVAV));
The changes are intended to be functionally equivalent but do the work
through one function call rather than two.