1*63d1fd59SEnji Cooper /* $NetBSD: t_modcmd.c,v 1.10 2017/01/13 21:30:43 christos Exp $ */ 257718be8SEnji Cooper 357718be8SEnji Cooper /* 457718be8SEnji Cooper * Copyright (c) 2009 The NetBSD Foundation, Inc. 557718be8SEnji Cooper * All rights reserved. 657718be8SEnji Cooper * 757718be8SEnji Cooper * Redistribution and use in source and binary forms, with or without 857718be8SEnji Cooper * modification, are permitted provided that the following conditions 957718be8SEnji Cooper * are met: 1057718be8SEnji Cooper * 1. Redistributions of source code must retain the above copyright 1157718be8SEnji Cooper * notice, this list of conditions and the following disclaimer. 1257718be8SEnji Cooper * 2. Redistributions in binary form must reproduce the above copyright 1357718be8SEnji Cooper * notice, this list of conditions and the following disclaimer in the 1457718be8SEnji Cooper * documentation and/or other materials provided with the distribution. 1557718be8SEnji Cooper * 1657718be8SEnji Cooper * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND 1757718be8SEnji Cooper * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, 1857718be8SEnji Cooper * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 1957718be8SEnji Cooper * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 2057718be8SEnji Cooper * IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY 2157718be8SEnji Cooper * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 2257718be8SEnji Cooper * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE 2357718be8SEnji Cooper * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 2457718be8SEnji Cooper * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER 2557718be8SEnji Cooper * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 2657718be8SEnji Cooper * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN 2757718be8SEnji Cooper * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 2857718be8SEnji Cooper */ 2957718be8SEnji Cooper 3057718be8SEnji Cooper #include <sys/types.h> 3157718be8SEnji Cooper #include <sys/mount.h> 3257718be8SEnji Cooper #include <sys/sysctl.h> 3357718be8SEnji Cooper 3457718be8SEnji Cooper #include <rump/rump.h> 3557718be8SEnji Cooper #include <rump/rump_syscalls.h> 3657718be8SEnji Cooper 3757718be8SEnji Cooper #include <fs/tmpfs/tmpfs_args.h> 3857718be8SEnji Cooper 3957718be8SEnji Cooper #include <atf-c.h> 4057718be8SEnji Cooper #include <dlfcn.h> 4157718be8SEnji Cooper #include <err.h> 4257718be8SEnji Cooper #include <errno.h> 4357718be8SEnji Cooper #include <stdio.h> 4457718be8SEnji Cooper #include <stdlib.h> 4557718be8SEnji Cooper #include <string.h> 4657718be8SEnji Cooper #include <unistd.h> 4757718be8SEnji Cooper #include <util.h> 4857718be8SEnji Cooper 49*63d1fd59SEnji Cooper #include "h_macros.h" 5057718be8SEnji Cooper /* 5157718be8SEnji Cooper * We verify that modules can be loaded and unloaded. 5257718be8SEnji Cooper * tmpfs was chosen because it does not depend on an image. 5357718be8SEnji Cooper */ 5457718be8SEnji Cooper 5557718be8SEnji Cooper ATF_TC(cmsg_modcmd); 5657718be8SEnji Cooper ATF_TC_HEAD(cmsg_modcmd, tc) 5757718be8SEnji Cooper { 5857718be8SEnji Cooper atf_tc_set_md_var(tc, "descr", "Checks that loading and unloading " 5957718be8SEnji Cooper "a module (vfs/tmpfs) is possible"); 6057718be8SEnji Cooper } 6157718be8SEnji Cooper 6257718be8SEnji Cooper static int 6357718be8SEnji Cooper disable_autoload(void) 6457718be8SEnji Cooper { 6557718be8SEnji Cooper struct sysctlnode q, ans[256]; 6657718be8SEnji Cooper int mib[3]; 6757718be8SEnji Cooper size_t alen; 6857718be8SEnji Cooper unsigned i; 6957718be8SEnji Cooper bool no; 7057718be8SEnji Cooper 7157718be8SEnji Cooper mib[0] = CTL_KERN; 7257718be8SEnji Cooper mib[1] = CTL_QUERY; 7357718be8SEnji Cooper alen = sizeof(ans); 7457718be8SEnji Cooper 7557718be8SEnji Cooper memset(&q, 0, sizeof(q)); 7657718be8SEnji Cooper q.sysctl_flags = SYSCTL_VERSION; 7757718be8SEnji Cooper 7857718be8SEnji Cooper if (rump_sys___sysctl(mib, 2, ans, &alen, &q, sizeof(q)) == -1) 7957718be8SEnji Cooper return -1; 8057718be8SEnji Cooper 8157718be8SEnji Cooper for (i = 0; i < __arraycount(ans); i++) 8257718be8SEnji Cooper if (strcmp("module", ans[i].sysctl_name) == 0) 8357718be8SEnji Cooper break; 8457718be8SEnji Cooper if (i == __arraycount(ans)) { 8557718be8SEnji Cooper errno = ENOENT; 8657718be8SEnji Cooper return -1; 8757718be8SEnji Cooper } 8857718be8SEnji Cooper 8957718be8SEnji Cooper mib[1] = ans[i].sysctl_num; 9057718be8SEnji Cooper mib[2] = CTL_QUERY; 9157718be8SEnji Cooper 9257718be8SEnji Cooper if (rump_sys___sysctl(mib, 3, ans, &alen, &q, sizeof(q)) == -1) 9357718be8SEnji Cooper return errno; 9457718be8SEnji Cooper 9557718be8SEnji Cooper for (i = 0; i < __arraycount(ans); i++) 9657718be8SEnji Cooper if (strcmp("autoload", ans[i].sysctl_name) == 0) 9757718be8SEnji Cooper break; 9857718be8SEnji Cooper if (i == __arraycount(ans)) { 9957718be8SEnji Cooper errno = ENOENT; 10057718be8SEnji Cooper return -1; 10157718be8SEnji Cooper } 10257718be8SEnji Cooper 10357718be8SEnji Cooper mib[2] = ans[i].sysctl_num; 10457718be8SEnji Cooper 10557718be8SEnji Cooper no = false; 10657718be8SEnji Cooper alen = 0; 10757718be8SEnji Cooper if (rump_sys___sysctl(mib, 3, NULL, &alen, &no, sizeof(no)) == -1) 10857718be8SEnji Cooper return errno; 10957718be8SEnji Cooper 11057718be8SEnji Cooper return 0; 11157718be8SEnji Cooper 11257718be8SEnji Cooper } 11357718be8SEnji Cooper 11457718be8SEnji Cooper #define TMPFSMODULE "librumpfs_tmpfs.so" 11557718be8SEnji Cooper ATF_TC_BODY(cmsg_modcmd, tc) 11657718be8SEnji Cooper { 11757718be8SEnji Cooper struct tmpfs_args args; 11857718be8SEnji Cooper const struct modinfo *const *mi_start, *const *mi_end; 11957718be8SEnji Cooper void *handle; 12057718be8SEnji Cooper int i, rv, loop = 0; 12157718be8SEnji Cooper 12257718be8SEnji Cooper rump_init(); 12357718be8SEnji Cooper 12457718be8SEnji Cooper if (disable_autoload() == -1) 12557718be8SEnji Cooper atf_tc_fail_errno("count not disable module autoload"); 12657718be8SEnji Cooper 12757718be8SEnji Cooper memset(&args, 0, sizeof(args)); 12857718be8SEnji Cooper args.ta_version = TMPFS_ARGS_VERSION; 12957718be8SEnji Cooper args.ta_root_mode = 0777; 13057718be8SEnji Cooper 13157718be8SEnji Cooper if (rump_sys_mkdir("/mp", 0777) == -1) 13257718be8SEnji Cooper atf_tc_fail_errno("mkdir mountpoint"); 13357718be8SEnji Cooper if (rump_sys_mount(MOUNT_TMPFS, "/mp", 0, &args, sizeof(args)) != -1) 13457718be8SEnji Cooper atf_tc_fail("mount unexpectedly succeeded"); 13557718be8SEnji Cooper 13657718be8SEnji Cooper handle = dlopen(TMPFSMODULE, RTLD_GLOBAL); 13757718be8SEnji Cooper if (handle == NULL) { 13857718be8SEnji Cooper const char *dlmsg = dlerror(); 13957718be8SEnji Cooper atf_tc_fail("cannot open %s: %s", TMPFSMODULE, dlmsg); 14057718be8SEnji Cooper } 14157718be8SEnji Cooper 14257718be8SEnji Cooper again: 14357718be8SEnji Cooper mi_start = dlsym(handle, "__start_link_set_modules"); 14457718be8SEnji Cooper mi_end = dlsym(handle, "__stop_link_set_modules"); 14557718be8SEnji Cooper if (mi_start == NULL || mi_end == NULL) 14657718be8SEnji Cooper atf_tc_fail("cannot find module info"); 14757718be8SEnji Cooper if ((rv = rump_pub_module_init(mi_start, (size_t)(mi_end-mi_start)))!=0) 14857718be8SEnji Cooper atf_tc_fail("module init failed: %d (%s)", rv, strerror(rv)); 14957718be8SEnji Cooper if ((rv = rump_pub_module_init(mi_start, (size_t)(mi_end-mi_start)))==0) 15057718be8SEnji Cooper atf_tc_fail("module double init succeeded"); 15157718be8SEnji Cooper 15257718be8SEnji Cooper if (rump_sys_mount(MOUNT_TMPFS, "/mp", 0, &args, sizeof(args)) == -1) 15357718be8SEnji Cooper atf_tc_fail_errno("still cannot mount"); 15457718be8SEnji Cooper if (rump_sys_unmount("/mp", 0) == -1) 15557718be8SEnji Cooper atf_tc_fail("cannot unmount"); 15657718be8SEnji Cooper for (i = 0; i < (int)(mi_end-mi_start); i++) { 15757718be8SEnji Cooper if ((rv = rump_pub_module_fini(mi_start[i])) != 0) 15857718be8SEnji Cooper atf_tc_fail("module fini failed: %d (%s)", 15957718be8SEnji Cooper rv, strerror(rv)); 16057718be8SEnji Cooper } 16157718be8SEnji Cooper for (i = 0; i < (int)(mi_end-mi_start); i++) { 16257718be8SEnji Cooper if ((rv = rump_pub_module_fini(mi_start[i])) == 0) 16357718be8SEnji Cooper atf_tc_fail("module double fini succeeded"); 16457718be8SEnji Cooper } 16557718be8SEnji Cooper if (loop++ == 0) 16657718be8SEnji Cooper goto again; 16757718be8SEnji Cooper 16857718be8SEnji Cooper if (dlclose(handle)) { 16957718be8SEnji Cooper const char *dlmsg = dlerror(); 17057718be8SEnji Cooper atf_tc_fail("cannot close %s: %s", TMPFSMODULE, dlmsg); 17157718be8SEnji Cooper } 17257718be8SEnji Cooper 17357718be8SEnji Cooper if (rump_sys_mount(MOUNT_TMPFS, "/mp", 0, &args, sizeof(args)) != -1) 17457718be8SEnji Cooper atf_tc_fail("mount unexpectedly succeeded"); 17557718be8SEnji Cooper } 17657718be8SEnji Cooper 17757718be8SEnji Cooper ATF_TP_ADD_TCS(tp) 17857718be8SEnji Cooper { 17957718be8SEnji Cooper ATF_TP_ADD_TC(tp, cmsg_modcmd); 18057718be8SEnji Cooper 18157718be8SEnji Cooper return atf_no_error(); 18257718be8SEnji Cooper } 183