xref: /freebsd/contrib/netbsd-tests/fs/common/snapshot.c (revision 9268022b74279434ed6300244e3f977e56a8ceb5)
1*57718be8SEnji Cooper /*	$NetBSD: snapshot.c,v 1.7 2013/02/06 09:05:01 hannken Exp $	*/
2*57718be8SEnji Cooper 
3*57718be8SEnji Cooper #include <sys/types.h>
4*57718be8SEnji Cooper #include <sys/ioctl.h>
5*57718be8SEnji Cooper #include <sys/mount.h>
6*57718be8SEnji Cooper 
7*57718be8SEnji Cooper #include <dev/fssvar.h>
8*57718be8SEnji Cooper 
9*57718be8SEnji Cooper #include <atf-c.h>
10*57718be8SEnji Cooper #include <fcntl.h>
11*57718be8SEnji Cooper #include <pthread.h>
12*57718be8SEnji Cooper #include <stdio.h>
13*57718be8SEnji Cooper #include <stdlib.h>
14*57718be8SEnji Cooper #include <string.h>
15*57718be8SEnji Cooper #include <unistd.h>
16*57718be8SEnji Cooper 
17*57718be8SEnji Cooper ATF_TC_WITH_CLEANUP(snapshot);
ATF_TC_HEAD(snapshot,tc)18*57718be8SEnji Cooper ATF_TC_HEAD(snapshot, tc)
19*57718be8SEnji Cooper {
20*57718be8SEnji Cooper 
21*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "basic snapshot features");
22*57718be8SEnji Cooper }
23*57718be8SEnji Cooper 
24*57718be8SEnji Cooper static void
makefile(const char * path)25*57718be8SEnji Cooper makefile(const char *path)
26*57718be8SEnji Cooper {
27*57718be8SEnji Cooper 	int fd;
28*57718be8SEnji Cooper 
29*57718be8SEnji Cooper 	fd = rump_sys_open(path, O_CREAT | O_RDWR, 0777);
30*57718be8SEnji Cooper 	if (fd == -1)
31*57718be8SEnji Cooper 		atf_tc_fail_errno("create %s", path);
32*57718be8SEnji Cooper 	rump_sys_close(fd);
33*57718be8SEnji Cooper }
34*57718be8SEnji Cooper 
ATF_TC_BODY(snapshot,tc)35*57718be8SEnji Cooper ATF_TC_BODY(snapshot, tc)
36*57718be8SEnji Cooper {
37*57718be8SEnji Cooper 	char buf[1024];
38*57718be8SEnji Cooper 	struct fss_set fss;
39*57718be8SEnji Cooper 	int fssfd;
40*57718be8SEnji Cooper 	int fd, fd2, i;
41*57718be8SEnji Cooper 
42*57718be8SEnji Cooper 	if (system(NEWFS) == -1)
43*57718be8SEnji Cooper 		atf_tc_fail_errno("cannot create file system");
44*57718be8SEnji Cooper 
45*57718be8SEnji Cooper 	rump_init();
46*57718be8SEnji Cooper 	begin();
47*57718be8SEnji Cooper 
48*57718be8SEnji Cooper 	if (rump_sys_mkdir("/mnt", 0777) == -1)
49*57718be8SEnji Cooper 		atf_tc_fail_errno("mount point create");
50*57718be8SEnji Cooper 	if (rump_sys_mkdir("/snap", 0777) == -1)
51*57718be8SEnji Cooper 		atf_tc_fail_errno("mount point 2 create");
52*57718be8SEnji Cooper 
53*57718be8SEnji Cooper 	rump_pub_etfs_register("/diskdev", IMGNAME, RUMP_ETFS_BLK);
54*57718be8SEnji Cooper 
55*57718be8SEnji Cooper 	mount_diskfs("/diskdev", "/mnt");
56*57718be8SEnji Cooper 
57*57718be8SEnji Cooper #define TESTSTR1 "huihai\n"
58*57718be8SEnji Cooper #define TESTSZ1 (sizeof(TESTSTR1)-1)
59*57718be8SEnji Cooper #define TESTSTR2 "baana liten\n"
60*57718be8SEnji Cooper #define TESTSZ2 (sizeof(TESTSTR2)-1)
61*57718be8SEnji Cooper 
62*57718be8SEnji Cooper 	fd = rump_sys_open("/mnt/myfile", O_RDWR | O_CREAT, 0777);
63*57718be8SEnji Cooper 	if (fd == -1)
64*57718be8SEnji Cooper 		atf_tc_fail_errno("create file");
65*57718be8SEnji Cooper 	if (rump_sys_write(fd, TESTSTR1, TESTSZ1) != TESTSZ1)
66*57718be8SEnji Cooper 		atf_tc_fail_errno("write fail");
67*57718be8SEnji Cooper 
68*57718be8SEnji Cooper 	fssfd = rump_sys_open("/dev/rfss0", O_RDWR);
69*57718be8SEnji Cooper 	if (fssfd == -1)
70*57718be8SEnji Cooper 		atf_tc_fail_errno("cannot open fss");
71*57718be8SEnji Cooper 	makefile(BAKNAME);
72*57718be8SEnji Cooper 	memset(&fss, 0, sizeof(fss));
73*57718be8SEnji Cooper 	fss.fss_mount = __UNCONST("/mnt");
74*57718be8SEnji Cooper 	fss.fss_bstore = __UNCONST(BAKNAME);
75*57718be8SEnji Cooper 	fss.fss_csize = 0;
76*57718be8SEnji Cooper 	if (rump_sys_ioctl(fssfd, FSSIOCSET, &fss) == -1)
77*57718be8SEnji Cooper 		atf_tc_fail_errno("create snapshot");
78*57718be8SEnji Cooper 
79*57718be8SEnji Cooper 	for (i = 0; i < 10000; i++) {
80*57718be8SEnji Cooper 		if (rump_sys_write(fd, TESTSTR2, TESTSZ2) != TESTSZ2)
81*57718be8SEnji Cooper 			atf_tc_fail_errno("write fail");
82*57718be8SEnji Cooper 	}
83*57718be8SEnji Cooper 	rump_sys_sync();
84*57718be8SEnji Cooper 
85*57718be8SEnji Cooper 	/* technically we should fsck it first? */
86*57718be8SEnji Cooper 	mount_diskfs("/dev/fss0", "/snap");
87*57718be8SEnji Cooper 
88*57718be8SEnji Cooper 	/* check for old contents */
89*57718be8SEnji Cooper 	fd2 = rump_sys_open("/snap/myfile", O_RDONLY);
90*57718be8SEnji Cooper 	if (fd2 == -1)
91*57718be8SEnji Cooper 		atf_tc_fail_errno("fail");
92*57718be8SEnji Cooper 	memset(buf, 0, sizeof(buf));
93*57718be8SEnji Cooper 	if (rump_sys_read(fd2, buf, sizeof(buf)) == -1)
94*57718be8SEnji Cooper 		atf_tc_fail_errno("read snap");
95*57718be8SEnji Cooper 	ATF_CHECK(strcmp(buf, TESTSTR1) == 0);
96*57718be8SEnji Cooper 
97*57718be8SEnji Cooper 	/* check that new files are invisible in the snapshot */
98*57718be8SEnji Cooper 	makefile("/mnt/newfile");
99*57718be8SEnji Cooper 	if (rump_sys_open("/snap/newfile", O_RDONLY) != -1)
100*57718be8SEnji Cooper 		atf_tc_fail("newfile exists in snapshot");
101*57718be8SEnji Cooper 	if (errno != ENOENT)
102*57718be8SEnji Cooper 		atf_tc_fail_errno("newfile open should fail with ENOENT");
103*57718be8SEnji Cooper 
104*57718be8SEnji Cooper 	/* check that removed files are still visible in the snapshot */
105*57718be8SEnji Cooper 	rump_sys_unlink("/mnt/myfile");
106*57718be8SEnji Cooper 	if (rump_sys_open("/snap/myfile", O_RDONLY) == -1)
107*57718be8SEnji Cooper 		atf_tc_fail_errno("unlinked file no longer in snapshot");
108*57718be8SEnji Cooper 
109*57718be8SEnji Cooper 	/* done for now */
110*57718be8SEnji Cooper }
111*57718be8SEnji Cooper 
ATF_TC_CLEANUP(snapshot,tc)112*57718be8SEnji Cooper ATF_TC_CLEANUP(snapshot, tc)
113*57718be8SEnji Cooper {
114*57718be8SEnji Cooper 
115*57718be8SEnji Cooper 	unlink(IMGNAME);
116*57718be8SEnji Cooper }
117*57718be8SEnji Cooper 
118*57718be8SEnji Cooper ATF_TC_WITH_CLEANUP(snapshotstress);
ATF_TC_HEAD(snapshotstress,tc)119*57718be8SEnji Cooper ATF_TC_HEAD(snapshotstress, tc)
120*57718be8SEnji Cooper {
121*57718be8SEnji Cooper 
122*57718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "snapshot on active file system");
123*57718be8SEnji Cooper }
124*57718be8SEnji Cooper 
125*57718be8SEnji Cooper #define NACTIVITY 4
126*57718be8SEnji Cooper 
127*57718be8SEnji Cooper static bool activity_stop = false;
128*57718be8SEnji Cooper static pid_t wrkpid;
129*57718be8SEnji Cooper 
130*57718be8SEnji Cooper static void *
fs_activity(void * arg)131*57718be8SEnji Cooper fs_activity(void *arg)
132*57718be8SEnji Cooper {
133*57718be8SEnji Cooper 	int di, fi;
134*57718be8SEnji Cooper 	char *prefix = arg, path[128];
135*57718be8SEnji Cooper 
136*57718be8SEnji Cooper 	rump_pub_lwproc_newlwp(wrkpid);
137*57718be8SEnji Cooper 
138*57718be8SEnji Cooper 	RL(rump_sys_mkdir(prefix, 0777));
139*57718be8SEnji Cooper 	while (! activity_stop) {
140*57718be8SEnji Cooper 		for (di = 0; di < 5; di++) {
141*57718be8SEnji Cooper 			snprintf(path, sizeof(path), "%s/d%d", prefix, di);
142*57718be8SEnji Cooper 			RL(rump_sys_mkdir(path, 0777));
143*57718be8SEnji Cooper 			for (fi = 0; fi < 5; fi++) {
144*57718be8SEnji Cooper 				snprintf(path, sizeof(path), "%s/d%d/f%d",
145*57718be8SEnji Cooper 				    prefix, di, fi);
146*57718be8SEnji Cooper 				makefile(path);
147*57718be8SEnji Cooper 			}
148*57718be8SEnji Cooper 		}
149*57718be8SEnji Cooper 		for (di = 0; di < 5; di++) {
150*57718be8SEnji Cooper 			for (fi = 0; fi < 5; fi++) {
151*57718be8SEnji Cooper 				snprintf(path, sizeof(path), "%s/d%d/f%d",
152*57718be8SEnji Cooper 				    prefix, di, fi);
153*57718be8SEnji Cooper 				RL(rump_sys_unlink(path));
154*57718be8SEnji Cooper 			}
155*57718be8SEnji Cooper 			snprintf(path, sizeof(path), "%s/d%d", prefix, di);
156*57718be8SEnji Cooper 			RL(rump_sys_rmdir(path));
157*57718be8SEnji Cooper 		}
158*57718be8SEnji Cooper 	}
159*57718be8SEnji Cooper 	RL(rump_sys_rmdir(prefix));
160*57718be8SEnji Cooper 
161*57718be8SEnji Cooper 	rump_pub_lwproc_releaselwp();
162*57718be8SEnji Cooper 
163*57718be8SEnji Cooper 	return NULL;
164*57718be8SEnji Cooper }
165*57718be8SEnji Cooper 
ATF_TC_BODY(snapshotstress,tc)166*57718be8SEnji Cooper ATF_TC_BODY(snapshotstress, tc)
167*57718be8SEnji Cooper {
168*57718be8SEnji Cooper 	pthread_t at[NACTIVITY];
169*57718be8SEnji Cooper 	struct fss_set fss;
170*57718be8SEnji Cooper 	char prefix[NACTIVITY][128];
171*57718be8SEnji Cooper 	int i, fssfd;
172*57718be8SEnji Cooper 
173*57718be8SEnji Cooper 	if (system(NEWFS) == -1)
174*57718be8SEnji Cooper 		atf_tc_fail_errno("cannot create file system");
175*57718be8SEnji Cooper 	/* Force SMP so the stress makes sense. */
176*57718be8SEnji Cooper 	RL(setenv("RUMP_NCPU", "4", 1));
177*57718be8SEnji Cooper 	RZ(rump_init());
178*57718be8SEnji Cooper 	/* Prepare for fsck to use the RUMP /dev/fss0. */
179*57718be8SEnji Cooper 	RL(rump_init_server("unix://commsock"));
180*57718be8SEnji Cooper 	RL(setenv("LD_PRELOAD", "/usr/lib/librumphijack.so", 1));
181*57718be8SEnji Cooper 	RL(setenv("RUMP_SERVER", "unix://commsock", 1));
182*57718be8SEnji Cooper 	RL(setenv("RUMPHIJACK", "blanket=/dev/rfss0", 1));
183*57718be8SEnji Cooper 	begin();
184*57718be8SEnji Cooper 
185*57718be8SEnji Cooper 	RL(rump_sys_mkdir("/mnt", 0777));
186*57718be8SEnji Cooper 
187*57718be8SEnji Cooper 	rump_pub_etfs_register("/diskdev", IMGNAME, RUMP_ETFS_BLK);
188*57718be8SEnji Cooper 
189*57718be8SEnji Cooper 	mount_diskfs("/diskdev", "/mnt");
190*57718be8SEnji Cooper 
191*57718be8SEnji Cooper 	/* Start file system activity. */
192*57718be8SEnji Cooper 	RL(wrkpid = rump_sys_getpid());
193*57718be8SEnji Cooper 	for (i = 0; i < NACTIVITY; i++) {
194*57718be8SEnji Cooper 		snprintf(prefix[i], sizeof(prefix[i]),  "/mnt/a%d", i);
195*57718be8SEnji Cooper 		RL(pthread_create(&at[i], NULL, fs_activity, prefix[i]));
196*57718be8SEnji Cooper 		sleep(1);
197*57718be8SEnji Cooper 	}
198*57718be8SEnji Cooper 
199*57718be8SEnji Cooper 	fssfd = rump_sys_open("/dev/rfss0", O_RDWR);
200*57718be8SEnji Cooper 	if (fssfd == -1)
201*57718be8SEnji Cooper 		atf_tc_fail_errno("cannot open fss");
202*57718be8SEnji Cooper 	makefile(BAKNAME);
203*57718be8SEnji Cooper 	memset(&fss, 0, sizeof(fss));
204*57718be8SEnji Cooper 	fss.fss_mount = __UNCONST("/mnt");
205*57718be8SEnji Cooper 	fss.fss_bstore = __UNCONST(BAKNAME);
206*57718be8SEnji Cooper 	fss.fss_csize = 0;
207*57718be8SEnji Cooper 	if (rump_sys_ioctl(fssfd, FSSIOCSET, &fss) == -1)
208*57718be8SEnji Cooper 		atf_tc_fail_errno("create snapshot");
209*57718be8SEnji Cooper 
210*57718be8SEnji Cooper 	activity_stop = true;
211*57718be8SEnji Cooper 	for (i = 0; i < NACTIVITY; i++)
212*57718be8SEnji Cooper 		RL(pthread_join(at[i], NULL));
213*57718be8SEnji Cooper 
214*57718be8SEnji Cooper 	RL(system(FSCK " /dev/rfss0"));
215*57718be8SEnji Cooper }
216*57718be8SEnji Cooper 
ATF_TC_CLEANUP(snapshotstress,tc)217*57718be8SEnji Cooper ATF_TC_CLEANUP(snapshotstress, tc)
218*57718be8SEnji Cooper {
219*57718be8SEnji Cooper 
220*57718be8SEnji Cooper 	unlink(IMGNAME);
221*57718be8SEnji Cooper }
222*57718be8SEnji Cooper 
ATF_TP_ADD_TCS(tp)223*57718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
224*57718be8SEnji Cooper {
225*57718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, snapshot);
226*57718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, snapshotstress);
227*57718be8SEnji Cooper 	return 0;
228*57718be8SEnji Cooper }
229