This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
ae6aac60799e779f3e7121c35be218035e065a24
[perl5.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 is not
33 # allowed to write large files or that the file system you are running
34 # the tests on doesn't support large files, or both.  You may also need
35 # to reconfigure your kernel. (This is all very system-dependent.)
36 #
37 # Perl may still be able to support large files, once you have
38 # such a process and such a (file) system.
39 #
40 EOM
41 }
42
43 # Known have-nots.
44 if ($^O eq 'win32' || $^O eq 'vms') {
45     print "1..0\n# no sparse files\n";
46     bye();
47 }
48
49 # Then try to deduce whether we have sparse files.
50
51 # Let's not depend on Fcntl or any other extension.
52
53 my ($SEEK_SET, $SEEK_CUR, $SEEK_END) = (0, 1, 2);
54
55 # We'll start off by creating a one megabyte file which has
56 # only three "true" bytes.  If we have sparseness, we should
57 # consume less blocks than one megabyte (assuming nobody has
58 # one megabyte blocks...)
59
60 open(BIG, ">big") or do { warn "open failed: $!\n"; bye };
61 binmode BIG;
62 seek(BIG, 1_000_000, $SEEK_SET);
63 print BIG "big";
64 close(BIG);
65
66 my @s;
67
68 @s = stat("big");
69
70 print "# @s\n";
71
72 my $BLOCKSIZE = 512; # is this really correct everywhere?
73
74 unless (@s == 13 &&
75         $s[7] == 1_000_003 &&
76         defined $s[12] &&
77         $BLOCKSIZE * $s[12] < 1_000_003) {
78     print "1..0\n# no sparse files?\n";
79     bye();
80 }
81
82 # By now we better be sure that we do have sparse files:
83 # if we are not, the following will hog 5 gigabytes of disk.  Ooops.
84
85 open(BIG, ">big") or do { warn "open failed: $!\n"; bye };
86 binmode BIG;
87 seek(BIG, 5_000_000_000, $SEEK_SET);
88 # Either the print or (more likely, thanks to buffering) the close will
89 # fail if there are are filesize limitations (process or fs).
90 my $print = print BIG "big";
91 my $close = close BIG if $print;
92 unless ($print && $close) {
93     $ENV{LC_ALL} = "C";
94     if ($! =~/File too large/) {
95         print "1..0\n# writing past 2GB failed\n";
96         explain();
97     }
98     bye();
99 }
100
101 @s = stat("big");
102
103 print "# @s\n";
104
105 sub fail () {
106     print "not ";
107     $fail++;
108 }
109
110 print "1..17\n";
111
112 my $fail = 0;
113
114 fail unless $s[7] == 5_000_000_003;     # exercizes pp_stat
115 print "ok 1\n";
116
117 fail unless -s "big" == 5_000_000_003;  # exercizes pp_ftsize
118 print "ok 2\n";
119
120 fail unless -e "big";
121 print "ok 3\n";
122
123 fail unless -f "big";
124 print "ok 4\n";
125
126 open(BIG, "big") or do { warn "open failed: $!\n"; bye };
127 binmode BIG;
128
129 fail unless seek(BIG, 4_500_000_000, $SEEK_SET);
130 print "ok 5\n";
131
132 fail unless tell(BIG) == 4_500_000_000;
133 print "ok 6\n";
134
135 fail unless seek(BIG, 1, $SEEK_CUR);
136 print "ok 7\n";
137
138 fail unless tell(BIG) == 4_500_000_001;
139 print "ok 8\n";
140
141 fail unless seek(BIG, -1, $SEEK_CUR);
142 print "ok 9\n";
143
144 fail unless tell(BIG) == 4_500_000_000;
145 print "ok 10\n";
146
147 fail unless seek(BIG, -3, $SEEK_END);
148 print "ok 11\n";
149
150 fail unless tell(BIG) == 5_000_000_000;
151 print "ok 12\n";
152
153 my $big;
154
155 fail unless read(BIG, $big, 3) == 3;
156 print "ok 13\n";
157
158 fail unless $big eq "big";
159 print "ok 14\n";
160
161 # 705_032_704 = (I32)5_000_000_000
162 fail unless seek(BIG, 705_032_704, $SEEK_SET);
163 print "ok 15\n";
164
165 my $zero;
166
167 fail unless read(BIG, $zero, 3) == 3;
168 print "ok 16\n";
169
170 fail unless $zero eq "\0\0\0";
171 print "ok 17\n";
172
173 explain if $fail;
174
175 bye(); # does the necessary cleanup
176
177 # eof