xref: /minix3/external/bsd/libarchive/dist/cpio/test/test_option_c.c (revision 543adbed3a3a783ed36434adafbc258b6bde442d)
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