tests: skip when a debian libc6-2.7-11 bug makes printf segfault
[platform/upstream/coreutils.git] / tests / misc / pwd-long
1 #!/bin/sh
2 # -*- perl -*-
3 # Ensure that pwd works even when run from a very deep directory.
4
5 # Copyright (C) 2006-2008 Free Software Foundation, Inc.
6
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License as published by
9 # the Free Software Foundation, either version 3 of the License, or
10 # (at your option) any later version.
11
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 # GNU General Public License for more details.
16
17 # You should have received a copy of the GNU General Public License
18 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
19
20 : ${srcdir=.}
21 . $top_srcdir/tests/require-perl
22
23 . $top_srcdir/tests/test-lib.sh
24 require_readable_root_
25
26 ARGV_0=$0
27 export ARGV_0
28
29 # Don't use CuTmpdir here, since File::Temp's use of rmtree can't
30 # remove the deep tree we create.
31 $PERL -Tw -- - <<\EOF
32
33 # Show that pwd works even when the length of the resulting
34 # directory name is longer than PATH_MAX.
35 use strict;
36
37 (my $ME = $ENV{ARGV_0}) =~ s|.*/||;
38
39 sub normalize_to_cwd_relative ($$$)
40 {
41   my ($dir, $dev, $ino) = @_;
42   my $slash = -1;
43   my $next_slash;
44   while (1)
45     {
46       $slash = index $dir, '/', $slash + 1;
47       $slash <= -1
48         and die "$ME: $dir does not contain old CWD\n";
49       my $dir_prefix = $slash ? substr ($dir, 0, $slash) : '/';
50       my ($d, $i) = (stat $dir_prefix)[0, 1];
51       $d == $dev && $i == $ino
52         and return substr $dir, $slash + 1;
53     }
54 }
55
56 # Set up a safe, well-known environment
57 delete @ENV{qw(BASH_ENV CDPATH ENV PATH)};
58 $ENV{IFS}  = '';
59
60 # Save CWD's device and inode numbers.
61 my ($dev, $ino) = (stat '.')[0, 1];
62
63 # Construct the expected "."-relative part of pwd's output.
64 my $z = 'z' x 31;
65 my $n = 256;
66 my $expected = "/$z" x $n;
67 # Remove the leading "/".
68 substr ($expected, 0, 1) = '';
69
70 my $i = 0;
71 do
72   {
73     mkdir $z, 0700
74       or die "$ME: at depth $i: $!\n";
75     chdir $z;
76   }
77 until (++$i == $n);
78
79 my $abs_top_builddir = $ENV{abs_top_builddir};
80 $abs_top_builddir
81   or die "$ME: envvar abs_top_builddir not defined\n";
82 my $build_src_dir = "$abs_top_builddir/src";
83 if ($build_src_dir !~ m!^([-+.:/\w]+)$!)
84   {
85     warn "$0: skipping this test; odd build source directory name:\n"
86       . "$build_src_dir\n";
87     exit 77;
88   }
89 $build_src_dir = $1;
90
91 my $pwd_binary = "$build_src_dir/pwd";
92
93 -x $pwd_binary
94   or die "$ME: $pwd_binary is not an executable file\n";
95 chomp (my $actual = `$pwd_binary`);
96
97 # Convert the absolute name from pwd into a $CWD-relative name.
98 # This is necessary in order to avoid a spurious failure when run
99 # from a directory in a bind-mounted partition.  What happens is
100 # pwd reads a ".." that contains two or more entries with identical
101 # dev,ino that match the ones we're looking for, and it chooses a
102 # name that does not correspond to the one already recorded in $CWD.
103 $actual = normalize_to_cwd_relative $actual, $dev, $ino;
104
105 if ($expected ne $actual)
106   {
107     my $e_len = length $expected;
108     my $a_len = length $actual;
109     warn "expected len: $e_len\n";
110     warn "actual len:   $a_len\n";
111     warn "expected: $expected\n";
112     warn "actual: $actual\n";
113     exit 1;
114   }
115 EOF
116
117 fail=$?
118
119 (exit $fail); exit $fail