Plan 9 from Bell Labs’s /usr/web/sources/plan9/sys/src/cmd/bsplit.c

Copyright © 2021 Plan 9 Foundation.
Distributed under the MIT License.
Download the Plan 9 distribution.


/*
 * bsplit - split big binaries (copy-less plan 9 version)
 */

#include <u.h>
#include <libc.h>
#include <authsrv.h>		/* for ANAMELEN */

enum {
	Stdin,
	Sectsiz = 512,
	Bufsiz = 256*Sectsiz,
};

/* disk address (in bytes or sectors), also type of 2nd arg. to seek */
typedef uvlong Daddr;

#define BLEN(s)	((s)->wp - (s)->rp)
#define BALLOC(s) ((s)->lim - (s)->base)

typedef struct {
	uchar*	rp;		/* first unconsumed byte */
	uchar*	wp;		/* first empty byte */
	uchar*	lim;		/* 1 past the end of the buffer */
	uchar*	base;		/* start of the buffer */
} Buffer;

typedef struct {
	/* parameters */
	Daddr	maxoutsz;	/* maximum size of output file(s) */

	Daddr	fileout;	/* bytes written to the current output file */
	char	*filenm;
	int	filesz;		/* size of filenm */
	char	*prefix;
	long	filenum;
	int	outf;		/* open output file */

	Buffer	buff;
} Copy;

/* global data */
char *argv0;

/* private data */
static Copy cp = { 512*1024*1024 };	/* default maximum size */
static int debug;

static void
bufreset(Buffer *bp)
{
	bp->rp = bp->wp = bp->base;
}

static void
bufinit(Buffer *bp, uchar *block, unsigned size)
{
	bp->base = block;
	bp->lim = bp->base + size;
	bufreset(bp);
}

static int 
eopen(char *file, int mode)
{
	int fd = open(file, mode);

	if (fd < 0)
		sysfatal("can't open %s: %r", file);
	return fd;
}

static int 
ecreate(char *file, int mode)
{
	int fd = create(file, mode, 0666);

	if (fd < 0)
		sysfatal("can't create %s: %r", file);
	return fd;
}

static char *
filename(Copy *cp)
{
	return cp->filenm;
}

static int 
opennext(Copy *cp)
{
	if (cp->outf >= 0)
		sysfatal("opennext called with file open");
	snprint(cp->filenm, cp->filesz, "%s%5.5ld", cp->prefix, cp->filenum++);
	cp->outf = ecreate(cp->filenm, OWRITE);
	cp->fileout = 0;
	return cp->outf;
}

static int 
closeout(Copy *cp)
{
	if (cp->outf >= 0) {
		if (close(cp->outf) < 0)
			sysfatal("error writing %s: %r", filename(cp));
		cp->outf = -1;
		cp->fileout = 0;
	}
	return cp->outf;
}

/*
 * process - process input file
 */
static void
process(int in, char *inname)
{
	int n = 1;
	unsigned avail, tolim, wsz;
	Buffer *bp = &cp.buff;

	USED(inname);
	do {
		if (BLEN(bp) == 0) {
			if (bp->lim == bp->wp)
				bufreset(bp);
			n = read(in, bp->wp, bp->lim - bp->wp);
			if (n <= 0)
				break;
			bp->wp += n;
		}
		if (cp.outf < 0)
			opennext(&cp);

		/*
		 * write from buffer's current point to end or enough bytes to
		 * reach file-size limit.
		 */
		avail = BLEN(bp);
		tolim = cp.maxoutsz - cp.fileout;
		wsz = (tolim < avail? tolim: avail);

		/* try to write full sectors */
		if (tolim >= avail && n > 0 && wsz >= Sectsiz)
			wsz = (wsz / Sectsiz) * Sectsiz;
		if (write(cp.outf, bp->rp, wsz) != wsz)
			sysfatal("error writing %s: %r", filename(&cp));
		bp->rp += wsz;

		cp.fileout += wsz;
		if (cp.fileout >= cp.maxoutsz)
			closeout(&cp);
	} while (n > 0 || BLEN(bp) != 0);
}

static void
usage(void)
{
	fprint(2, "usage: %s [-d][-p pfx][-s size] [file...]\n", argv0);
	exits("usage");
}

void
main(int argc, char **argv)
{
	int i, errflg = 0;
	uchar block[Bufsiz];

	cp.prefix = "bs.";
	ARGBEGIN {
	case 'd':
		debug++;
		break;
	case 'p':
		cp.prefix = EARGF(usage());
		break;
	case 's':
		cp.maxoutsz = atoll(EARGF(usage()));
		if (cp.maxoutsz < 1)
			errflg++;
		break;
	default:
		errflg++;
		break;
	} ARGEND
	if (errflg || argc < 0)
		usage();

	bufinit(&cp.buff, block, sizeof block);
	cp.outf = -1;
	cp.filesz = strlen(cp.prefix) + 2*ANAMELEN;	/* 2* is slop */
	cp.filenm = malloc(cp.filesz + 1);
	if (cp.filenm == nil)
		sysfatal("no memory: %r");

	if (argc == 0)
		process(Stdin, "/fd/0");
	else
		for (i = 0; i < argc; i++) {
			int in = eopen(argv[i], OREAD);

			process(in, argv[i]);
			close(in);
		}
	closeout(&cp);
	exits(0);
}

Bell Labs OSI certified Powered by Plan 9

(Return to Plan 9 Home Page)

Copyright © 2021 Plan 9 Foundation. All Rights Reserved.
Comments to webmaster@9p.io.