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);
ATF_TC_HEAD(cmsg_modcmd,tc)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
disable_autoload(void)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"
ATF_TC_BODY(cmsg_modcmd,tc)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
ATF_TP_ADD_TCS(tp)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