jehanne/sys/src/cmd/disk/prep/prep.c

544 lines
12 KiB
C

/*
* This file is part of the UCB release of Plan 9. It is subject to the license
* terms in the LICENSE file found in the top-level directory of this
* distribution and at http://akaros.cs.berkeley.edu/files/Plan9License. No
* part of the UCB release of Plan 9, including this file, may be copied,
* modified, propagated, or distributed except according to the terms contained
* in the LICENSE file.
*/
/* Portions of this file are Copyright (C) 2015-2018 Giacomo Tesio <giacomo@tesio.it>
* See /doc/license/gpl-2.0.txt for details about the licensing.
*/
/* Portions of this file are Copyright (C) 9front's team.
* See /doc/license/9front-mit for details about the licensing.
* See http://git.9front.org/plan9front/plan9front/HEAD/info.html for a list of authors.
*/
/*
* prep - prepare plan9 disk partition
*/
#include <u.h>
#include <libc.h>
#include <bio.h>
#include <disk.h>
#include "edit.h"
enum {
Maxpath = 128,
};
static int blank;
static int file;
static int doautox;
static int printflag;
static Part **opart;
static int nopart;
static char *osecbuf;
static char *secbuf;
static int rdonly;
static int dowrite;
static int docache;
static int donvram;
static void autoxpart(Edit*);
static Part *mkpart(char*, int64_t, int64_t, int);
static void rdpart(Edit*);
static void wrpart(Edit*);
static void checkfat(Disk*);
static void cmdsum(Edit*, Part*, int64_t, int64_t);
static char *cmdadd(Edit*, char*, int64_t, int64_t);
static char *cmddel(Edit*, Part*);
static char *cmdokname(Edit*, char*);
static char *cmdwrite(Edit*);
Edit edit = {
.add= cmdadd,
.del= cmddel,
.okname=cmdokname,
.sum= cmdsum,
.write= cmdwrite,
.unit= "sector",
};
typedef struct Auto Auto;
struct Auto
{
char *name;
uint64_t min;
uint64_t max;
uint32_t weight;
uint8_t alloc;
uint64_t size;
};
#define TB (1024LL*GB)
#define GB (1024*1024*1024)
#define MB (1024*1024)
#define KB (1024)
/*
* Order matters -- this is the layout order on disk.
*/
Auto autox[] =
{
{ "9fat", 10*MB, 100*MB, 10, },
{ "nvram", 512, 512, 1, },
{ "fscfg", 512, 512, 1, },
{ "fs", 200*MB, 0, 10, },
{ "fossil", 200*MB, 0, 4, },
{ "arenas", 500*MB, 0, 20, },
{ "isect", 25*MB, 0, 1, },
{ "bloom", 4*MB, 512*MB, 1, },
{ "other", 200*MB, 0, 4, },
{ "swap", 100*MB, 512*MB, 1, },
{ "cache", 50*MB, 1*GB, 2, },
{ "fscache", 200*MB, 0, 4, },
{ "fsworm", 500*MB, 0, 20, },
};
void
usage(void)
{
jehanne_fprint(2, "usage: disk/prep [-bcfprw] [-a partname]... [-s sectorsize] /dev/sdC0/plan9\n");
jehanne_exits("usage");
}
void
main(int argc, char **argv)
{
int i;
char *p;
Disk *disk;
int64_t secsize;
secsize = 0;
ARGBEGIN{
case 'a':
p = EARGF(usage());
for(i=0; i<nelem(autox); i++){
if(jehanne_strcmp(p, autox[i].name) == 0){
if(autox[i].alloc){
jehanne_fprint(2, "you said -a %s more than once.\n", p);
usage();
}
autox[i].alloc = 1;
break;
}
}
if(i == nelem(autox)){
jehanne_fprint(2, "don't know how to create automatic partition %s\n", p);
usage();
}
doautox = 1;
break;
case 'b':
blank++;
break;
case 'c':
docache++;
break;
case 'f':
file++;
break;
case 'n':
donvram++;
break;
case 'p':
printflag++;
rdonly++;
break;
case 'r':
rdonly++;
break;
case 's':
secsize = jehanne_atoi(ARGF());
break;
case 'w':
dowrite++;
break;
default:
usage();
}ARGEND;
if(argc != 1)
usage();
disk = opendisk(argv[0], rdonly, file);
if(disk == nil)
jehanne_sysfatal("cannot open disk: %r");
if(secsize != 0) {
disk->secsize = secsize;
disk->secs = disk->size / secsize;
}
edit.unitsz = disk->secsize;
edit.end = disk->secs;
checkfat(disk);
secbuf = emalloc(disk->secsize+1);
osecbuf = emalloc(disk->secsize+1);
edit.disk = disk;
if(blank == 0)
rdpart(&edit);
opart = emalloc(edit.npart*sizeof(opart[0]));
/* save old partition table */
for(i=0; i<edit.npart; i++)
opart[i] = edit.part[i];
nopart = edit.npart;
if(printflag) {
runcmd(&edit, (char[]){"P"});
jehanne_exits(0);
}
if(doautox)
autoxpart(&edit);
if(dowrite) {
runcmd(&edit, (char[]){"w"});
jehanne_exits(0);
}
runcmd(&edit, (char[]){"p"});
for(;;) {
jehanne_fprint(2, ">>> ");
runcmd(&edit, getline(&edit));
}
}
static void
cmdsum(Edit *edit, Part *p, int64_t a, int64_t b)
{
int64_t sz, div;
char *suf, *name;
char c;
c = p && p->changed ? '\'' : ' ';
name = p ? p->name : "empty";
sz = (b-a)*edit->disk->secsize;
if(sz >= 1*TB){
suf = "TB";
div = TB;
}else if(sz >= 1*GB){
suf = "GB";
div = GB;
}else if(sz >= 1*MB){
suf = "MB";
div = MB;
}else if(sz >= 1*KB){
suf = "KB";
div = KB;
}else{
if (sz < 0)
jehanne_fprint(2, "%s: negative size!\n", argv0);
suf = "B ";
div = 1;
}
if(div == 1)
jehanne_print("%c %-12s %*lld %-*lld (%lld sectors, %lld %s)\n", c, name,
edit->disk->width, a, edit->disk->width, b, b-a, sz, suf);
else
jehanne_print("%c %-12s %*lld %-*lld (%lld sectors, %lld.%.2d %s)\n", c, name,
edit->disk->width, a, edit->disk->width, b, b-a,
sz/div, (int)(((sz%div)*100)/div), suf);
}
static char*
cmdadd(Edit *edit, char *name, int64_t start, int64_t end)
{
if(start < 2 && jehanne_strcmp(name, "9fat") != 0)
return "overlaps with the pbs and/or the partition table";
return addpart(edit, mkpart(name, start, end, 1));
}
static char*
cmddel(Edit *edit, Part *p)
{
return delpart(edit, p);
}
static char*
cmdwrite(Edit *edit)
{
wrpart(edit);
return nil;
}
static char isfrog[256]={
/*NUL*/ 1, 1, 1, 1, 1, 1, 1, 1,
/*BKS*/ 1, 1, 1, 1, 1, 1, 1, 1,
/*DLE*/ 1, 1, 1, 1, 1, 1, 1, 1,
/*CAN*/ 1, 1, 1, 1, 1, 1, 1, 1,
[' '] 1,
['/'] 1,
[0x7f] 1,
};
static char*
cmdokname(Edit* _, char *elem)
{
for(; *elem; elem++)
if(isfrog[*(uint8_t*)elem])
return "bad character in name";
return nil;
}
static Part*
mkpart(char *name, int64_t start, int64_t end, int changed)
{
Part *p;
p = emalloc(sizeof(*p));
p->name = estrdup(name);
p->ctlname = estrdup(name);
p->start = start;
p->end = end;
p->changed = changed;
return p;
}
/* plan9 partition is first sector of the disk */
static void
rdpart(Edit *edit)
{
int i, nline, nf, waserr;
int64_t a, b;
char *line[128];
char *f[5];
char *err;
Disk *disk;
disk = edit->disk;
sys_seek(disk->fd, disk->secsize, 0);
if(jehanne_readn(disk->fd, osecbuf, disk->secsize) != disk->secsize)
return;
osecbuf[disk->secsize] = '\0';
jehanne_memmove(secbuf, osecbuf, disk->secsize+1);
if(jehanne_strncmp(secbuf, "part", 4) != 0){
jehanne_fprint(2, "no plan9 partition table found\n");
return;
}
waserr = 0;
nline = jehanne_getfields(secbuf, line, nelem(line), 1, "\n");
for(i=0; i<nline; i++){
if(jehanne_strncmp(line[i], "part", 4) != 0) {
Error:
if(waserr == 0)
jehanne_fprint(2, "syntax error reading partition\n");
waserr = 1;
continue;
}
nf = jehanne_getfields(line[i], f, nelem(f), 1, " \t\r");
if(nf != 4 || jehanne_strcmp(f[0], "part") != 0)
goto Error;
a = jehanne_strtoll(f[2], 0, 0);
b = jehanne_strtoll(f[3], 0, 0);
if(a >= b)
goto Error;
if(err = addpart(edit, mkpart(f[1], a, b, 0))) {
jehanne_fprint(2, "?%s: not continuing\n", err);
jehanne_exits("partition");
}
}
}
static void
autoxpart(Edit *edit)
{
int i, totw, futz;
int64_t secs, secsize, psecsize, s, e, pa;
int32_t stride;
char *err;
if(edit->npart > 0) {
if(doautox)
jehanne_fprint(2, "partitions already exist; not repartitioning\n");
return;
}
secs = edit->disk->secs;
secsize = edit->disk->secsize;
psecsize = edit->disk->psecsize;
stride = psecsize / secsize;
pa = (edit->disk->offset - edit->disk->physalign + stride) % stride;
secs -= (secs + pa) % stride;
for(;;){
/* compute total weights */
totw = 0;
for(i=0; i<nelem(autox); i++){
if(autox[i].alloc==0 || autox[i].size)
continue;
totw += autox[i].weight;
}
if(totw == 0)
break;
if(secs <= 0){
jehanne_fprint(2, "ran out of disk space during autoxpartition.\n");
return;
}
/* assign any minimums for small disks */
futz = 0;
for(i=0; i<nelem(autox); i++){
if(autox[i].alloc==0 || autox[i].size)
continue;
s = (secs*autox[i].weight)/totw;
if(s < autox[i].min/secsize){
autox[i].size = autox[i].min/secsize;
secs -= autox[i].size;
futz = 1;
break;
}
}
if(futz)
continue;
/* assign any maximums for big disks */
futz = 0;
for(i=0; i<nelem(autox); i++){
if(autox[i].alloc==0 || autox[i].size)
continue;
s = (secs*autox[i].weight)/totw;
if(autox[i].max && s > autox[i].max/secsize){
autox[i].size = autox[i].max/secsize;
secs -= autox[i].size;
futz = 1;
break;
}
}
if(futz)
continue;
/* finally, assign partition sizes according to weights */
for(i=0; i<nelem(autox); i++){
if(autox[i].alloc==0 || autox[i].size)
continue;
s = (secs*autox[i].weight)/totw;
autox[i].size = s;
/* use entire disk even in face of rounding errors */
secs -= autox[i].size;
totw -= autox[i].weight;
}
}
s = 0;
if(autox[0].alloc == 0){
/* if no 9fat, reserve space for plan9 partition table */
s = 2;
if((s + pa) % stride) s += stride - (s + pa) % stride;
}
secs = edit->disk->secs;
for(i=0; i<nelem(autox); i++){
if(autox[i].alloc == 0)
continue;
e = (s + autox[i].size);
if((e + pa) % stride) e += stride - (e + pa) % stride;
if(e>secs) e = secs - stride + (secs + pa) % stride;
jehanne_print("%s %llud\n", autox[i].name, e - s);
if(err = addpart(edit, mkpart(autox[i].name, s, e, 1)))
jehanne_fprint(2, "addpart %s: %s\n", autox[i].name, err);
s = e;
}
}
static void
restore(Edit *edit, int ctlfd)
{
int i;
int64_t offset;
offset = edit->disk->offset;
jehanne_fprint(2, "attempting to restore partitions to previous state\n");
if(sys_seek(edit->disk->wfd, edit->disk->secsize, 0) != 0){
jehanne_fprint(2, "cannot restore: error seeking on disk\n");
jehanne_exits("inconsistent");
}
if(jehanne_write(edit->disk->wfd, osecbuf, edit->disk->secsize) != edit->disk->secsize){
jehanne_fprint(2, "cannot restore: couldn't write old partition table to disk\n");
jehanne_exits("inconsistent");
}
if(ctlfd >= 0){
for(i=0; i<edit->npart; i++)
jehanne_fprint(ctlfd, "delpart %s", edit->part[i]->name);
for(i=0; i<nopart; i++){
if(jehanne_fprint(ctlfd, "part %s %lld %lld", opart[i]->name, opart[i]->start+offset, opart[i]->end+offset) < 0){
jehanne_fprint(2, "restored disk partition table but not kernel; reboot\n");
jehanne_exits("inconsistent");
}
}
}
jehanne_exits("restored");
}
static void
wrpart(Edit *edit)
{
int i, n;
Disk *disk;
disk = edit->disk;
jehanne_memset(secbuf, 0, disk->secsize);
n = 0;
for(i=0; i<edit->npart; i++)
n += jehanne_snprint(secbuf+n, disk->secsize-n, "part %s %lld %lld\n",
edit->part[i]->name, edit->part[i]->start, edit->part[i]->end);
if(sys_seek(disk->wfd, disk->secsize, 0) != disk->secsize){
jehanne_fprint(2, "error seeking %d %lld on disk: %r\n", disk->wfd, disk->secsize);
jehanne_exits("seek");
}
if(jehanne_write(disk->wfd, secbuf, disk->secsize) != disk->secsize){
jehanne_fprint(2, "error writing partition table to disk\n");
restore(edit, -1);
}
if(ctldiff(edit, disk->ctlfd) < 0)
jehanne_fprint(2, "?warning: partitions could not be updated in devsd\n");
}
/*
* Look for a boot sector in sector 1, as would be
* the case if editing /dev/sdC0/data when that
* was really a bootable disk.
*/
static void
checkfat(Disk *disk)
{
uint8_t buf[32];
if(sys_seek(disk->fd, disk->secsize, 0) < 0
|| jehanne_read(disk->fd, buf, sizeof(buf)) < sizeof(buf))
return;
if(buf[0] != 0xEB || buf[1] != 0x3C || buf[2] != 0x90)
return;
jehanne_fprint(2,
"there's a fat partition where the\n"
"plan9 partition table would go.\n"
"if you really want to overwrite it, zero\n"
"the second sector of the disk and try again\n");
jehanne_exits("fat partition");
}