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); 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 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 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 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); 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 * 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 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 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 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