/*	$NetBSD: disklabel_swap.c,v 1.1 2021/05/17 08:50:36 mrg Exp $	*/

/*
 * Copyright (c) 1982, 1986, 1988 Regents of the University of California.
 * 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.
 * 3. Neither the name of the University nor the names of its contributors
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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.
 *
 *	@(#)ufs_disksubr.c	7.16 (Berkeley) 5/4/91
 */

#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: disklabel_swap.c,v 1.1 2021/05/17 08:50:36 mrg Exp $");

#ifdef _KERNEL_OPT
#include "opt_disklabel.h"
#endif /* _KERNEL_OPT */

#if defined(DISKLABEL_EI) || defined(LIBSA_DISKLABEL_EI)

#include <sys/param.h>
#include <sys/systm.h>
#include <sys/disklabel.h>
#include <sys/conf.h>
#include <lib/libkern/libkern.h>

/*
 * from sh3/disksubr.c and kern/subr_disk_mbr.c with modifications:
 *	- update d_checksum properly
 *	- replace memcpy(9) by memmove(9) as a precaution
 *	- avoid memmove(9) for libkern version, check if the labels
 *	  are the same and skip copying in-place.
 */
void
disklabel_swap(struct disklabel *nlp, struct disklabel *olp)
{
	int i;
	uint16_t npartitions;

#define	SWAP16(x)	nlp->x = bswap16(olp->x)
#define	SWAP32(x)	nlp->x = bswap32(olp->x)

	SWAP32(d_magic);
	SWAP16(d_type);
	SWAP16(d_subtype);
	if (nlp != olp) {
		/* Do not need to swap char strings. */
		memcpy(nlp->d_typename, olp->d_typename,
			sizeof(nlp->d_typename));

		/*
		 * XXX What should we do for d_un (an union of char and
		 * pointers)?
		 */
		memcpy(nlp->d_packname, olp->d_packname,
			sizeof(nlp->d_packname));
	}

	SWAP32(d_secsize);
	SWAP32(d_nsectors);
	SWAP32(d_ntracks);
	SWAP32(d_ncylinders);
	SWAP32(d_secpercyl);
	SWAP32(d_secperunit);

	SWAP16(d_sparespertrack);
	SWAP16(d_sparespercyl);

	SWAP32(d_acylinders);

	SWAP16(d_rpm);
	SWAP16(d_interleave);
	SWAP16(d_trackskew);
	SWAP16(d_cylskew);
	SWAP32(d_headswitch);
	SWAP32(d_trkseek);
	SWAP32(d_flags);
	for (i = 0; i < NDDATA; i++)
		SWAP32(d_drivedata[i]);
	for (i = 0; i < NSPARE; i++)
		SWAP32(d_spare[i]);
	SWAP32(d_magic2);
	/* d_checksum is updated later. */

	SWAP16(d_npartitions);
	SWAP32(d_bbsize);
	SWAP32(d_sbsize);
	for (i = 0; i < MAXPARTITIONS; i++) {
		SWAP32(d_partitions[i].p_size);
		SWAP32(d_partitions[i].p_offset);
		SWAP32(d_partitions[i].p_fsize);
		/* p_fstype and p_frag is uint8_t, so no need to swap. */
		nlp->d_partitions[i].p_fstype = olp->d_partitions[i].p_fstype;
		nlp->d_partitions[i].p_frag = olp->d_partitions[i].p_frag;
		SWAP16(d_partitions[i].p_cpg);
	}

#undef SWAP16
#undef SWAP32

	/* Update checksum in the target endian. */
	nlp->d_checksum = 0;
	npartitions = nlp->d_magic == DISKMAGIC ?
	    nlp->d_npartitions : olp->d_npartitions;
	/*
	 * npartitions can be larger than MAXPARTITIONS when the label was not
	 * validated by setdisklabel. If so, the label is intentionally(?)
	 * corrupted and checksum should be meaningless.
	 */
	if (npartitions <= MAXPARTITIONS)
		nlp->d_checksum = dkcksum_sized(nlp, npartitions);
}
#endif /* DISKLABEL_EI || LIBSA_DISKLABEL_EI */