C++程序  |  236行  |  5.81 KB

/*
 * Copyright (c) 2000 Silicon Graphics, Inc.  All Rights Reserved.
 *    AUTHOR		: Bill Branum
 *    CO-PILOT		: Steve Shaw
 *
 * 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.
 *
 * Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
 * Mountain View, CA  94043, or:
 *
 * http://www.sgi.com
 *
 * For further information regarding this notice, see:
 *
 * http://oss.sgi.com/projects/GenInfo/NoticeExplan/
 *
 */
 /*
  * TEST CASES
  *  rmdir(2) test for errno(s) EINVAL, EMLINK, EFAULT
  */

#include <errno.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <stdlib.h>
#include <string.h>
#include "test.h"
#include "safe_macros.h"

static void setup(void);
static void cleanup(void);

#if !defined(UCLINUX)
extern char *get_high_address();
int TST_TOTAL = 6;
#else
int TST_TOTAL = 4;
#endif

char *TCID = "rmdir05";

static struct stat stat_buf;
static char dir_name[256];

static char *bad_addr = NULL;

int main(int argc, char **argv)
{
	int lc;

	tst_parse_opts(argc, argv, NULL, NULL);

	setup();

	for (lc = 0; TEST_LOOPING(lc); lc++) {
		tst_count = 0;

		/*
		 * TEST CASE: 1
		 * path points to the current directory
		 */
		TEST(rmdir("."));

		if (TEST_RETURN == -1) {
			if (TEST_ERRNO & (EBUSY | ENOTEMPTY)) {
				/* For functionality tests, verify that the
				 * directory wasn't removed.
				 */
				if (stat(".", &stat_buf) == -1) {
					tst_resm(TFAIL,
						 "rmdir(\".\") removed the current working directory when it should have failed.");
				} else {
					tst_resm(TPASS,
						 "rmdir(\".\") failed to remove the current working directory. Returned %d : %s",
						 TEST_ERRNO,
						 strerror(TEST_ERRNO));
				}
			} else {
				tst_resm(TFAIL,
					 "rmdir(\".\") failed with errno %d : %s but expected %d (EBUSY)",
					 TEST_ERRNO,
					 strerror(TEST_ERRNO), EBUSY);
			}
		} else {
			tst_resm(TFAIL,
				 "rmdir(\".\") succeeded unexpectedly.");
		}

		/*
		 * TEST CASE: 2
		 * path points to the "." (dot) entry of a directory
		 */
		tst_resm(TCONF, "rmdir on \"dir/.\" supported on Linux");

		tst_resm(TCONF,
			 "linked directories test not implemented on Linux");

		/*
		 * TEST CASE: 4
		 * path argument points below the minimum allocated address space
		 */
#if !defined(UCLINUX)
		TEST(rmdir(bad_addr));

		if (TEST_RETURN == -1) {
		}

		if (TEST_RETURN == -1) {
			if (TEST_ERRNO == EFAULT) {
				tst_resm(TPASS,
					 "rmdir() - path argument points below the minimum allocated address space failed as expected with errno %d : %s",
					 TEST_ERRNO,
					 strerror(TEST_ERRNO));
			} else {
				tst_resm(TFAIL,
					 "rmdir() - path argument points below the minimum allocated address space failed with errno %d : %s but expected %d (EFAULT)",
					 TEST_ERRNO,
					 strerror(TEST_ERRNO), EFAULT);
			}
		} else {
			tst_resm(TFAIL,
				 "rmdir() - path argument points below the minimum allocated address space succeeded unexpectedly.");
		}

		/*
		 * TEST CASE: 5
		 * path argument points above the maximum allocated address space
		 */

		TEST(rmdir(get_high_address()));

		if (TEST_RETURN == -1) {
		}

		if (TEST_RETURN == -1) {
			if (TEST_ERRNO == EFAULT) {
				tst_resm(TPASS,
					 "rmdir() - path argument points above the maximum allocated address space failed as expected with errno %d : %s",
					 TEST_ERRNO,
					 strerror(TEST_ERRNO));
			} else {
				tst_resm(TFAIL,
					 "rmdir() - path argument points above the maximum allocated address space failed with errno %d : %s but expected %d (EFAULT)",
					 TEST_ERRNO,
					 strerror(TEST_ERRNO), EFAULT);
			}
		} else {
			tst_resm(TFAIL,
				 "rmdir() - path argument points above the maximum allocated address space succeeded unexpectedly.");
		}
#endif

		/*
		 * TEST CASE: 6
		 * able to remove a directory
		 */

		if (mkdir(dir_name, 0777) != 0) {
			tst_brkm(TBROK, cleanup,
				 "mkdir(\"%s\") failed with errno %d : %s",
				 dir_name, errno, strerror(errno));
		}

		TEST(rmdir(dir_name));

		if (TEST_RETURN == -1) {
			tst_resm(TFAIL,
				 "rmdir(\"%s\") failed when it should have passed. Returned %d : %s",
				 dir_name, TEST_ERRNO, strerror(TEST_ERRNO));
		} else {
			/* Verify the directory was removed. */
			if (stat(dir_name, &stat_buf) != 0) {
				tst_resm(TPASS,
					 "rmdir(\"%s\") removed the directory as expected.",
					 dir_name);
			} else {
				tst_resm(TFAIL,
					 "rmdir(\"%s\") returned a zero exit status but failed to remove the directory.",
					 dir_name);
			}
		}

	}

	cleanup();
	tst_exit();
}

void setup(void)
{
	tst_sig(FORK, DEF_HANDLER, cleanup);

	TEST_PAUSE;

	tst_tmpdir();

	/* Create a directory. */
	SAFE_MKDIR(cleanup, "dir1", 0777);

	/* Create a unique directory name. */
	sprintf(dir_name, "./dir_%d", getpid());

#if !defined(UCLINUX)
	bad_addr = mmap(0, 1, PROT_NONE,
			MAP_PRIVATE_EXCEPT_UCLINUX | MAP_ANONYMOUS, 0, 0);
	if (bad_addr == MAP_FAILED) {
		tst_brkm(TBROK, cleanup, "mmap failed");
	}
#endif
}

void cleanup(void)
{
	tst_rmdir();
}