a4ac56
From 4ac7295514f35016a79dbcc07500f6c9ca4729b7 Mon Sep 17 00:00:00 2001
a4ac56
From: Tony Cook <tony@develop-help.com>
a4ac56
Date: Thu, 2 Nov 2017 20:18:56 +0000
a4ac56
Subject: [PATCH] (perl #131895) fail stat on names with \0 embedded
a4ac56
MIME-Version: 1.0
a4ac56
Content-Type: text/plain; charset=UTF-8
a4ac56
Content-Transfer-Encoding: 8bit
a4ac56
a4ac56
Also lstat() and the file test ops.
a4ac56
a4ac56
Petr Písař: Port to 5.26.1.
a4ac56
a4ac56
Signed-off-by: Petr Písař <ppisar@redhat.com>
a4ac56
---
a4ac56
 doio.c                | 21 ++++++++++++++++-----
a4ac56
 pp_sys.c              | 29 +++++++++++++++++++++++------
a4ac56
 t/lib/warnings/pp_sys | 14 ++++++++++++++
a4ac56
 t/op/filetest.t       | 10 +++++++++-
a4ac56
 t/op/stat.t           | 12 +++++++++++-
a4ac56
 5 files changed, 73 insertions(+), 13 deletions(-)
a4ac56
a4ac56
diff --git a/doio.c b/doio.c
a4ac56
index becb19b..70d7747 100644
a4ac56
--- a/doio.c
a4ac56
+++ b/doio.c
a4ac56
@@ -1466,7 +1466,7 @@ Perl_my_stat_flags(pTHX_ const U32 flags)
a4ac56
 	return PL_laststatval;
a4ac56
     else {
a4ac56
 	SV* const sv = TOPs;
a4ac56
-	const char *s;
a4ac56
+	const char *s, *d;
a4ac56
 	STRLEN len;
a4ac56
 	if ((gv = MAYBE_DEREF_GV_flags(sv,flags))) {
a4ac56
 	    goto do_fstat;
a4ac56
@@ -1480,9 +1480,14 @@ Perl_my_stat_flags(pTHX_ const U32 flags)
a4ac56
 	s = SvPV_flags_const(sv, len, flags);
a4ac56
 	PL_statgv = NULL;
a4ac56
 	sv_setpvn(PL_statname, s, len);
a4ac56
-	s = SvPVX_const(PL_statname);		/* s now NUL-terminated */
a4ac56
+	d = SvPVX_const(PL_statname);		/* s now NUL-terminated */
a4ac56
 	PL_laststype = OP_STAT;
a4ac56
-	PL_laststatval = PerlLIO_stat(s, &PL_statcache);
a4ac56
+        if (!IS_SAFE_PATHNAME(s, len, OP_NAME(PL_op))) {
a4ac56
+            PL_laststatval = -1;
a4ac56
+        }
a4ac56
+        else {
a4ac56
+            PL_laststatval = PerlLIO_stat(d, &PL_statcache);
a4ac56
+        }
a4ac56
 	if (PL_laststatval < 0 && ckWARN(WARN_NEWLINE) && should_warn_nl(s)) {
a4ac56
             GCC_DIAG_IGNORE(-Wformat-nonliteral); /* PL_warn_nl is constant */
a4ac56
 	    Perl_warner(aTHX_ packWARN(WARN_NEWLINE), PL_warn_nl, "stat");
a4ac56
@@ -1499,6 +1504,7 @@ Perl_my_lstat_flags(pTHX_ const U32 flags)
a4ac56
     static const char* const no_prev_lstat = "The stat preceding -l _ wasn't an lstat";
a4ac56
     dSP;
a4ac56
     const char *file;
a4ac56
+    STRLEN len;
a4ac56
     SV* const sv = TOPs;
a4ac56
     bool isio = FALSE;
a4ac56
     if (PL_op->op_flags & OPf_REF) {
a4ac56
@@ -1542,9 +1548,14 @@ Perl_my_lstat_flags(pTHX_ const U32 flags)
a4ac56
                               HEKfARG(GvENAME_HEK((const GV *)
a4ac56
                                           (SvROK(sv) ? SvRV(sv) : sv))));
a4ac56
     }
a4ac56
-    file = SvPV_flags_const_nolen(sv, flags);
a4ac56
+    file = SvPV_flags_const(sv, len, flags);
a4ac56
     sv_setpv(PL_statname,file);
a4ac56
-    PL_laststatval = PerlLIO_lstat(file,&PL_statcache);
a4ac56
+    if (!IS_SAFE_PATHNAME(file, len, OP_NAME(PL_op))) {
a4ac56
+        PL_laststatval = -1;
a4ac56
+    }
a4ac56
+    else {
a4ac56
+        PL_laststatval = PerlLIO_lstat(file,&PL_statcache);
a4ac56
+    }
a4ac56
     if (PL_laststatval < 0 && ckWARN(WARN_NEWLINE) && should_warn_nl(file)) {
a4ac56
         GCC_DIAG_IGNORE(-Wformat-nonliteral); /* PL_warn_nl is constant */
a4ac56
         Perl_warner(aTHX_ packWARN(WARN_NEWLINE), PL_warn_nl, "lstat");
a4ac56
diff --git a/pp_sys.c b/pp_sys.c
a4ac56
index 0b60584..1b81fda 100644
a4ac56
--- a/pp_sys.c
a4ac56
+++ b/pp_sys.c
a4ac56
@@ -2963,19 +2963,24 @@ PP(pp_stat)
a4ac56
     }
a4ac56
     else {
a4ac56
         const char *file;
a4ac56
+        const char *temp;
a4ac56
+        STRLEN len;
a4ac56
 	if (SvROK(sv) && SvTYPE(SvRV(sv)) == SVt_PVIO) { 
a4ac56
             io = MUTABLE_IO(SvRV(sv));
a4ac56
             if (PL_op->op_type == OP_LSTAT)
a4ac56
                 goto do_fstat_warning_check;
a4ac56
             goto do_fstat_have_io; 
a4ac56
         }
a4ac56
-        
a4ac56
 	SvTAINTED_off(PL_statname); /* previous tainting irrelevant */
a4ac56
-	sv_setpv(PL_statname, SvPV_nomg_const_nolen(sv));
a4ac56
+        temp = SvPV_nomg_const(sv, len);
a4ac56
+	sv_setpv(PL_statname, temp);
a4ac56
 	PL_statgv = NULL;
a4ac56
 	PL_laststype = PL_op->op_type;
a4ac56
         file = SvPV_nolen_const(PL_statname);
a4ac56
-	if (PL_op->op_type == OP_LSTAT)
a4ac56
+        if (!IS_SAFE_PATHNAME(temp, len, OP_NAME(PL_op))) {
a4ac56
+            PL_laststatval = -1;
a4ac56
+        }
a4ac56
+	else if (PL_op->op_type == OP_LSTAT)
a4ac56
 	    PL_laststatval = PerlLIO_lstat(file, &PL_statcache);
a4ac56
 	else
a4ac56
 	    PL_laststatval = PerlLIO_stat(file, &PL_statcache);
a4ac56
@@ -3211,8 +3216,12 @@ PP(pp_ftrread)
a4ac56
 
a4ac56
     if (use_access) {
a4ac56
 #if defined(HAS_ACCESS) || defined (PERL_EFF_ACCESS)
a4ac56
-	const char *name = SvPV_nolen(*PL_stack_sp);
a4ac56
-	if (effective) {
a4ac56
+        STRLEN len;
a4ac56
+	const char *name = SvPV(*PL_stack_sp, len);
a4ac56
+        if (!IS_SAFE_PATHNAME(name, len, OP_NAME(PL_op))) {
a4ac56
+            result = -1;
a4ac56
+        }
a4ac56
+	else if (effective) {
a4ac56
 #  ifdef PERL_EFF_ACCESS
a4ac56
 	    result = PERL_EFF_ACCESS(name, access_mode);
a4ac56
 #  else
a4ac56
@@ -3537,10 +3546,18 @@ PP(pp_fttext)
a4ac56
     }
a4ac56
     else {
a4ac56
         const char *file;
a4ac56
+        const char *temp;
a4ac56
+        STRLEN temp_len;
a4ac56
         int fd; 
a4ac56
 
a4ac56
         assert(sv);
a4ac56
-	sv_setpv(PL_statname, SvPV_nomg_const_nolen(sv));
a4ac56
+        temp = SvPV_nomg_const(sv, temp_len);
a4ac56
+	sv_setpv(PL_statname, temp);
a4ac56
+        if (!IS_SAFE_PATHNAME(temp, temp_len, OP_NAME(PL_op))) {
a4ac56
+            PL_laststatval = -1;
a4ac56
+            PL_laststype = OP_STAT;
a4ac56
+            FT_RETURNUNDEF;
a4ac56
+        }
a4ac56
       really_filename:
a4ac56
         file = SvPVX_const(PL_statname);
a4ac56
 	PL_statgv = NULL;
a4ac56
diff --git a/t/lib/warnings/pp_sys b/t/lib/warnings/pp_sys
a4ac56
index 9c544e0..c599aa3 100644
a4ac56
--- a/t/lib/warnings/pp_sys
a4ac56
+++ b/t/lib/warnings/pp_sys
a4ac56
@@ -972,3 +972,17 @@ close $fh;
a4ac56
 unlink $file;
a4ac56
 EXPECT
a4ac56
 syswrite() is deprecated on :utf8 handles. This will be a fatal error in Perl 5.30 at - line 5.
a4ac56
+########
a4ac56
+# NAME stat on name with \0
a4ac56
+use warnings;
a4ac56
+my @x = stat("./\0-");
a4ac56
+my @y = lstat("./\0-");
a4ac56
+-T ".\0-";
a4ac56
+-x ".\0-";
a4ac56
+-l ".\0-";
a4ac56
+EXPECT
a4ac56
+Invalid \0 character in pathname for stat: ./\0- at - line 2.
a4ac56
+Invalid \0 character in pathname for lstat: ./\0- at - line 3.
a4ac56
+Invalid \0 character in pathname for fttext: .\0- at - line 4.
a4ac56
+Invalid \0 character in pathname for fteexec: .\0- at - line 5.
a4ac56
+Invalid \0 character in pathname for ftlink: .\0- at - line 6.
a4ac56
diff --git a/t/op/filetest.t b/t/op/filetest.t
a4ac56
index 8883381..bd1d08c 100644
a4ac56
--- a/t/op/filetest.t
a4ac56
+++ b/t/op/filetest.t
a4ac56
@@ -9,7 +9,7 @@ BEGIN {
a4ac56
     set_up_inc(qw '../lib ../cpan/Perl-OSType/lib');
a4ac56
 }
a4ac56
 
a4ac56
-plan(tests => 53 + 27*14);
a4ac56
+plan(tests => 57 + 27*14);
a4ac56
 
a4ac56
 if ($^O =~ /MSWin32|cygwin|msys/ && !is_miniperl) {
a4ac56
   require Win32; # for IsAdminUser()
a4ac56
@@ -393,3 +393,11 @@ SKIP: {
a4ac56
     is $failed_stat2, $failed_stat1,
a4ac56
 	'failed -r($gv_with_io_but_no_fp) with and w/out fatal warnings';
a4ac56
 } 
a4ac56
+
a4ac56
+{
a4ac56
+    # [perl #131895] stat() doesn't fail on filenames containing \0 / NUL
a4ac56
+    ok(!-T "TEST\0-", '-T on name with \0');
a4ac56
+    ok(!-B "TEST\0-", '-B on name with \0');
a4ac56
+    ok(!-f "TEST\0-", '-f on name with \0');
a4ac56
+    ok(!-r "TEST\0-", '-r on name with \0');
a4ac56
+}
a4ac56
diff --git a/t/op/stat.t b/t/op/stat.t
a4ac56
index 323c498..dbbe6ec 100644
a4ac56
--- a/t/op/stat.t
a4ac56
+++ b/t/op/stat.t
a4ac56
@@ -25,7 +25,7 @@ if ($^O eq 'MSWin32') {
a4ac56
     ${^WIN32_SLOPPY_STAT} = 0;
a4ac56
 }
a4ac56
 
a4ac56
-plan tests => 118;
a4ac56
+plan tests => 120;
a4ac56
 
a4ac56
 my $Perl = which_perl();
a4ac56
 
a4ac56
@@ -653,6 +653,16 @@ SKIP:
a4ac56
       'stat on an array of valid paths should return ENOENT';
a4ac56
 }
a4ac56
 
a4ac56
+# [perl #131895] stat() doesn't fail on filenames containing \0 / NUL
a4ac56
+ok !stat("TEST\0-"), 'stat on filename with \0';
a4ac56
+SKIP: {
a4ac56
+    my $link = "TEST.symlink.$$";
a4ac56
+    my $can_symlink = eval { symlink "TEST", $link };
a4ac56
+    skip "cannot symlink", 1 unless $can_symlink;
a4ac56
+    ok !lstat("$link\0-"), 'lstat on filename with \0';
a4ac56
+    unlink $link;
a4ac56
+}
a4ac56
+
a4ac56
 END {
a4ac56
     chmod 0666, $tmpfile;
a4ac56
     unlink_all $tmpfile;
a4ac56
-- 
a4ac56
2.13.6
a4ac56