/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*
 * This file contains all the functions that manipualte the file
 * system where the GRUB menu resides.
 */
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <strings.h>
#include <unistd.h>
#include <fcntl.h>
#include <assert.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mount.h>
#include <sys/mntent.h>
#include <sys/mnttab.h>
#include <sys/fs/ufs_mount.h>
#include <sys/dktp/fdisk.h>
#include <libfstyp.h>

#include "libgrub_impl.h"

static int
slice_match(const char *physpath, int slice)
{
	const char *pos;

	return ((pos = strrchr(physpath, slice)) == NULL ||
	    pos[1] != 0 || pos[-1] != ':');
}

/*
 * Returns zero if path contains ufs
 */
static int
slice_ufs(const char *path)
{
	int fd, ret;
	const char *id;
	fstyp_handle_t hdl;

	fd = open(path, O_RDONLY);
	if ((ret = fstyp_init(fd, 0, NULL, &hdl)) == 0) {
		ret = fstyp_ident(hdl, "ufs", &id);
		fstyp_fini(hdl);
	}
	(void) close(fd);
	return (ret);
}


static int
get_sol_prtnum(const char *physpath)
{
	int i, fd;
	char *pos;
	size_t sz;
	struct mboot *mb;
	struct ipart *ipart;
	char boot_sect[512];
	char rdev[MAXNAMELEN];

	(void) snprintf(rdev, sizeof (rdev), "/devices%s,raw", physpath);

	if ((pos = strrchr(rdev, ':')) == NULL)
		return (PRTNUM_INVALID);

	pos[1] = SLCNUM_WHOLE_DISK;

	fd = open(rdev, O_RDONLY);
	sz = read(fd, boot_sect, sizeof (boot_sect));
	(void) close(fd);

	if (sz != sizeof (boot_sect))
		return (PRTNUM_INVALID);

	/* parse fdisk table */
	mb = (struct mboot *)(uintptr_t)boot_sect;
	ipart = (struct ipart *)(uintptr_t)mb->parts;
	for (i = 0; i < FD_NUMPART; ++i) {
		if (ipart[i].systid == SUNIXOS || ipart[i].systid == SUNIXOS2)
			return (i);
	}
	return (PRTNUM_INVALID);
}

/*
 * Get physpath, topfs and bootfs for ZFS root dataset.
 * Return 0 on success, non-zero (not errno) on failure.
 */
static int
get_zfs_root(zfs_handle_t *zfh, grub_fs_t *fs, grub_root_t *root)
{
	int ret;
	zpool_handle_t *zph;
	const char *name;

	if (zfs_get_type(zfh) != ZFS_TYPE_FILESYSTEM ||
	    (name = zfs_get_name(zfh)) == NULL ||
	    (zph = zpool_open(fs->gf_lzfh, name)) == NULL)
		return (-1);

	if ((ret = zpool_get_physpath(zph, root->gr_physpath,
	    sizeof (root->gr_physpath))) == 0 &&
	    (ret = zpool_get_prop(zph, ZPOOL_PROP_BOOTFS,
	    root->gr_fs[GRBM_ZFS_BOOTFS].gfs_dev,
	    sizeof (root->gr_fs[GRBM_ZFS_BOOTFS].gfs_dev), NULL)) == 0) {

		(void) strlcpy(root->gr_fs[GRBM_ZFS_TOPFS].gfs_dev, name,
		    sizeof (root->gr_fs[GRBM_ZFS_TOPFS].gfs_dev));
		(void) grub_fsd_get_mountp(root->gr_fs + GRBM_ZFS_BOOTFS,
		    MNTTYPE_ZFS);
		(void) grub_fsd_get_mountp(root->gr_fs + GRBM_ZFS_TOPFS,
		    MNTTYPE_ZFS);
	}

	zpool_close(zph);
	return (ret);
}

/*
 * On entry physpath parameter supposed to contain:
 * <disk_physpath>[<space><disk_physpath>]*.
 * Retireives first <disk_physpath> that matches both partition and slice.
 * If any partition and slice is acceptable, first <disk_physpath> is returned.
 */
static int
get_one_physpath(char *physpath, uint_t prtnum, uint_t slcnum)
{
	int ret;
	char *tmp, *tok;

	if (!IS_SLCNUM_VALID(slcnum) && !IS_PRTNUM_VALID(prtnum)) {
		(void) strtok(physpath, " ");
		return (0);
	}

	if ((tmp = strdup(physpath)) == NULL)
		return (errno);

	ret = ENODEV;
	for (tok = strtok(tmp, " "); tok != NULL; tok = strtok(NULL, " ")) {
		if ((ret = (slice_match(tok, slcnum) != 0 ||
		    get_sol_prtnum(tok) != prtnum)) == 0) {
			(void) strcpy(physpath, tok);
			break;
		}
	}

	free(tmp);
	return (ret);
}

static int
zfs_bootsign(zfs_handle_t *zfh, void *data)
{
	grub_barg_t *barg;
	grub_menu_t *menu;
	struct stat st;
	char path[MAXPATHLEN];

	barg = (grub_barg_t *)data;
	menu = barg->gb_entry->ge_menu;

	do {
		if (get_zfs_root(zfh, &menu->gm_fs, &barg->gb_root) != 0 ||
		    get_one_physpath(barg->gb_root.gr_physpath, barg->gb_prtnum,
		    barg->gb_slcnum) != 0)
			break;

		/*
		 * if top zfs dataset is not mounted, mount it now
		 */
		if (barg->gb_root.gr_fs[GRBM_ZFS_TOPFS].gfs_mountp[0] == 0) {
			if (grub_fsd_mount_tmp(barg->gb_root.gr_fs +
			    GRBM_ZFS_TOPFS, MNTTYPE_ZFS) != 0)
				break;
		}

		/* check that bootsign exists and it is a regular file */
		(void) snprintf(path, sizeof (path), "%s%s",
		    barg->gb_root.gr_fs[GRBM_ZFS_TOPFS].gfs_mountp,
		    barg->gb_bootsign);

		if (lstat(path, &st) != 0 || S_ISREG(st.st_mode) == 0 ||
		    (st.st_mode & S_IRUSR) == 0)
			break;

		(void) strlcpy(barg->gb_root.gr_fstyp, MNTTYPE_ZFS,
		    sizeof (barg->gb_root.gr_fstyp));
		barg->gb_walkret = 0;
	/* LINTED: E_CONSTANT_CONDITION */
	} while (0);

	grub_fsd_umount_tmp(barg->gb_root.gr_fs + GRBM_ZFS_TOPFS);
	zfs_close(zfh);

	/* return non-zero to terminate the walk */
	return (barg->gb_walkret == 0);
}

static int
get_devlink(di_devlink_t dl, void *arg)
{
	const char *path;
	grub_barg_t *barg;

	barg = (grub_barg_t *)arg;
	if ((path = di_devlink_path(dl)) != NULL)
		(void) strlcpy(barg->gb_root.gr_fs[GRBM_UFS].gfs_dev, path,
		    sizeof (barg->gb_root.gr_fs[GRBM_UFS].gfs_dev));
	return (DI_WALK_TERMINATE);
}

static int
ufs_bootsign_check(grub_barg_t *barg)
{
	int ret;
	struct stat st;
	grub_menu_t *mp;
	char path[MAXPATHLEN];

	mp = barg->gb_entry->ge_menu;

	/* get /dev/dsk link */
	if (di_devlink_walk(mp->gm_fs.gf_dvlh, "^dsk/",
	    barg->gb_root.gr_physpath, DI_PRIMARY_LINK, barg, get_devlink) != 0)
		return (errno);
	/*
	 * if disk is not mounted, mount it now
	 */
	if (grub_fsd_get_mountp(barg->gb_root.gr_fs + GRBM_UFS,
	    MNTTYPE_UFS) != 0) {
		if ((ret =
		    slice_ufs(barg->gb_root.gr_fs[GRBM_UFS].gfs_dev)) != 0 ||
		    (ret = grub_fsd_mount_tmp(barg->gb_root.gr_fs + GRBM_UFS,
		    MNTTYPE_UFS)) != 0)
			return (ret);
	}

	(void) snprintf(path, sizeof (path), "%s%s",
	    barg->gb_root.gr_fs[GRBM_UFS].gfs_mountp, barg->gb_bootsign);

	if (lstat(path, &st) == 0 && S_ISREG(st.st_mode) &&
	    (st.st_mode & S_IRUSR) != 0) {
		barg->gb_walkret = 0;
		(void) strlcpy(barg->gb_root.gr_fstyp, MNTTYPE_UFS,
		    sizeof (barg->gb_root.gr_fstyp));
	}

	grub_fsd_umount_tmp(barg->gb_root.gr_fs + GRBM_UFS);
	return (barg->gb_walkret);
}

static int
ufs_bootsign(di_node_t node, di_minor_t minor, void *arg)
{
	uint_t prtnum;
	char *name, *path;
	grub_barg_t *barg;

	barg = (grub_barg_t *)arg;

	if (di_minor_spectype(minor) != S_IFBLK)
		return (DI_WALK_CONTINUE);

	name = di_minor_name(minor);
	if (name[0] != barg->gb_slcnum || name[1] != 0)
		return (DI_WALK_CONTINUE);

	path = di_devfs_path(node);
	(void) snprintf(barg->gb_root.gr_physpath,
	    sizeof (barg->gb_root.gr_physpath), "%s:%c", path, barg->gb_slcnum);
	di_devfs_path_free(path);

	prtnum = get_sol_prtnum(barg->gb_root.gr_physpath);
	if (!IS_PRTNUM_VALID(prtnum))
		return (DI_WALK_CONTINUE);

	/*
	 * check only specified partition, slice
	 */

	if (IS_PRTNUM_VALID(barg->gb_prtnum)) {
		if (prtnum != barg->gb_prtnum || ufs_bootsign_check(barg) != 0)
			return (DI_WALK_CONTINUE);
		return (DI_WALK_TERMINATE);
	}

	/*
	 * Walk through all slices in found solaris partition
	 */

	barg->gb_prtnum = prtnum;
	minor = DI_MINOR_NIL;

	while ((minor = di_minor_next(node, minor)) != DI_MINOR_NIL) {

		if (di_minor_spectype(minor) != S_IFBLK)
			continue;

		name = di_minor_name(minor);
		if (!IS_SLCNUM_VALID(name[0]) || name[1] != 0)
			continue;

		barg->gb_slcnum = name[0];
		path = strrchr(barg->gb_root.gr_physpath, ':');
		path[1] = barg->gb_slcnum;

		if (ufs_bootsign_check(barg) == 0)
			return (DI_WALK_TERMINATE);
	}

	barg->gb_prtnum = (uint_t)PRTNUM_INVALID;
	barg->gb_slcnum = (uint_t)SLCNUM_WHOLE_DISK;
	return (DI_WALK_CONTINUE);
}

/*
 * Differs from what GRUB is doing: GRUB searchs through all disks seen by bios
 * for bootsign, if bootsign is found on ufs slice GRUB sets it as a root,
 * if on zfs, then GRUB uses zfs slice as root only if bootsign wasn't found
 * on other slices.
 * That function first searches through all top datasets of active zpools,
 * then if bootsign still not found walks through all disks and tries to
 * find ufs slice with the bootsign.
 */
int
grub_find_bootsign(grub_barg_t *barg)
{
	grub_menu_t *mp;
	mp = barg->gb_entry->ge_menu;

	/* try to find bootsign over zfs pools */
	barg->gb_walkret = EG_BOOTSIGN;
	(void) zfs_iter_root(mp->gm_fs.gf_lzfh, zfs_bootsign, barg);

	/* try ufs now */
	if (barg->gb_walkret != 0 && di_walk_minor(mp->gm_fs.gf_diroot,
	    DDI_NT_BLOCK, 0, barg, ufs_bootsign) != 0)
		return (errno);

	return (barg->gb_walkret);
}

/*
 * Get current root file system.
 * Return 0 on success, errno code on failure.
 */
int
grub_current_root(grub_fs_t *fs, grub_root_t *root)
{
	int rc = 0;
	FILE *fp = NULL;
	char *name = NULL;
	zfs_handle_t *zfh = NULL;
	struct mnttab mp = {0};
	struct mnttab mpref = {0};
	char buf[MAXNAMELEN] = {0};

	mpref.mnt_mountp = "/";

	if ((fp = fopen(MNTTAB, "r")) == NULL)
		return (errno);

	/*
	 * getmntany returns non-zero for failure, and sets errno
	 */
	rc = getmntany(fp, &mp, &mpref);
	if (rc != 0)
		rc = errno;

	(void) fclose(fp);

	if (rc != 0)
		return (rc);

	(void) strlcpy(root->gr_fstyp, mp.mnt_fstype, sizeof (root->gr_fstyp));

	if (strcmp(root->gr_fstyp, MNTTYPE_ZFS) == 0) {

		(void) strlcpy(buf, mp.mnt_special, sizeof (buf));
		if ((name = strtok(buf, "/")) == NULL)
			return (EG_CURROOT);

		if ((zfh = zfs_open(fs->gf_lzfh, name, ZFS_TYPE_FILESYSTEM)) ==
		    NULL)
			return (EG_OPENZFS);

		/*
		 * get_zfs_root returns non-zero on failure, not
		 * errno.
		 */
		if (get_zfs_root(zfh, fs, root))
			rc = EG_CURROOT;

		zfs_close(zfh);

	} else if (strcmp(mp.mnt_fstype, MNTTYPE_UFS) == 0) {
		(void) strlcpy(root->gr_fs[GRBM_UFS].gfs_dev, mp.mnt_special,
		    sizeof (root->gr_fs[GRBM_UFS].gfs_dev));
		(void) strlcpy(root->gr_fs[GRBM_UFS].gfs_mountp, mp.mnt_mountp,
		    sizeof (root->gr_fs[GRBM_UFS].gfs_mountp));
	} else {
		rc = EG_UNKNOWNFS;
	}

	return (rc);
}

grub_fsdesc_t *
grub_get_rootfsd(const grub_root_t *root)
{
	grub_fsdesc_t *fsd = NULL;

	assert(root);
	if (strcmp(MNTTYPE_UFS, root->gr_fstyp) == 0)
		fsd = (grub_fsdesc_t *)root->gr_fs + GRBM_UFS;
	else if (strcmp(MNTTYPE_ZFS, root->gr_fstyp) == 0)
		fsd = (grub_fsdesc_t *)root->gr_fs + GRBM_ZFS_BOOTFS;

	return (fsd);
}

/*
 * Gets file systems mount point if any.
 * Return 0 if filesystem is mounted, errno on failure.
 */
int
grub_fsd_get_mountp(grub_fsdesc_t *fsd, char *fstyp)
{
	int rc;
	FILE *fp = NULL;
	struct mnttab mp = {0};
	struct mnttab mpref = {0};

	fsd->gfs_mountp[0] = 0;

	if ((fp = fopen(MNTTAB, "r")) == NULL)
		return (errno);

	mpref.mnt_special = fsd->gfs_dev;
	mpref.mnt_fstype = fstyp;

	if ((rc = getmntany(fp, &mp, &mpref)) == 0)
		(void) strlcpy(fsd->gfs_mountp, mp.mnt_mountp,
		    sizeof (fsd->gfs_mountp));
	else
		rc = EG_GETMNTTAB;

	(void) fclose(fp);
	return (rc);
}

static const char tmp_mountp[] = "/tmp/.libgrubmgmt.%s.XXXXXX";

/*
 * Mount file system at tmp_mountp.
 * Return 0 on success, errno on failure.
 */
int
grub_fsd_mount_tmp(grub_fsdesc_t *fsd, const char *fstyp)
{
	const char *pos;
	void *data = NULL;
	int dtsz = 0;
	struct ufs_args ufs_args = {UFSMNT_LARGEFILES};
	char mntopts[MNT_LINE_MAX] = "";
	int rc = 0;

	assert(fsd);
	assert(!fsd->gfs_is_tmp_mounted);

	fsd->gfs_mountp[0] = 0;

	if (strcmp(fstyp, MNTTYPE_UFS) == 0) {
		(void) strlcpy(mntopts, MNTOPT_LARGEFILES, sizeof (mntopts));
		data = &ufs_args;
		dtsz = sizeof (ufs_args);
	} else if (strcmp(fstyp, MNTTYPE_ZFS) != 0) {
		return (EG_UNKNOWNFS);
	}

	/* construct name for temporary mount point */
	pos = strrchr(fsd->gfs_dev, '/');
	pos = (pos == NULL) ? fsd->gfs_dev : pos + 1;

	(void) snprintf(fsd->gfs_mountp, sizeof (fsd->gfs_mountp),
	    tmp_mountp, pos);
	if (mkdtemp(fsd->gfs_mountp) != NULL) {
		if ((rc = mount(fsd->gfs_dev, fsd->gfs_mountp,
		    MS_DATA | MS_OPTIONSTR | MS_RDONLY,
		    fstyp, data, dtsz, mntopts, sizeof (mntopts))) != 0) {
			/*
			 * mount failed, collect errno and remove temp dir
			 */
			rc = errno;
			(void) rmdir(fsd->gfs_mountp);
		}
	} else {
		rc = errno;
	}

	if (rc != 0)
		fsd->gfs_mountp[0] = 0;

	/*
	 * Note that valid values for gfs_is_tmp_mounted are 0,1.
	 * Any other value indicates that something bad happened.
	 * Probably grub_fsd_umount_tmp() wasn't called or didn't
	 * work as expected.
	 */
	fsd->gfs_is_tmp_mounted += (rc == 0);
	return (rc);
}

/*
 * Unmount file system at tmp_mountp.
 */
void
grub_fsd_umount_tmp(grub_fsdesc_t *fsd)
{
	if (fsd == NULL)
		return;

	if (fsd->gfs_is_tmp_mounted) {
		if (fsd->gfs_mountp[0] != 0) {
			(void) umount2(fsd->gfs_mountp, 0);
			(void) rmdir(fsd->gfs_mountp);
			fsd->gfs_mountp[0] = 0;
		}
		fsd->gfs_is_tmp_mounted = 0;
	}
}