/*	$NetBSD: h_tools.c,v 1.4 2011/06/11 18:03:17 christos Exp $	*/

/*
 * Copyright (c) 2005, 2006 The NetBSD Foundation, Inc.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

/*
 * Helper tools for several tests.  These are kept in a single file due
 * to the limitations of bsd.prog.mk to build a single program in a
 * given directory.
 */

#include <sys/param.h>
#include <sys/types.h>
#include <sys/event.h>
#include <sys/mount.h>
#include <sys/statvfs.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/un.h>

#include <assert.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#ifdef __FreeBSD__
#include <inttypes.h>
#endif

/* --------------------------------------------------------------------- */

static int getfh_main(int, char **);
static int kqueue_main(int, char **);
static int rename_main(int, char **);
static int sockets_main(int, char **);
static int statvfs_main(int, char **);

/* --------------------------------------------------------------------- */

int
getfh_main(int argc, char **argv)
{
	int error;
	void *fh;
	size_t fh_size;

	if (argc < 2)
		return EXIT_FAILURE;

#ifdef __FreeBSD__
	fh_size = sizeof(fhandle_t);
#else
	fh_size = 0;
#endif

	fh = NULL;
	for (;;) {
		if (fh_size) {
			fh = malloc(fh_size);
			if (fh == NULL) {
				fprintf(stderr, "out of memory");
				return EXIT_FAILURE;
			}
		}
		/*
		 * The kernel provides the necessary size in fh_size -
		 * but it may change if someone moves things around,
		 * so retry untill we have enough memory.
		 */
#ifdef __FreeBSD__
		error = getfh(argv[1], fh);
#else
		error = getfh(argv[1], fh, &fh_size);
#endif
		if (error == 0) {
			break;
		} else {
			if (fh != NULL)
				free(fh);
			if (errno != E2BIG) {
				warn("getfh");
				return EXIT_FAILURE;
			}
		}
	}

	error = write(STDOUT_FILENO, fh, fh_size);
	if (error == -1) {
		warn("write");
		return EXIT_FAILURE;
	}
	free(fh);

	return 0;
}

/* --------------------------------------------------------------------- */

int
kqueue_main(int argc, char **argv)
{
	char *line;
	int i, kq;
	size_t len;
	struct kevent *changes, event;

	if (argc < 2)
		return EXIT_FAILURE;

	argc--;
	argv++;

	changes = malloc(sizeof(struct kevent) * argc);
	if (changes == NULL)
		errx(EXIT_FAILURE, "not enough memory");

	for (i = 0; i < argc; i++) {
		int fd;

		fd = open(argv[i], O_RDONLY);
		if (fd == -1)
			err(EXIT_FAILURE, "cannot open %s", argv[i]);

		EV_SET(&changes[i], fd, EVFILT_VNODE,
		    EV_ADD | EV_ENABLE | EV_ONESHOT,
		    NOTE_ATTRIB | NOTE_DELETE | NOTE_EXTEND | NOTE_LINK |
		    NOTE_RENAME | NOTE_REVOKE | NOTE_WRITE,
		    0, 0);
	}

	kq = kqueue();
	if (kq == -1)
		err(EXIT_FAILURE, "kqueue");

	while ((line = fgetln(stdin, &len)) != NULL) {
		int ec, nev;
		struct timespec to;

		to.tv_sec = 0;
		to.tv_nsec = 100000;

		(void)kevent(kq, changes, argc, &event, 1, &to);

		assert(len > 0);
		assert(line[len - 1] == '\n');
		line[len - 1] = '\0';
		ec = system(line);
		if (ec != EXIT_SUCCESS)
			errx(ec, "%s returned %d", line, ec);

		do {
			nev = kevent(kq, changes, argc, &event, 1, &to);
			if (nev == -1)
				err(EXIT_FAILURE, "kevent");
			else if (nev > 0) {
				for (i = 0; i < argc; i++)
					if (event.ident == changes[i].ident)
						break;

				if (event.fflags & NOTE_ATTRIB)
					printf("%s - NOTE_ATTRIB\n", argv[i]);
				if (event.fflags & NOTE_DELETE)
					printf("%s - NOTE_DELETE\n", argv[i]);
				if (event.fflags & NOTE_EXTEND)
					printf("%s - NOTE_EXTEND\n", argv[i]);
				if (event.fflags & NOTE_LINK)
					printf("%s - NOTE_LINK\n", argv[i]);
				if (event.fflags & NOTE_RENAME)
					printf("%s - NOTE_RENAME\n", argv[i]);
				if (event.fflags & NOTE_REVOKE)
					printf("%s - NOTE_REVOKE\n", argv[i]);
				if (event.fflags & NOTE_WRITE)
					printf("%s - NOTE_WRITE\n", argv[i]);
			}
		} while (nev > 0);
	}

	for (i = 0; i < argc; i++)
		close(changes[i].ident);
	free(changes);

	return EXIT_SUCCESS;
}

/* --------------------------------------------------------------------- */

int
rename_main(int argc, char **argv)
{

	if (argc < 3)
		return EXIT_FAILURE;

	if (rename(argv[1], argv[2]) == -1) {
		warn("rename");
		return EXIT_FAILURE;
	}

	return EXIT_SUCCESS;
}

/* --------------------------------------------------------------------- */

int
sockets_main(int argc, char **argv)
{
	int error, fd;
	struct sockaddr_un addr;

	if (argc < 2)
		return EXIT_FAILURE;

	fd = socket(PF_LOCAL, SOCK_STREAM, 0);
	if (fd == -1) {
		warn("socket");
		return EXIT_FAILURE;
	}

#ifdef	__FreeBSD__
	memset(&addr, 0, sizeof(addr));
#endif
	(void)strlcpy(addr.sun_path, argv[1], sizeof(addr.sun_path));
	addr.sun_family = PF_UNIX;
#ifdef	__FreeBSD__
	error = bind(fd, (struct sockaddr *)&addr, SUN_LEN(&addr));
#else
	error = bind(fd, (struct sockaddr *)&addr, sizeof(addr));
#endif
	if (error == -1) {
		warn("connect");
#ifdef	__FreeBSD__
		(void)close(fd);
#endif
		return EXIT_FAILURE;
	}

	close(fd);

	return EXIT_SUCCESS;
}

/* --------------------------------------------------------------------- */

int
statvfs_main(int argc, char **argv)
{
	int error;
	struct statvfs buf;

	if (argc < 2)
		return EXIT_FAILURE;

	error = statvfs(argv[1], &buf);
	if (error != 0) {
		warn("statvfs");
		return EXIT_FAILURE;
	}

	(void)printf("f_bsize=%lu\n", buf.f_bsize);
	(void)printf("f_blocks=%" PRId64 "\n", buf.f_blocks);
	(void)printf("f_bfree=%" PRId64 "\n", buf.f_bfree);
	(void)printf("f_files=%" PRId64 "\n", buf.f_files);

	return EXIT_SUCCESS;
}

/* --------------------------------------------------------------------- */

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

	if (argc < 2)
		return EXIT_FAILURE;

	argc -= 1;
	argv += 1;

	if (strcmp(argv[0], "getfh") == 0)
		error = getfh_main(argc, argv);
	else if (strcmp(argv[0], "kqueue") == 0)
		error = kqueue_main(argc, argv);
	else if (strcmp(argv[0], "rename") == 0)
		error = rename_main(argc, argv);
	else if (strcmp(argv[0], "sockets") == 0)
		error = sockets_main(argc, argv);
	else if (strcmp(argv[0], "statvfs") == 0)
		error = statvfs_main(argc, argv);
	else
		error = EXIT_FAILURE;

	return error;
}