⛏️ index : haiku.git

/*
 * Copyright 2020, FranΓ§ois Revol, revol@free.fr.
 * Distributed under the terms of the MIT License.
 */

#include <fcntl.h>
#include <stddef.h>
#include <stdio.h>
#include <stdint.h>
#include <unistd.h>
#include <arpa/inet.h>

#include <disklabel.h>

/* for now we actually write all of the disk label here. */

#define DL_SIZE (offsetof(struct disk_label, dl_un.DL_v3_checksum) \
	+ sizeof(uint16_t))
#define SUM_CNT (offsetof(struct disk_label, dl_un.DL_v3_checksum) \
	/ sizeof(uint16_t))

#define H2B32 htonl
#define H2B16 htons
#define B2H16 ntohs


// see https://unix.superglobalmegacorp.com/darwin01/newsrc/machdep/i386/checksum_16.c.html
static uint16_t checksum_16(uint16_t *p, int count)
{
	uint32_t sum = 0;
	for (;count--;)
		sum += B2H16(*p++);
	// sum both shorts
	sum = (sum & 0x0ffff) + (sum >> 16);
	if (sum > 65535)
		sum -= 65535;
	return sum;
}


int main(int argc, char **argv)
{
	int fd;
	uint16_t sum;
	int labelOffsets[] = { 0, 15, 30, 45 };
	// HACK: for now we force a single label plus the bootlock at 8kb
	// (min offset the ROM allows) to fit before the tgz
	int numLabels = 1; // sizeof(labelOffsets) / sizeof(int);
	uint32_t bootBlockStart = 0x8; // usually 0x20

	struct disk_label disklabel = {
		H2B32(DL_V3),
		H2B32(0),
		H2B32(0),
		"HaikuBoot",//"NextBoot",
		H2B32(0),
		// dl_tag, we use 'HAIK'
		H2B32(0x4841494b), // H2B32(0xa991637a),
		"Sony MPX-111N 2880-512", // same as NS image, not sure it matters
		"removable_rw_floppy",
		H2B32(1024), // !! 1024 bytes / sector !!
		H2B32(2),
		H2B32(9),
		H2B32(80),
		H2B32(300),
		H2B16(96),
		H2B16(0),
		H2B16(0),
		H2B16(0),
		H2B16(0),
		H2B16(0),
		// boot blocks in 1024 bytes sectors
		H2B32(bootBlockStart),H2B32(0xffffffff),
		"haiku_loader", // "fdmach"
		"schredder", // "silly"
		'a', 'b',
		// partitions
		{
			// Nextstep uses this:
			//{ H2B32(0), H2B32(0x540), H2B16(0x2000), H2B16(0x400), 't',
			//	H2B16(0x20), H2B16(0x800), 0, 1, "", 1, "4.3BSD"},
			// XXX: should we fake an fs anyway?
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""},
			{ H2B32(-1), H2B32(-1), H2B16(-1), H2B16(-1), 0,
				H2B16(-1), H2B16(-1), -1, 0, "", 0, ""}
		},
		{ 0 },
		0
	};
	fd = open(argv[1], O_RDWR);
	if (fd < 0) {
		perror("open");
		return 1;
	}
	//XXX: support simple checksum of existing label?
#if 0
	if (read(fd, bootblock, DL_SIZE) < DL_SIZE) {
		perror("read");
		return 1;
	}
#endif

	// TODO: take boot block offsets as arg?

	sum = checksum_16((uint16_t *)&disklabel, SUM_CNT);
	fprintf(stderr, "checksum: 0x%04x\n", sum);
	disklabel.dl_un.DL_v3_checksum = H2B16(sum);

	for (unsigned int i = 0; i < numLabels; i++) {
		/* also write copies elsewhere, note we don't update the checksum */
		disklabel.dl_label_blkno = H2B32(labelOffsets[i]);
		/* oddly this field seems to use 512 bytes sectors */
		lseek(fd, labelOffsets[i] * 0x200LL, SEEK_SET);
		write(fd, &disklabel, DL_SIZE);
	}

	// TODO: patch the bootblock text segment to include the tgz

	return 0;
}