1*543adbedSBen Gras /*-
2*543adbedSBen Gras * Copyright (c) 2003-2007 Tim Kientzle
3*543adbedSBen Gras * All rights reserved.
4*543adbedSBen Gras *
5*543adbedSBen Gras * Redistribution and use in source and binary forms, with or without
6*543adbedSBen Gras * modification, are permitted provided that the following conditions
7*543adbedSBen Gras * are met:
8*543adbedSBen Gras * 1. Redistributions of source code must retain the above copyright
9*543adbedSBen Gras * notice, this list of conditions and the following disclaimer.
10*543adbedSBen Gras * 2. Redistributions in binary form must reproduce the above copyright
11*543adbedSBen Gras * notice, this list of conditions and the following disclaimer in the
12*543adbedSBen Gras * documentation and/or other materials provided with the distribution.
13*543adbedSBen Gras *
14*543adbedSBen Gras * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
15*543adbedSBen Gras * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
16*543adbedSBen Gras * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
17*543adbedSBen Gras * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
18*543adbedSBen Gras * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
19*543adbedSBen Gras * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
20*543adbedSBen Gras * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
21*543adbedSBen Gras * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22*543adbedSBen Gras * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
23*543adbedSBen Gras * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24*543adbedSBen Gras */
25*543adbedSBen Gras #include "test.h"
26*543adbedSBen Gras __FBSDID("$FreeBSD$");
27*543adbedSBen Gras
28*543adbedSBen Gras static int
is_octal(const char * p,size_t l)29*543adbedSBen Gras is_octal(const char *p, size_t l)
30*543adbedSBen Gras {
31*543adbedSBen Gras while (l > 0) {
32*543adbedSBen Gras if (*p < '0' || *p > '7')
33*543adbedSBen Gras return (0);
34*543adbedSBen Gras --l;
35*543adbedSBen Gras ++p;
36*543adbedSBen Gras }
37*543adbedSBen Gras return (1);
38*543adbedSBen Gras }
39*543adbedSBen Gras
40*543adbedSBen Gras static int
from_octal(const char * p,size_t l)41*543adbedSBen Gras from_octal(const char *p, size_t l)
42*543adbedSBen Gras {
43*543adbedSBen Gras int r = 0;
44*543adbedSBen Gras
45*543adbedSBen Gras while (l > 0) {
46*543adbedSBen Gras r *= 8;
47*543adbedSBen Gras r += *p - '0';
48*543adbedSBen Gras --l;
49*543adbedSBen Gras ++p;
50*543adbedSBen Gras }
51*543adbedSBen Gras return (r);
52*543adbedSBen Gras }
53*543adbedSBen Gras
DEFINE_TEST(test_option_c)54*543adbedSBen Gras DEFINE_TEST(test_option_c)
55*543adbedSBen Gras {
56*543adbedSBen Gras FILE *filelist;
57*543adbedSBen Gras int r;
58*543adbedSBen Gras int uid = -1;
59*543adbedSBen Gras int dev, ino, gid;
60*543adbedSBen Gras time_t t, now;
61*543adbedSBen Gras char *p, *e;
62*543adbedSBen Gras size_t s;
63*543adbedSBen Gras
64*543adbedSBen Gras assertUmask(0);
65*543adbedSBen Gras
66*543adbedSBen Gras #if !defined(_WIN32)
67*543adbedSBen Gras uid = getuid();
68*543adbedSBen Gras #endif
69*543adbedSBen Gras
70*543adbedSBen Gras /*
71*543adbedSBen Gras * Create an assortment of files.
72*543adbedSBen Gras * TODO: Extend this to cover more filetypes.
73*543adbedSBen Gras */
74*543adbedSBen Gras filelist = fopen("filelist", "w");
75*543adbedSBen Gras
76*543adbedSBen Gras /* "file" */
77*543adbedSBen Gras assertMakeFile("file", 0644, "1234567890");
78*543adbedSBen Gras fprintf(filelist, "file\n");
79*543adbedSBen Gras
80*543adbedSBen Gras /* "symlink" */
81*543adbedSBen Gras if (canSymlink()) {
82*543adbedSBen Gras assertMakeSymlink("symlink", "file");
83*543adbedSBen Gras fprintf(filelist, "symlink\n");
84*543adbedSBen Gras }
85*543adbedSBen Gras
86*543adbedSBen Gras /* "dir" */
87*543adbedSBen Gras assertMakeDir("dir", 0775);
88*543adbedSBen Gras /* Record some facts about what we just created: */
89*543adbedSBen Gras now = time(NULL); /* They were all created w/in last two seconds. */
90*543adbedSBen Gras fprintf(filelist, "dir\n");
91*543adbedSBen Gras
92*543adbedSBen Gras /* Use the cpio program to create an archive. */
93*543adbedSBen Gras fclose(filelist);
94*543adbedSBen Gras r = systemf("%s -oc <filelist >basic.out 2>basic.err", testprog);
95*543adbedSBen Gras /* Verify that nothing went to stderr. */
96*543adbedSBen Gras assertTextFileContents("1 block\n", "basic.err");
97*543adbedSBen Gras
98*543adbedSBen Gras /* Assert that the program finished. */
99*543adbedSBen Gras failure("%s -oc crashed", testprog);
100*543adbedSBen Gras if (!assertEqualInt(r, 0))
101*543adbedSBen Gras return;
102*543adbedSBen Gras
103*543adbedSBen Gras /* Verify that stdout is a well-formed cpio file in "odc" format. */
104*543adbedSBen Gras p = slurpfile(&s, "basic.out");
105*543adbedSBen Gras assertEqualInt(s, 512);
106*543adbedSBen Gras e = p;
107*543adbedSBen Gras
108*543adbedSBen Gras /*
109*543adbedSBen Gras * Some of these assertions could be stronger, but it's
110*543adbedSBen Gras * a little tricky because they depend on the local environment.
111*543adbedSBen Gras */
112*543adbedSBen Gras
113*543adbedSBen Gras /* First entry is "file" */
114*543adbedSBen Gras assert(is_octal(e, 76)); /* Entire header is octal digits. */
115*543adbedSBen Gras assertEqualMem(e + 0, "070707", 6); /* Magic */
116*543adbedSBen Gras assert(is_octal(e + 6, 6)); /* dev */
117*543adbedSBen Gras dev = from_octal(e + 6, 6);
118*543adbedSBen Gras assert(is_octal(e + 12, 6)); /* ino */
119*543adbedSBen Gras ino = from_octal(e + 12, 6);
120*543adbedSBen Gras #if defined(_WIN32) && !defined(__CYGWIN__)
121*543adbedSBen Gras /* Group members bits and others bits do not work. */
122*543adbedSBen Gras assertEqualMem(e + 18, "100666", 6); /* Mode */
123*543adbedSBen Gras #else
124*543adbedSBen Gras assertEqualMem(e + 18, "100644", 6); /* Mode */
125*543adbedSBen Gras #endif
126*543adbedSBen Gras if (uid < 0)
127*543adbedSBen Gras uid = from_octal(e + 24, 6);
128*543adbedSBen Gras assertEqualInt(from_octal(e + 24, 6), uid); /* uid */
129*543adbedSBen Gras assert(is_octal(e + 30, 6)); /* gid */
130*543adbedSBen Gras gid = from_octal(e + 30, 6);
131*543adbedSBen Gras assertEqualMem(e + 36, "000001", 6); /* nlink */
132*543adbedSBen Gras failure("file entries should not have rdev set (dev field was 0%o)",
133*543adbedSBen Gras dev);
134*543adbedSBen Gras assertEqualMem(e + 42, "000000", 6); /* rdev */
135*543adbedSBen Gras t = from_octal(e + 48, 11); /* mtime */
136*543adbedSBen Gras assert(t <= now); /* File wasn't created in future. */
137*543adbedSBen Gras assert(t >= now - 2); /* File was created w/in last 2 secs. */
138*543adbedSBen Gras assertEqualMem(e + 59, "000005", 6); /* Name size */
139*543adbedSBen Gras assertEqualMem(e + 65, "00000000012", 11); /* File size */
140*543adbedSBen Gras assertEqualMem(e + 76, "file\0", 5); /* Name contents */
141*543adbedSBen Gras assertEqualMem(e + 81, "1234567890", 10); /* File contents */
142*543adbedSBen Gras e += 91;
143*543adbedSBen Gras
144*543adbedSBen Gras /* "symlink" pointing to "file" */
145*543adbedSBen Gras if (canSymlink()) {
146*543adbedSBen Gras assert(is_octal(e, 76)); /* Entire header is octal digits. */
147*543adbedSBen Gras assertEqualMem(e + 0, "070707", 6); /* Magic */
148*543adbedSBen Gras assertEqualInt(dev, from_octal(e + 6, 6)); /* dev */
149*543adbedSBen Gras assert(ino != from_octal(e + 12, 6)); /* ino */
150*543adbedSBen Gras #if !defined(_WIN32) || defined(__CYGWIN__)
151*543adbedSBen Gras /* On Windows, symbolic link and group members bits and
152*543adbedSBen Gras * others bits do not work. */
153*543adbedSBen Gras assertEqualMem(e + 18, "120777", 6); /* Mode */
154*543adbedSBen Gras #endif
155*543adbedSBen Gras assertEqualInt(from_octal(e + 24, 6), uid); /* uid */
156*543adbedSBen Gras assertEqualInt(gid, from_octal(e + 30, 6)); /* gid */
157*543adbedSBen Gras assertEqualMem(e + 36, "000001", 6); /* nlink */
158*543adbedSBen Gras failure("file entries should have rdev == 0 (dev was 0%o)",
159*543adbedSBen Gras from_octal(e + 6, 6));
160*543adbedSBen Gras assertEqualMem(e + 42, "000000", 6); /* rdev */
161*543adbedSBen Gras t = from_octal(e + 48, 11); /* mtime */
162*543adbedSBen Gras assert(t <= now); /* File wasn't created in future. */
163*543adbedSBen Gras assert(t >= now - 2); /* File was created w/in last 2 secs. */
164*543adbedSBen Gras assertEqualMem(e + 59, "000010", 6); /* Name size */
165*543adbedSBen Gras assertEqualMem(e + 65, "00000000004", 11); /* File size */
166*543adbedSBen Gras assertEqualMem(e + 76, "symlink\0", 8); /* Name contents */
167*543adbedSBen Gras assertEqualMem(e + 84, "file", 4); /* Symlink target. */
168*543adbedSBen Gras e += 88;
169*543adbedSBen Gras }
170*543adbedSBen Gras
171*543adbedSBen Gras /* "dir" */
172*543adbedSBen Gras assert(is_octal(e, 76));
173*543adbedSBen Gras assertEqualMem(e + 0, "070707", 6); /* Magic */
174*543adbedSBen Gras /* Dev should be same as first entry. */
175*543adbedSBen Gras assert(is_octal(e + 6, 6)); /* dev */
176*543adbedSBen Gras assertEqualInt(dev, from_octal(e + 6, 6));
177*543adbedSBen Gras /* Ino must be different from first entry. */
178*543adbedSBen Gras assert(is_octal(e + 12, 6)); /* ino */
179*543adbedSBen Gras assert(dev != from_octal(e + 12, 6));
180*543adbedSBen Gras #if defined(_WIN32) && !defined(__CYGWIN__)
181*543adbedSBen Gras /* Group members bits and others bits do not work. */
182*543adbedSBen Gras assertEqualMem(e + 18, "040777", 6); /* Mode */
183*543adbedSBen Gras #else
184*543adbedSBen Gras /* Accept 042775 to accomodate systems where sgid bit propagates. */
185*543adbedSBen Gras if (memcmp(e + 18, "042775", 6) != 0)
186*543adbedSBen Gras assertEqualMem(e + 18, "040775", 6); /* Mode */
187*543adbedSBen Gras #endif
188*543adbedSBen Gras assertEqualInt(from_octal(e + 24, 6), uid); /* uid */
189*543adbedSBen Gras /* Gid should be same as first entry. */
190*543adbedSBen Gras assert(is_octal(e + 30, 6)); /* gid */
191*543adbedSBen Gras assertEqualInt(gid, from_octal(e + 30, 6));
192*543adbedSBen Gras #ifndef NLINKS_INACCURATE_FOR_DIRS
193*543adbedSBen Gras assertEqualMem(e + 36, "000002", 6); /* Nlink */
194*543adbedSBen Gras #endif
195*543adbedSBen Gras t = from_octal(e + 48, 11); /* mtime */
196*543adbedSBen Gras assert(t <= now); /* File wasn't created in future. */
197*543adbedSBen Gras assert(t >= now - 2); /* File was created w/in last 2 secs. */
198*543adbedSBen Gras assertEqualMem(e + 59, "000004", 6); /* Name size */
199*543adbedSBen Gras assertEqualMem(e + 65, "00000000000", 11); /* File size */
200*543adbedSBen Gras assertEqualMem(e + 76, "dir\0", 4); /* name */
201*543adbedSBen Gras e += 80;
202*543adbedSBen Gras
203*543adbedSBen Gras /* TODO: Verify other types of entries. */
204*543adbedSBen Gras
205*543adbedSBen Gras /* Last entry is end-of-archive marker. */
206*543adbedSBen Gras assert(is_octal(e, 76));
207*543adbedSBen Gras assertEqualMem(e + 0, "070707", 6); /* Magic */
208*543adbedSBen Gras assertEqualMem(e + 6, "000000", 6); /* dev */
209*543adbedSBen Gras assertEqualMem(e + 12, "000000", 6); /* ino */
210*543adbedSBen Gras assertEqualMem(e + 18, "000000", 6); /* Mode */
211*543adbedSBen Gras assertEqualMem(e + 24, "000000", 6); /* uid */
212*543adbedSBen Gras assertEqualMem(e + 30, "000000", 6); /* gid */
213*543adbedSBen Gras assertEqualMem(e + 36, "000001", 6); /* Nlink */
214*543adbedSBen Gras assertEqualMem(e + 42, "000000", 6); /* rdev */
215*543adbedSBen Gras assertEqualMem(e + 48, "00000000000", 11); /* mtime */
216*543adbedSBen Gras assertEqualMem(e + 59, "000013", 6); /* Name size */
217*543adbedSBen Gras assertEqualMem(e + 65, "00000000000", 11); /* File size */
218*543adbedSBen Gras assertEqualMem(e + 76, "TRAILER!!!\0", 11); /* Name */
219*543adbedSBen Gras
220*543adbedSBen Gras free(p);
221*543adbedSBen Gras }
222