C++程序  |  119行  |  3.17 KB

/*
 * Copyright (c) International Business Machines  Corp., 2001
 *    03/2001 - Written by Wayne Boyer
 *    12/03/2008 Matthieu Fertré (Matthieu.Fertre@irisa.fr)
 * Copyright (c) 2018 Cyril Hrubis <chrubis@suse.cz>
 *
 * This program is free software;  you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY;  without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
 * the GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program;  if not, write to the Free Software Foundation,
 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */
/*
 * Test for EACCES, EFAULT and EINVAL errors using a variety of incorrect
 * calls.
 */
#include <errno.h>
#include <pwd.h>

#include "tst_test.h"
#include "tst_safe_sysv_ipc.h"
#include "libnewipc.h"

static int msg_id1 = -1;
static int msg_id2 = -1;
static int msg_id3 = -1;
static int bad_q = -1;

struct msqid_ds q_buf;

struct tcase {
	int *msg_id;
	int cmd;
	struct msqid_ds *buf;
	int error;
} tc[] = {
	/* EACCES - there is no read permission for the queue */
	{&msg_id1, IPC_STAT, &q_buf, EACCES},
	/* EFAULT - the structure address is invalid - IPC_STAT */
	{&msg_id2, IPC_STAT, (struct msqid_ds *)-1, EFAULT},
	/* EFAULT - the structure address is invalid - IPC_SET */
	{&msg_id2, IPC_SET, (struct msqid_ds *)-1, EFAULT},
	/* EINVAL - the command (-1) is invalid */
	{&msg_id2, -1, &q_buf, EINVAL},
	/* EINVAL - the queue id is invalid - IPC_STAT */
	{&bad_q, IPC_STAT, &q_buf, EINVAL},
	/* EINVAL - the queue id is invalid - IPC_SET */
	{&bad_q, IPC_SET, &q_buf, EINVAL},
	/* EPERM - cannot delete root owned queue */
	{&msg_id3, IPC_RMID, NULL, EPERM},
};

static void verify_msgctl(unsigned int i)
{
	TEST(msgctl(*(tc[i].msg_id), tc[i].cmd, tc[i].buf));

	if (TST_RET != -1) {
		tst_res(TFAIL, "msgctl() returned %li", TST_RET);
		return;
	}

	if (TST_ERR == tc[i].error) {
		tst_res(TPASS | TTERRNO, "msgctl(%i, %i, %p)",
			*tc[i].msg_id, tc[i].cmd, tc[i].buf);
		return;
	}

	tst_res(TFAIL | TTERRNO, "msgctl(%i, %i, %p) expected %s",
		*tc[i].msg_id, tc[i].cmd, tc[i].buf, tst_strerrno(tc[i].error));
}

static void setup(void)
{
	key_t msgkey1, msgkey2;
	struct passwd *ltpuser;

	msg_id3 = SAFE_MSGGET(IPC_PRIVATE, IPC_CREAT | MSG_RW);

	ltpuser = SAFE_GETPWNAM("nobody");
	SAFE_SETEUID(ltpuser->pw_uid);

	msgkey1 = GETIPCKEY();
	msgkey2 = GETIPCKEY();

	msg_id1 = SAFE_MSGGET(msgkey1, IPC_CREAT | IPC_EXCL);
	msg_id2 = SAFE_MSGGET(msgkey2, IPC_CREAT | IPC_EXCL | MSG_RD | MSG_WR);
}

static void cleanup(void)
{
	if (msg_id1 >= 0)
		SAFE_MSGCTL(msg_id1, IPC_RMID, NULL);

	if (msg_id2 >= 0)
		SAFE_MSGCTL(msg_id2, IPC_RMID, NULL);

	if (msg_id3 >= 0) {
		SAFE_SETEUID(0);
		SAFE_MSGCTL(msg_id3, IPC_RMID, NULL);
	}
}

static struct tst_test test = {
	.setup = setup,
	.cleanup = cleanup,
	.test = verify_msgctl,
	.tcnt = ARRAY_SIZE(tc),
	.needs_tmpdir = 1,
	.needs_root = 1,
};