110f81a9bSEnji Cooper /*- 2*148a8da8SEnji Cooper * Copyright (c) 2017 Enji Cooper <ngie@freebsd.org> 310f81a9bSEnji Cooper * 410f81a9bSEnji Cooper * Redistribution and use in source and binary forms, with or without 510f81a9bSEnji Cooper * modification, are permitted provided that the following conditions 610f81a9bSEnji Cooper * are met: 710f81a9bSEnji Cooper * 1. Redistributions of source code must retain the above copyright 810f81a9bSEnji Cooper * notice, this list of conditions and the following disclaimer. 910f81a9bSEnji Cooper * 2. Redistributions in binary form must reproduce the above copyright 1010f81a9bSEnji Cooper * notice, this list of conditions and the following disclaimer in the 1110f81a9bSEnji Cooper * documentation and/or other materials provided with the distribution. 1210f81a9bSEnji Cooper * 1310f81a9bSEnji Cooper * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 1410f81a9bSEnji Cooper * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 1510f81a9bSEnji Cooper * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 1610f81a9bSEnji Cooper * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 1710f81a9bSEnji Cooper * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 1810f81a9bSEnji Cooper * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 1910f81a9bSEnji Cooper * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2010f81a9bSEnji Cooper * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2110f81a9bSEnji Cooper * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2210f81a9bSEnji Cooper * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 2310f81a9bSEnji Cooper * SUCH DAMAGE. 2410f81a9bSEnji Cooper */ 2510f81a9bSEnji Cooper 2610f81a9bSEnji Cooper #include <sys/cdefs.h> 2710f81a9bSEnji Cooper __FBSDID("$FreeBSD$"); 2810f81a9bSEnji Cooper 2910f81a9bSEnji Cooper #include <sys/param.h> 3010f81a9bSEnji Cooper #include <errno.h> 3110f81a9bSEnji Cooper #include <fcntl.h> 3210f81a9bSEnji Cooper #include <kvm.h> 3310f81a9bSEnji Cooper #include <limits.h> 3410f81a9bSEnji Cooper #include <signal.h> 3510f81a9bSEnji Cooper #include <stdio.h> 3610f81a9bSEnji Cooper #include <stdlib.h> 3710f81a9bSEnji Cooper 3810f81a9bSEnji Cooper #include <atf-c.h> 3910f81a9bSEnji Cooper 4010f81a9bSEnji Cooper #include "kvm_private.h" 4110f81a9bSEnji Cooper 4210f81a9bSEnji Cooper #include "kvm_test_common.h" 4310f81a9bSEnji Cooper 4410f81a9bSEnji Cooper ATF_TC(kvm_geterr_negative_test_NULL); 4510f81a9bSEnji Cooper ATF_TC_HEAD(kvm_geterr_negative_test_NULL, tc) 4610f81a9bSEnji Cooper { 4710f81a9bSEnji Cooper 4810f81a9bSEnji Cooper atf_tc_set_md_var(tc, "descr", 4910f81a9bSEnji Cooper "test that kvm_geterr(NULL) returns NULL"); 5010f81a9bSEnji Cooper } 5110f81a9bSEnji Cooper 5210f81a9bSEnji Cooper ATF_TC_BODY(kvm_geterr_negative_test_NULL, tc) 5310f81a9bSEnji Cooper { 5410f81a9bSEnji Cooper 5510f81a9bSEnji Cooper ATF_REQUIRE(!errbuf_has_error(kvm_geterr(NULL))); 5610f81a9bSEnji Cooper } 5710f81a9bSEnji Cooper 589f825b1bSEnji Cooper /* 1100090 was where kvm_open2(3) was introduced. */ 599f825b1bSEnji Cooper #if __FreeBSD_version >= 1100091 6010f81a9bSEnji Cooper ATF_TC(kvm_geterr_positive_test_error); 6110f81a9bSEnji Cooper ATF_TC_HEAD(kvm_geterr_positive_test_error, tc) 6210f81a9bSEnji Cooper { 6310f81a9bSEnji Cooper 6410f81a9bSEnji Cooper atf_tc_set_md_var(tc, "descr", 6510f81a9bSEnji Cooper "test that kvm_geterr(kd) when kd doesn't contain an error returns \"\""); 6610f81a9bSEnji Cooper atf_tc_set_md_var(tc, "require.user", "root"); 6710f81a9bSEnji Cooper } 6810f81a9bSEnji Cooper 6910f81a9bSEnji Cooper ATF_TC_BODY(kvm_geterr_positive_test_error, tc) 7010f81a9bSEnji Cooper { 7110f81a9bSEnji Cooper kvm_t *kd; 7210f81a9bSEnji Cooper char *error_msg; 7310f81a9bSEnji Cooper 7410f81a9bSEnji Cooper errbuf_clear(); 7510f81a9bSEnji Cooper kd = kvm_open2(NULL, NULL, O_RDONLY, errbuf, NULL); 7610f81a9bSEnji Cooper ATF_CHECK(!errbuf_has_error(errbuf)); 7710f81a9bSEnji Cooper ATF_REQUIRE_MSG(kd != NULL, "kvm_open2 failed: %s", errbuf); 7810f81a9bSEnji Cooper ATF_REQUIRE_MSG(kvm_write(kd, 0, NULL, 0) == -1, 7910f81a9bSEnji Cooper "kvm_write succeeded unexpectedly on an O_RDONLY file descriptor"); 8010f81a9bSEnji Cooper error_msg = kvm_geterr(kd); 8110f81a9bSEnji Cooper ATF_CHECK(errbuf_has_error(error_msg)); 8210f81a9bSEnji Cooper ATF_REQUIRE_MSG(kvm_close(kd) == 0, "kvm_close failed: %s", 8310f81a9bSEnji Cooper strerror(errno)); 8410f81a9bSEnji Cooper } 8510f81a9bSEnji Cooper 8610f81a9bSEnji Cooper ATF_TC(kvm_geterr_positive_test_no_error); 8710f81a9bSEnji Cooper ATF_TC_HEAD(kvm_geterr_positive_test_no_error, tc) 8810f81a9bSEnji Cooper { 8910f81a9bSEnji Cooper 9010f81a9bSEnji Cooper atf_tc_set_md_var(tc, "descr", 9110f81a9bSEnji Cooper "test that kvm_geterr(kd) when kd contains an error returns an error message"); 9210f81a9bSEnji Cooper atf_tc_set_md_var(tc, "require.user", "root"); 9310f81a9bSEnji Cooper } 9410f81a9bSEnji Cooper 9510f81a9bSEnji Cooper ATF_TC_BODY(kvm_geterr_positive_test_no_error, tc) 9610f81a9bSEnji Cooper { 9710f81a9bSEnji Cooper #define ALL_IS_WELL "that ends well" 9810f81a9bSEnji Cooper kvm_t *kd; 9910f81a9bSEnji Cooper char *error_msg; 10010f81a9bSEnji Cooper struct nlist nl[] = { 10110f81a9bSEnji Cooper #define SYMNAME "_mp_maxcpus" 10210f81a9bSEnji Cooper #define X_MAXCPUS 0 10310f81a9bSEnji Cooper { SYMNAME, 0, 0, 0, 0 }, 10410f81a9bSEnji Cooper { NULL, 0, 0, 0, 0 }, 10510f81a9bSEnji Cooper }; 10610f81a9bSEnji Cooper ssize_t rc; 10710f81a9bSEnji Cooper int mp_maxcpus, retcode; 10810f81a9bSEnji Cooper 10910f81a9bSEnji Cooper errbuf_clear(); 11010f81a9bSEnji Cooper kd = kvm_open2(NULL, NULL, O_RDONLY, errbuf, NULL); 11110f81a9bSEnji Cooper ATF_CHECK(!errbuf_has_error(errbuf)); 11210f81a9bSEnji Cooper ATF_REQUIRE_MSG(kd != NULL, "kvm_open2 failed: %s", errbuf); 11310f81a9bSEnji Cooper retcode = kvm_nlist(kd, nl); 11410f81a9bSEnji Cooper ATF_REQUIRE_MSG(retcode != -1, 11510f81a9bSEnji Cooper "kvm_nlist failed (returned %d): %s", retcode, kvm_geterr(kd)); 11610f81a9bSEnji Cooper if (nl[X_MAXCPUS].n_type == 0) 11710f81a9bSEnji Cooper atf_tc_skip("symbol (\"%s\") couldn't be found", SYMNAME); 11810f81a9bSEnji Cooper _kvm_err(kd, NULL, "%s", ALL_IS_WELL); /* XXX: internal API */ 11910f81a9bSEnji Cooper rc = kvm_read(kd, nl[X_MAXCPUS].n_value, &mp_maxcpus, 12010f81a9bSEnji Cooper sizeof(mp_maxcpus)); 12110f81a9bSEnji Cooper 12210f81a9bSEnji Cooper ATF_REQUIRE_MSG(rc != -1, "kvm_read failed: %s", kvm_geterr(kd)); 12310f81a9bSEnji Cooper error_msg = kvm_geterr(kd); 12410f81a9bSEnji Cooper ATF_REQUIRE_MSG(strcmp(error_msg, ALL_IS_WELL) == 0, 12510f81a9bSEnji Cooper "error message changed: %s", error_msg); 12610f81a9bSEnji Cooper ATF_REQUIRE_MSG(kvm_close(kd) == 0, "kvm_close failed: %s", 12710f81a9bSEnji Cooper strerror(errno)); 12810f81a9bSEnji Cooper } 129fbf8ca6dSEnji Cooper #endif 13010f81a9bSEnji Cooper 13110f81a9bSEnji Cooper ATF_TP_ADD_TCS(tp) 13210f81a9bSEnji Cooper { 13310f81a9bSEnji Cooper 13410f81a9bSEnji Cooper ATF_TP_ADD_TC(tp, kvm_geterr_negative_test_NULL); 135fbf8ca6dSEnji Cooper #if __FreeBSD_version >= 1100091 13610f81a9bSEnji Cooper ATF_TP_ADD_TC(tp, kvm_geterr_positive_test_error); 13710f81a9bSEnji Cooper ATF_TP_ADD_TC(tp, kvm_geterr_positive_test_no_error); 138fbf8ca6dSEnji Cooper #endif 13910f81a9bSEnji Cooper 14010f81a9bSEnji Cooper return (atf_no_error()); 14110f81a9bSEnji Cooper } 142