/*
 * 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 2008 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

#include <sys/types.h>
#include <sys/time.h>
#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <values.h>
#include <locale.h>
#include <sys/stat.h>
#include <strings.h>
#include <stdarg.h>
#include <sys/param.h>
#include <nsctl.h>

#include <sys/nsctl/cfg.h>
#include <sys/unistat/spcs_s.h>
#include <sys/unistat/spcs_s_u.h>
#include <sys/unistat/spcs_errors.h>
#include <sys/nsctl/dsw.h>
#include <sys/nsctl/dsw_dev.h>		/* for bitmap format */

#define	DSW_TEXT_DOMAIN	"II"
#define	BITMAP_TOKEN	"ii.set%d.bitmap"
#define	SHADOW_TOKEN	"ii.set%d.shadow"
#define	SV_TOKEN	"sv.set%d.vol"
#define	DSVOL_TOKEN	"dsvol.set%d.path"

void iicpshd_usage();
void copyshd(char *, char *);
int find_cfg_info(char *, char *);
int copy_shadow_vol(char *, char *);
void convert_to_blockdevice();
int update_dscfg(char *);

extern int optind;

extern	char *optarg;
extern	int optind, opterr, optopt;
int	copy_shadow = 1;
CFGFILE	*cfg;
char	real_bitmap[DSW_NAMELEN];
char	buf[CFG_MAX_BUF];
char	key[CFG_MAX_KEY];
int	set_number;
int	sv_number;
int	dsvol_number;

#ifdef lint
int
iicpshd_lintmain(int argc, char *argv[])
#else
int
main(int argc, char *argv[])
#endif
{
	if (argc > 1) {
		if (strcmp(argv[1], "-s") == 0) {
			/* don't copy shadow, only update dscfg and ii header */
			copy_shadow = 0;
			argc--;
			argv++;
		}
	}

	if (argc == 1 || (argc%2) == 0)	/* must have pairs of filenames */
		iicpshd_usage();

	/* open dscfg anyway */
	if ((cfg = cfg_open(NULL)) == NULL) {
		fprintf(stderr, gettext("Error opening config\n"));
		exit(1);
	}

	for (argv++; *argv != NULL; argv += 2)
		copyshd(argv[0], argv[1]);

	/* close dscfg */
	cfg_close(cfg);
	exit(0);
	return (0);
}

void
iicpshd_usage()
{
	fprintf(stderr, gettext("Usage:\n"));
	fprintf(stderr, gettext("\tiicpshd [-s] old_shadow new_shadow\n"));
	exit(1);
}

void
copyshd(char *old_vol, char *new_vol)
{
	int dsw_fd;
	FILE *ifp;
	char header[FBA_SIZE(1) * DSW_CBLK_FBA];
	ii_header_t *hp;
	dsw_stat_t args;

	/*LINTED pointer alignment*/
	hp = (ii_header_t *)&header;

	dsw_fd = open(DSWDEV, O_RDONLY);
	if (dsw_fd < 0) {
		perror(DSWDEV);
		exit(1);
	}
	if (*old_vol != '/' || *new_vol != '/') {
		fprintf(stderr, gettext(
		"Both old and new shadow file names must begin with a /.\n"));
		exit(1);
	}

	if (strlen(new_vol) > DSW_NAMELEN) {
		fprintf(stderr, gettext("New shadow name is to long.\n"));
		exit(1);
	}

	/* check old shadow is in dscfg */
	if (find_cfg_info(old_vol, SHADOW_TOKEN) == 0) {
		fprintf(stderr, gettext("Old shadow not in existing cfg\n"));
		exit(1);
	}

	/* check ii set status, suspend if need */
	strncpy(args.shadow_vol, old_vol, DSW_NAMELEN);
	args.shadow_vol[DSW_NAMELEN-1] = '\0';
	args.status = spcs_s_ucreate();
	if (ioctl(dsw_fd, DSWIOC_STAT, &args) != -1) {
		fprintf(stderr, gettext("Suspend the Point-in-Time Copy "
		    "set first\n"));
		close(dsw_fd);
		exit(1);
	}

	if (copy_shadow) {
		if (copy_shadow_vol(old_vol, new_vol) == 0) {
			perror(gettext("Write new shadow failed"));
			close(dsw_fd);
			exit(1);
		}
	}
	if (find_cfg_info(old_vol, SV_TOKEN) == 0) {
		fprintf(stderr, gettext("Old shadow not in existing cfg\n"));
		exit(1);
	}
	if (find_cfg_info(old_vol, DSVOL_TOKEN) == 0) {
		fprintf(stderr, gettext("Old shadow not in existing cfg\n"));
		exit(1);
	}
	if (strstr(real_bitmap, "/rdsk/") == NULL) {
		fprintf(stderr,
			gettext("%s is not a character device\n"), real_bitmap);
		exit(1);
	}

	/* use block device /dsk/ to update bitmap header */
	convert_to_blockdevice();

	/* open bitmap by using update mode */
	if ((ifp = fopen(real_bitmap, "r+")) == NULL) {
		fprintf(stderr, gettext("Can't open bitmap file\n"));
		exit(1);
	}

	/* Check old header looks like an II bitmap header */
	if (fread(&header, DSW_CBLK_FBA, FBA_SIZE(1), ifp) != FBA_SIZE(1)) {
		fprintf(stderr, gettext("Can't read bitmap file\n"));
		exit(1);
	}

	if (hp->ii_magic != DSW_CLEAN && hp->ii_magic != DSW_DIRTY) {
		fprintf(stderr, gettext("%s is not an Point-in-Time Copy "
		    "shadow.\n"), old_vol);
		exit(1);
	}

	if (strncmp(hp->shadow_vol, old_vol, DSW_NAMELEN) != 0) {
		fprintf(stderr, gettext(
		"%s has Point-in-Time Copy shadow magic number,\n"
		"but does not contain correct data.\n"), old_vol);
		exit(1);
	}

	memset(hp->shadow_vol, 0, DSW_NAMELEN);
	strncpy(hp->shadow_vol, new_vol, DSW_NAMELEN);

	/* reset the pointer position */
	rewind(ifp);
	if (fwrite(&header, DSW_CBLK_FBA, FBA_SIZE(1), ifp) != FBA_SIZE(1)) {
		perror(new_vol);
		fprintf(stderr, gettext("Can't write new bitmap header\n"));
		exit(1);
	}
	fclose(ifp);
	close(dsw_fd);
	if (update_dscfg(new_vol) == 0) {
		fprintf(stderr, gettext("Failed to update dscfg.\n"));
		exit(1);
	} else {
		spcs_log("ii", NULL,
		"iicpshd copy shadow from %s to %s",
		old_vol, new_vol);
	}
}

/*
 * find_cfg_info()
 *
 */

int
find_cfg_info(char *volume, char *token)
{
	int i;
	/* get read lock */
	if (!cfg_lock(cfg, CFG_RDLOCK)) {
		spcs_log("ii", NULL,
			"iicpbmp CFG_RDLOCK failed, errno %d", errno);
		fprintf(stderr, gettext("Error locking config\n"));
		exit(1);
	}
	for (i = 1; ; i++) {
		bzero(buf, CFG_MAX_BUF);
		snprintf(key, sizeof (key), token, i);
		if (cfg_get_cstring(cfg, key, buf, DSW_NAMELEN) < 0) {
			cfg_unlock(cfg);
			return (0);
		}
		if (strcmp(buf, volume) == 0) {
			if (strcmp(token, SHADOW_TOKEN) == 0) {
				snprintf(key, sizeof (key), BITMAP_TOKEN, i);
				cfg_get_cstring
					(cfg, key, real_bitmap, DSW_NAMELEN);
				set_number = i;
			} else if (strcmp(token, SV_TOKEN) == 0) {
				sv_number = i;
			} else if (strcmp(token, DSVOL_TOKEN) == 0) {
				dsvol_number = i;
			}
			/* release read lock */
			cfg_unlock(cfg);
			return (1);
		}
	}
}

int
copy_shadow_vol(char *old_shadow, char *new_shadow) {
	int i;
	char cp_buffer[256];
	FILE *ishdfp, *oshdfp;
	if ((ishdfp = fopen(old_shadow, "r")) == NULL) {
		fprintf(stderr, gettext("Can't open old shadow file\n"));
		return (0);
	}
	if ((oshdfp = fopen(new_shadow, "w")) == NULL) {
		fprintf(stderr, gettext("Can't open new shadow file\n"));
		return (0);
	}

	/* Copy the shadow vol */
	while ((i = fread(cp_buffer, sizeof (char), sizeof (cp_buffer), ishdfp))
		> 0) {
		if (fwrite(cp_buffer, sizeof (char), i, oshdfp) != i) {
			fclose(ishdfp);
			fclose(oshdfp);
			return (0);
		}
	}
	fclose(ishdfp);
	fclose(oshdfp);
	return (1);
}

int
update_dscfg(char *new_shadow) {

	int len = strlen(new_shadow);
	/* get write lock */
	if (!cfg_lock(cfg, CFG_WRLOCK)) {
		spcs_log("ii", NULL,
			"iicpbmp CFG_WRLOCK failed, errno %d", errno);
		fprintf(stderr, gettext("Error locking config\n"));
		return (0);
	}
	sprintf(key, SHADOW_TOKEN, set_number);
	if (cfg_put_cstring(cfg, key, new_shadow, len) < 0) {
		perror("cfg_put_cstring");
		return (0);
	}
	sprintf(key, SV_TOKEN, sv_number);
	if (cfg_put_cstring(cfg, key, new_shadow, len) < 0) {
		perror("cfg_put_cstring");
		return (0);
	}
	sprintf(key, DSVOL_TOKEN, dsvol_number);
	if (cfg_put_cstring(cfg, key, new_shadow, len) < 0) {
		perror("cfg_put_cstring");
		return (0);
	}
	cfg_commit(cfg);
	cfg_unlock(cfg);
	return (1);
}

void
convert_to_blockdevice() {
	int len = strlen(real_bitmap);
	int i = 0, j = 0;
	char *temp_string = malloc(len-1);
	while (i < len + 1) {
		if (real_bitmap[i] != 'r') {
			temp_string[j] = real_bitmap[i];
			j++;
		}
		i++;
	}
	strcpy(real_bitmap, temp_string);
	free(temp_string);
}