403 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			403 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * Copyright (C) 2012 Linux Test Project, Inc.
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  * Further, this software is distributed without any warranty that it
 | |
|  * is free of the rightful claim of any third person regarding
 | |
|  * infringement or the like.  Any license provided herein, whether
 | |
|  * implied or otherwise, applies only to this software file.  Patent
 | |
|  * licenses, if any, provided herein do not apply to combinations of
 | |
|  * this program with other software, or any other product whatsoever.
 | |
|  *
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * errno tests shared by process_vm_readv, process_vm_writev tests.
 | |
|  */
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/syscall.h>
 | |
| #include <sys/uio.h>
 | |
| #include <sys/wait.h>
 | |
| #include <sys/mman.h>
 | |
| #include <errno.h>
 | |
| #include <signal.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <unistd.h>
 | |
| #include <limits.h>
 | |
| #include <pwd.h>
 | |
| #include "config.h"
 | |
| #include "test.h"
 | |
| #include "safe_macros.h"
 | |
| #include "lapi/syscalls.h"
 | |
| 
 | |
| struct process_vm_params {
 | |
| 	int len;
 | |
| 	char *ldummy;
 | |
| 	char *rdummy;
 | |
| 	pid_t pid;
 | |
| 	struct iovec *lvec;
 | |
| 	unsigned long liovcnt;
 | |
| 	struct iovec *rvec;
 | |
| 	unsigned long riovcnt;
 | |
| 	unsigned long flags;
 | |
| };
 | |
| 
 | |
| static int rflag;
 | |
| static int wflag;
 | |
| 
 | |
| static option_t options[] = {
 | |
| 	{"r", &rflag, NULL},
 | |
| 	{"w", &wflag, NULL},
 | |
| 	{NULL, NULL, NULL}
 | |
| };
 | |
| 
 | |
| static char TCID_readv[] = "process_vm_readv";
 | |
| static char TCID_writev[] = "process_vm_writev";
 | |
| char *TCID = "cma01";
 | |
| int TST_TOTAL = 1;
 | |
| static void (*cma_test_params) (struct process_vm_params * params) = NULL;
 | |
| 
 | |
| static void setup(char *argv[]);
 | |
| static void cleanup(void);
 | |
| static void help(void);
 | |
| 
 | |
| static void cma_test_params_read(struct process_vm_params *params);
 | |
| static void cma_test_params_write(struct process_vm_params *params);
 | |
| static void cma_test_errnos(void);
 | |
| 
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
| 	int lc;
 | |
| 
 | |
| 	tst_parse_opts(argc, argv, options, &help);
 | |
| 
 | |
| 	setup(argv);
 | |
| 	for (lc = 0; TEST_LOOPING(lc); lc++) {
 | |
| 		tst_count = 0;
 | |
| 		cma_test_errnos();
 | |
| 	}
 | |
| 	cleanup();
 | |
| 	tst_exit();
 | |
| }
 | |
| 
 | |
| static void setup(char *argv[])
 | |
| {
 | |
| 	tst_require_root();
 | |
| 
 | |
| 	if (rflag && wflag)
 | |
| 		tst_brkm(TBROK, NULL, "Parameters -r -w can not be used"
 | |
| 			 " at the same time.");
 | |
| 	else if (rflag) {
 | |
| 		TCID = TCID_readv;
 | |
| 		cma_test_params = cma_test_params_read;
 | |
| 	} else if (wflag) {
 | |
| 		TCID = TCID_writev;
 | |
| 		cma_test_params = cma_test_params_write;
 | |
| 	} else
 | |
| 		tst_brkm(TBROK, NULL, "Parameter missing, required -r or -w.");
 | |
| 	TEST_PAUSE;
 | |
| }
 | |
| 
 | |
| static void cleanup(void)
 | |
| {
 | |
| }
 | |
| 
 | |
| static void help(void)
 | |
| {
 | |
| 	printf("  -r      Use process_vm_readv\n");
 | |
| 	printf("  -w      Use process_vm_writev\n");
 | |
| }
 | |
| 
 | |
| static void cma_test_params_read(struct process_vm_params *params)
 | |
| {
 | |
| 	TEST(ltp_syscall(__NR_process_vm_readv,
 | |
| 			 params->pid,
 | |
| 			 params->lvec, params->liovcnt,
 | |
| 			 params->rvec, params->riovcnt,
 | |
| 			 params->flags));
 | |
| }
 | |
| 
 | |
| static void cma_test_params_write(struct process_vm_params *params)
 | |
| {
 | |
| 	TEST(ltp_syscall(__NR_process_vm_writev,
 | |
| 			 params->pid,
 | |
| 			 params->lvec, params->liovcnt,
 | |
| 			 params->rvec, params->riovcnt,
 | |
| 			 params->flags));
 | |
| }
 | |
| 
 | |
| static int cma_check_ret(long expected_ret, long act_ret)
 | |
| {
 | |
| 	if (expected_ret == act_ret) {
 | |
| 		tst_resm(TPASS, "expected ret success - "
 | |
| 			 "returned value = %ld", act_ret);
 | |
| 	} else {
 | |
| 		tst_resm(TFAIL, "unexpected failure - "
 | |
| 			 "returned value = %ld, expected: %ld",
 | |
| 			 act_ret, expected_ret);
 | |
| 		return 1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cma_check_errno(long expected_errno)
 | |
| {
 | |
| 	if (TEST_ERRNO == expected_errno)
 | |
| 		tst_resm(TPASS | TTERRNO, "expected failure");
 | |
| 	else if (TEST_ERRNO == 0) {
 | |
| 		tst_resm(TFAIL, "call succeeded unexpectedly");
 | |
| 		return 1;
 | |
| 	} else {
 | |
| 		tst_resm(TFAIL | TTERRNO, "unexpected failure - "
 | |
| 			 "expected = %ld : %s, actual",
 | |
| 			 expected_errno, strerror(expected_errno));
 | |
| 		return 2;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct process_vm_params *cma_alloc_sane_params(void)
 | |
| {
 | |
| 	struct process_vm_params *sane_params;
 | |
| 	int len;
 | |
| 
 | |
| 	len = getpagesize();
 | |
| 	sane_params = SAFE_MALLOC(NULL, sizeof(struct process_vm_params));
 | |
| 	sane_params->len = len;
 | |
| 	sane_params->ldummy = SAFE_MALLOC(NULL, len);
 | |
| 	sane_params->rdummy = SAFE_MALLOC(NULL, len);
 | |
| 
 | |
| 	sane_params->lvec = SAFE_MALLOC(NULL, sizeof(struct iovec));
 | |
| 	sane_params->lvec->iov_base = sane_params->ldummy;
 | |
| 	sane_params->lvec->iov_len = len;
 | |
| 	sane_params->liovcnt = 1;
 | |
| 
 | |
| 	sane_params->rvec = SAFE_MALLOC(NULL, sizeof(struct iovec));
 | |
| 	sane_params->rvec->iov_base = sane_params->rdummy;
 | |
| 	sane_params->rvec->iov_len = len;
 | |
| 	sane_params->riovcnt = 1;
 | |
| 
 | |
| 	sane_params->flags = 0;
 | |
| 	sane_params->pid = getpid();
 | |
| 
 | |
| 	return sane_params;
 | |
| }
 | |
| 
 | |
| static void cma_free_params(struct process_vm_params *params)
 | |
| {
 | |
| 	if (params) {
 | |
| 		free(params->ldummy);
 | |
| 		free(params->rdummy);
 | |
| 		free(params->lvec);
 | |
| 		free(params->rvec);
 | |
| 		free(params);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void cma_test_sane_params(void)
 | |
| {
 | |
| 	struct process_vm_params *sane_params;
 | |
| 
 | |
| 	sane_params = cma_alloc_sane_params();
 | |
| 	tst_resm(TINFO, "test_sane_params");
 | |
| 	cma_test_params(sane_params);
 | |
| 	cma_check_ret(sane_params->len, TEST_RETURN);
 | |
| 	cma_free_params(sane_params);
 | |
| }
 | |
| 
 | |
| static void cma_test_flags(void)
 | |
| {
 | |
| 	struct process_vm_params *params;
 | |
| 	long flags[] = { -INT_MAX, -1, 1, INT_MAX, 0 };
 | |
| 	int flags_size = sizeof(flags) / sizeof(flags[0]);
 | |
| 	int i;
 | |
| 
 | |
| 	params = cma_alloc_sane_params();
 | |
| 	for (i = 0; i < flags_size; i++) {
 | |
| 		params->flags = flags[i];
 | |
| 		tst_resm(TINFO, "test_flags, flags=%ld", flags[i]);
 | |
| 		cma_test_params(params);
 | |
| 		/* atm. only flags == 0 is allowed, everything else
 | |
| 		 * should fail with EINVAL */
 | |
| 		if (flags[i] != 0) {
 | |
| 			cma_check_ret(-1, TEST_RETURN);
 | |
| 			cma_check_errno(EINVAL);
 | |
| 		} else {
 | |
| 			cma_check_ret(params->len, TEST_RETURN);
 | |
| 		}
 | |
| 	}
 | |
| 	cma_free_params(params);
 | |
| }
 | |
| 
 | |
| static void cma_test_iov_len_overflow(void)
 | |
| {
 | |
| 	struct process_vm_params *params;
 | |
| 	ssize_t maxlen = -1;
 | |
| 	params = cma_alloc_sane_params();
 | |
| 
 | |
| 	params->lvec->iov_len = maxlen;
 | |
| 	params->rvec->iov_len = maxlen;
 | |
| 	tst_resm(TINFO, "test_iov_len_overflow");
 | |
| 	cma_test_params(params);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EINVAL);
 | |
| 	cma_free_params(params);
 | |
| }
 | |
| 
 | |
| static void cma_test_iov_invalid(void)
 | |
| {
 | |
| 	struct process_vm_params *sane_params;
 | |
| 	struct process_vm_params params_copy;
 | |
| 
 | |
| 	sane_params = cma_alloc_sane_params();
 | |
| 	/* make a shallow copy we can 'damage' */
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	tst_resm(TINFO, "test_iov_invalid - lvec->iov_base");
 | |
| 	params_copy.lvec->iov_base = (void *)-1;
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	tst_resm(TINFO, "test_iov_invalid - rvec->iov_base");
 | |
| 	params_copy.rvec->iov_base = (void *)-1;
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	tst_resm(TINFO, "test_iov_invalid - lvec");
 | |
| 	params_copy.lvec = (void *)-1;
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	tst_resm(TINFO, "test_iov_invalid - rvec");
 | |
| 	params_copy.rvec = (void *)-1;
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	cma_free_params(sane_params);
 | |
| }
 | |
| 
 | |
| static void cma_test_invalid_pid(void)
 | |
| {
 | |
| 	pid_t invalid_pid = -1;
 | |
| 	struct process_vm_params *params;
 | |
| 
 | |
| 	params = cma_alloc_sane_params();
 | |
| 	tst_resm(TINFO, "test_invalid_pid");
 | |
| 	params->pid = invalid_pid;
 | |
| 	cma_test_params(params);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(ESRCH);
 | |
| 	cma_free_params(params);
 | |
| 
 | |
| 	invalid_pid = tst_get_unused_pid(cleanup);
 | |
| 
 | |
| 	params = cma_alloc_sane_params();
 | |
| 	params->pid = invalid_pid;
 | |
| 	cma_test_params(params);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(ESRCH);
 | |
| 	cma_free_params(params);
 | |
| }
 | |
| 
 | |
| static void cma_test_invalid_perm(void)
 | |
| {
 | |
| 	char nobody_uid[] = "nobody";
 | |
| 	struct passwd *ltpuser;
 | |
| 	int status;
 | |
| 	struct process_vm_params *params;
 | |
| 	pid_t child_pid;
 | |
| 	pid_t parent_pid;
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	tst_resm(TINFO, "test_invalid_perm");
 | |
| 	parent_pid = getpid();
 | |
| 	child_pid = fork();
 | |
| 	switch (child_pid) {
 | |
| 	case -1:
 | |
| 		tst_brkm(TBROK | TERRNO, cleanup, "fork");
 | |
| 		break;
 | |
| 	case 0:
 | |
| 		ltpuser = getpwnam(nobody_uid);
 | |
| 		if (ltpuser == NULL)
 | |
| 			tst_brkm(TBROK | TERRNO, NULL, "getpwnam failed");
 | |
| 		SAFE_SETUID(NULL, ltpuser->pw_uid);
 | |
| 
 | |
| 		params = cma_alloc_sane_params();
 | |
| 		params->pid = parent_pid;
 | |
| 		cma_test_params(params);
 | |
| 		ret |= cma_check_ret(-1, TEST_RETURN);
 | |
| 		ret |= cma_check_errno(EPERM);
 | |
| 		cma_free_params(params);
 | |
| 		exit(ret);
 | |
| 	default:
 | |
| 		SAFE_WAITPID(cleanup, child_pid, &status, 0);
 | |
| 		if (!WIFEXITED(status) || WEXITSTATUS(status) != 0)
 | |
| 			tst_resm(TFAIL, "child returns %d", status);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void cma_test_invalid_protection(void)
 | |
| {
 | |
| 	struct process_vm_params *sane_params;
 | |
| 	struct process_vm_params params_copy;
 | |
| 	void *p;
 | |
| 
 | |
| 	sane_params = cma_alloc_sane_params();
 | |
| 	/* make a shallow copy we can 'damage' */
 | |
| 
 | |
| 	p = mmap(NULL, getpagesize(), PROT_NONE,
 | |
| 		 MAP_PRIVATE | MAP_ANONYMOUS, 0, 0);
 | |
| 	if (p == MAP_FAILED)
 | |
| 		tst_brkm(TBROK | TERRNO, cleanup, "mmap");
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	params_copy.lvec->iov_base = p;
 | |
| 	tst_resm(TINFO, "test_invalid_protection lvec");
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	params_copy = *sane_params;
 | |
| 	params_copy.rvec->iov_base = p;
 | |
| 	tst_resm(TINFO, "test_invalid_protection rvec");
 | |
| 	cma_test_params(¶ms_copy);
 | |
| 	cma_check_ret(-1, TEST_RETURN);
 | |
| 	cma_check_errno(EFAULT);
 | |
| 
 | |
| 	SAFE_MUNMAP(cleanup, p, getpagesize());
 | |
| 
 | |
| 	cma_free_params(sane_params);
 | |
| }
 | |
| 
 | |
| static void cma_test_errnos(void)
 | |
| {
 | |
| 	cma_test_sane_params();
 | |
| 	cma_test_flags();
 | |
| 	cma_test_iov_len_overflow();
 | |
| 	cma_test_iov_invalid();
 | |
| 	cma_test_invalid_pid();
 | |
| 	cma_test_invalid_perm();
 | |
| 	cma_test_invalid_protection();
 | |
| }
 |