/*	$NetBSD: main.c,v 1.14 2020/05/24 18:42:20 jmcneill Exp $	*/

/*-
 * Copyright (c) 2002 Marcel Moolenaar
 * 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 AUTHOR ``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 AUTHOR 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.
 *
 * CRC32 code derived from work by Gary S. Brown.
 */

#if HAVE_NBTOOL_CONFIG_H
#include "nbtool_config.h"
#endif

#include <sys/cdefs.h>
#ifdef __RCSID
__RCSID("$NetBSD: main.c,v 1.14 2020/05/24 18:42:20 jmcneill Exp $");
#endif

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <err.h>
#include <errno.h>
#include <sys/stat.h>
#ifndef NBTOOL_CONFIG_H
#include <util.h>
#endif

#include "map.h"
#include "gpt.h"

static const struct gpt_cmd c_null;

extern const struct gpt_cmd
	c_add,
#ifndef HAVE_NBTOOL_CONFIG_H
	c_backup,
#endif
	c_biosboot,
	c_create,
	c_destroy,
	c_header,
	c_label,
	c_migrate,
	c_recover,
	c_remove,
	c_resize,
	c_resizedisk,
#ifndef HAVE_NBTOOL_CONFIG_H
	c_restore,
#endif
	c_set,
	c_show,
	c_type,
	c_unset,
	c_uuid;

static const struct gpt_cmd *cmdsw[] = {
	&c_add,
#ifndef HAVE_NBTOOL_CONFIG_H
	&c_backup,
#endif
	&c_biosboot,
	&c_create,
	&c_destroy,
	&c_header,
	&c_label,
	&c_migrate,
	&c_recover,
	&c_remove,
	&c_resize,
	&c_resizedisk,
#ifndef HAVE_NBTOOL_CONFIG_H
	&c_restore,
#endif
	&c_set,
	&c_show,
	&c_type,
	&c_unset,
	&c_uuid,
	&c_null,
};

__dead static void
usage(void)
{
	const char *p = getprogname();
	const char *f =
	    "[-nrqv] [-m mediasize] [-s sectorsize] [-T timestamp]";
	size_t i;

	if (strcmp(p, "gpt") == 0)
		fprintf(stderr,
		    "Usage: %s %s command device\n", p, f);
	else
		fprintf(stderr,
		    "Usage: %s %s device command\n", p, f);
	fprintf(stderr, "Commands:\n");
	for (i = 0; i < __arraycount(cmdsw); i++)
		gpt_usage("\t", cmdsw[i]);
	exit(EXIT_FAILURE);
}

static void
prefix(const char *cmd)
{
	char *pfx;

	if (asprintf(&pfx, "%s %s", getprogname(), cmd) < 0)
		pfx = NULL;
	else
		setprogname(pfx);
}

static time_t
get_tstamp(const char *b)
{
	struct stat st;
	char *eb;
	long long l;
#ifndef HAVE_NBTOOL_CONFIG_H
	time_t when;
#endif

	if (stat(b, &st) != -1)
		return (time_t)st.st_mtime;

#ifndef HAVE_NBTOOL_CONFIG_H
	errno = 0;
	if ((when = parsedate(b, NULL, NULL)) != -1 || errno == 0)
		return when;
#endif
	errno = 0;
	l = strtoll(b, &eb, 0);
	if (b == eb || *eb || errno)
		errx(EXIT_FAILURE, "Can't parse timestamp `%s'", b);
	return (time_t)l;
}

int
main(int argc, char *argv[])
{
	char *cmd, *p, *dev = NULL;
	int ch, i;
	u_int secsz = 0;
	off_t mediasz = 0;
	int flags = 0;
	int verbose = 0;
	time_t timestamp = 0;
	gpt_t gpt;

	setprogname(argv[0]);

	if (strcmp(getprogname(), "gpt") == 0) {
		if (argc < 3)
			usage();
		dev = argv[--argc];
	}

#ifdef __GLIBC__
#define GETOPT_BE_POSIX		"+"
#else
#define GETOPT_BE_POSIX		""
#endif

	/* Get the generic options */
	while ((ch = getopt(argc, argv, GETOPT_BE_POSIX "Hm:nqrs:T:v")) != -1) {
		switch(ch) {
		case 'H':
			flags |= GPT_HYBRID;
			break;
		case 'm':
			if (mediasz > 0)
				usage();
			mediasz = strtol(optarg, &p, 10);
			if (*p != 0 || mediasz < 1)
				usage();
			break;
		case 'n':
			flags |= GPT_NOSYNC;
			break;
		case 'r':
			flags |= GPT_READONLY;
			break;
		case 'q':
			flags |= GPT_QUIET;
			break;
		case 's':
			if (gpt_uint_get(NULL, &secsz) == -1)
				usage();
			break;
		case 'T':
			flags |= GPT_TIMESTAMP;
			timestamp = get_tstamp(optarg);
			break;
		case 'v':
			verbose++;
			break;
		default:
			usage();
		}
	}

	if (argc == optind)
		usage();

	if (dev == NULL)
		dev = argv[optind++];

	if (argc == optind)
		usage();

	cmd = argv[optind++];
	for (i = 0; cmdsw[i]->name != NULL && strcmp(cmd, cmdsw[i]->name); i++)
		continue;

	if (cmdsw[i]->fptr == NULL)
		errx(EXIT_FAILURE, "Unknown command: %s", cmd);

	prefix(cmd);

	if (*dev != '-') {
		gpt = gpt_open(dev, flags | cmdsw[i]->flags,
		    verbose, mediasz, secsz, timestamp);
		if (gpt == NULL)
			return EXIT_FAILURE;
	} else {
		if ((cmdsw[i]->flags & GPT_OPTDEV) == 0)
			errx(EXIT_FAILURE,
			     "Command %s needs a device parameter", cmd);
		argc++;
		gpt = NULL;
	}

	if ((*cmdsw[i]->fptr)(gpt, argc, argv) == -1)
		return EXIT_FAILURE;

	if (gpt)
		gpt_close(gpt);
	return EXIT_SUCCESS;
}