87060e74c6c8914c1dca6ff512530a2a58411b89
[p5sagit/p5-mst-13.2.git] / t / op / lfs.t
1 # NOTE: this file tests how large files (>2GB) work with perlio (stdio/sfio).
2 # sysopen(), sysseek(), syswrite(), sysread() are tested in t/lib/syslfs.t.
3 # If you modify/add tests here, remember to update also t/lib/syslfs.t.
4
5 BEGIN {
6         # Don't bother if there are no quads.
7         eval { my $q = pack "q", 0 };
8         if ($@) {
9                 print "1..0\n# no 64-bit types\n";
10                 exit(0);
11         }
12         chdir 't' if -d 't';
13         unshift @INC, '../lib';
14         # Don't bother if there are no quad offsets.
15         require Config; import Config;
16         if ($Config{lseeksize} < 8) {
17                 print "1..0\n# no 64-bit file offsets\n";
18                 exit(0);
19         }
20 }
21
22 sub bye {
23     close(BIG);
24     unlink "big";
25     exit(0);
26 }
27
28 sub explain {
29     print <<EOM;
30 #
31 # If the lfs (large file support: large meaning larger than two gigabytes)
32 # tests are skipped or fail, it may mean either that your process
33 # (or process group) is not allowed to write large files (resource
34 # limits) or that the file system you are running the tests on doesn't
35 # let your user/group have large files (quota) or the filesystem simply
36 # doesn't support large files.  You may even need to reconfigure your kernel.
37 # (This is all very operating system and site-dependent.)
38 #
39 # Perl may still be able to support large files, once you have
40 # such a process, enough quota, and such a (file) system.
41 #
42 EOM
43 }
44
45 # Known have-nots.
46 if ($^O eq 'win32' || $^O eq 'vms') {
47     print "1..0\n# no sparse files\n";
48     bye();
49 }
50
51 # Known haves that have problems running this test
52 # (for example because they do not support sparse files, like UNICOS)
53 if ($^O eq 'unicos') {
54     print "1..0\n# large files known to work but unable to test them here\n";
55     bye();
56 }
57
58 # Then try to deduce whether we have sparse files.
59
60 # Let's not depend on Fcntl or any other extension.
61
62 my ($SEEK_SET, $SEEK_CUR, $SEEK_END) = (0, 1, 2);
63
64 # We'll start off by creating a one megabyte file which has
65 # only three "true" bytes.  If we have sparseness, we should
66 # consume less blocks than one megabyte (assuming nobody has
67 # one megabyte blocks...)
68
69 open(BIG, ">big") or do { warn "open failed: $!\n"; bye };
70 binmode BIG;
71 seek(BIG, 1_000_000, $SEEK_SET);
72 print BIG "big";
73 close(BIG);
74
75 my @s;
76
77 @s = stat("big");
78
79 print "# @s\n";
80
81 my $BLOCKSIZE = $s[11] || 512;
82
83 unless (@s == 13 &&
84         $s[7] == 1_000_003 &&
85         defined $s[12] &&
86         $BLOCKSIZE * $s[12] < 1_000_003) {
87     print "1..0\n# no sparse files?\n";
88     bye();
89 }
90
91 # By now we better be sure that we do have sparse files:
92 # if we are not, the following will hog 5 gigabytes of disk.  Ooops.
93
94 $ENV{LC_ALL} = "C";
95
96 open(BIG, ">big") or do { warn "open failed: $!\n"; bye };
97 binmode BIG;
98 seek(BIG, 5_000_000_000, $SEEK_SET);
99
100 # Either the print or (more likely, thanks to buffering) the close will
101 # fail if there are are filesize limitations (process or fs).
102 my $print = print BIG "big";
103 my $close = close BIG if $print;
104 unless ($print && $close) {
105     unless ($print) {
106         print "# print failed: $!\n"
107     } else {
108         print "# close failed: $!\n"
109     }
110     if ($! =~/too large/i) {
111         print "1..0\n# writing past 2GB failed: process limits?\n";
112     } elsif ($! =~ /quota/i) {
113         print "1..0\n# filesystem quota limits?\n";
114     }
115     explain();
116     bye();
117 }
118
119 @s = stat("big");
120
121 print "# @s\n";
122
123 unless ($s[7] == 5_000_000_003) {
124     print "1..0\n# not configured to use large files?\n";
125     explain();
126     bye();
127 }
128
129 sub fail () {
130     print "not ";
131     $fail++;
132 }
133
134 print "1..17\n";
135
136 my $fail = 0;
137
138 fail unless $s[7] == 5_000_000_003;     # exercizes pp_stat
139 print "ok 1\n";
140
141 fail unless -s "big" == 5_000_000_003;  # exercizes pp_ftsize
142 print "ok 2\n";
143
144 fail unless -e "big";
145 print "ok 3\n";
146
147 fail unless -f "big";
148 print "ok 4\n";
149
150 open(BIG, "big") or do { warn "open failed: $!\n"; bye };
151 binmode BIG;
152
153 fail unless seek(BIG, 4_500_000_000, $SEEK_SET);
154 print "ok 5\n";
155
156 fail unless tell(BIG) == 4_500_000_000;
157 print "ok 6\n";
158
159 fail unless seek(BIG, 1, $SEEK_CUR);
160 print "ok 7\n";
161
162 fail unless tell(BIG) == 4_500_000_001;
163 print "ok 8\n";
164
165 fail unless seek(BIG, -1, $SEEK_CUR);
166 print "ok 9\n";
167
168 fail unless tell(BIG) == 4_500_000_000;
169 print "ok 10\n";
170
171 fail unless seek(BIG, -3, $SEEK_END);
172 print "ok 11\n";
173
174 fail unless tell(BIG) == 5_000_000_000;
175 print "ok 12\n";
176
177 my $big;
178
179 fail unless read(BIG, $big, 3) == 3;
180 print "ok 13\n";
181
182 fail unless $big eq "big";
183 print "ok 14\n";
184
185 # 705_032_704 = (I32)5_000_000_000
186 fail unless seek(BIG, 705_032_704, $SEEK_SET);
187 print "ok 15\n";
188
189 my $zero;
190
191 fail unless read(BIG, $zero, 3) == 3;
192 print "ok 16\n";
193
194 fail unless $zero eq "\0\0\0";
195 print "ok 17\n";
196
197 explain if $fail;
198
199 bye(); # does the necessary cleanup
200
201 END {
202    unlink "big"; # be paranoid about leaving 5 gig files lying around
203 }
204
205 # eof