/* * Copyright (c) 2013 Fujitsu Ltd. * Author: DAN LI <li.dan@cn.fujitsu.com> * * This program is free software; you can redistribute it and/or modify it * under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * This program is distributed in the hope that it would be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * You should have received a copy of the GNU General Public License along * with this program; if not, write the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * */ /* * DESCRIPTION * Test for feature MS_MOVE of mount(2). * "Move an existing mount point to the new location." */ #include <errno.h> #include <sys/mount.h> #include <unistd.h> #include <fcntl.h> #include "test.h" #include "safe_macros.h" #ifndef MS_MOVE #define MS_MOVE 8192 #endif #ifndef MS_PRIVATE #define MS_PRIVATE (1 << 18) #endif #define MNTPOINT_SRC "mnt_src" #define MNTPOINT_DES "mnt_des" #define LINELENGTH 256 #define DIR_MODE (S_IRWXU | S_IRUSR | S_IXUSR | S_IRGRP | S_IXGRP) static int ismount(char *mntpoint); static void setup(void); static void cleanup(void); char *TCID = "mount06"; int TST_TOTAL = 1; static const char *fs_type; static const char *device; static char path_name[PATH_MAX]; static char mntpoint_src[PATH_MAX]; static char mntpoint_des[PATH_MAX]; static int mount_flag; int main(int argc, char *argv[]) { int lc; tst_parse_opts(argc, argv, NULL, NULL); setup(); for (lc = 0; TEST_LOOPING(lc); lc++) { tst_count = 0; SAFE_MOUNT(cleanup, device, mntpoint_src, fs_type, 0, NULL); TEST(mount(mntpoint_src, mntpoint_des, fs_type, MS_MOVE, NULL)); if (TEST_RETURN != 0) { tst_resm(TFAIL | TTERRNO, "mount(2) failed"); } else { if (!ismount(mntpoint_src) && ismount(mntpoint_des)) tst_resm(TPASS, "move mount is ok"); else tst_resm(TFAIL, "move mount does not work"); TEST(tst_umount(mntpoint_des)); if (TEST_RETURN != 0) tst_brkm(TBROK | TTERRNO, cleanup, "umount(2) failed"); } } cleanup(); tst_exit(); } int ismount(char *mntpoint) { int ret = 0; FILE *file; char line[LINELENGTH]; file = fopen("/proc/mounts", "r"); if (file == NULL) tst_brkm(TFAIL | TERRNO, NULL, "Open /proc/mounts failed"); while (fgets(line, LINELENGTH, file) != NULL) { if (strstr(line, mntpoint) != NULL) { ret = 1; break; } } fclose(file); return ret; } static void setup(void) { tst_require_root(); tst_sig(NOFORK, DEF_HANDLER, cleanup); tst_tmpdir(); fs_type = tst_dev_fs_type(); device = tst_acquire_device(cleanup); if (!device) tst_brkm(TCONF, cleanup, "Failed to obtain block device"); tst_mkfs(cleanup, device, fs_type, NULL, NULL); if (getcwd(path_name, sizeof(path_name)) == NULL) tst_brkm(TBROK, cleanup, "getcwd failed"); /* * Turn current dir into a private mount point being a parent * mount which is required by move mount. */ SAFE_MOUNT(cleanup, path_name, path_name, "none", MS_BIND, NULL); mount_flag = 1; SAFE_MOUNT(cleanup, "none", path_name, "none", MS_PRIVATE, NULL); snprintf(mntpoint_src, PATH_MAX, "%s/%s", path_name, MNTPOINT_SRC); snprintf(mntpoint_des, PATH_MAX, "%s/%s", path_name, MNTPOINT_DES); SAFE_MKDIR(cleanup, mntpoint_src, DIR_MODE); SAFE_MKDIR(cleanup, mntpoint_des, DIR_MODE); TEST_PAUSE; } static void cleanup(void) { if (mount_flag && tst_umount(path_name) != 0) tst_resm(TWARN | TERRNO, "umount(2) %s failed", path_name); if (device) tst_release_device(device); tst_rmdir(); }