diff options
-rw-r--r-- | sbin/disklabel/editor.c | 156 |
1 files changed, 47 insertions, 109 deletions
diff --git a/sbin/disklabel/editor.c b/sbin/disklabel/editor.c index b57dc0fe140..f38f18bac5a 100644 --- a/sbin/disklabel/editor.c +++ b/sbin/disklabel/editor.c @@ -1,4 +1,4 @@ -/* $OpenBSD: editor.c,v 1.143 2008/01/06 22:28:13 krw Exp $ */ +/* $OpenBSD: editor.c,v 1.144 2008/01/07 16:51:35 krw Exp $ */ /* * Copyright (c) 1997-2000 Todd C. Miller <Todd.Miller@courtesan.com> @@ -17,7 +17,7 @@ */ #ifndef lint -static char rcsid[] = "$OpenBSD: editor.c,v 1.143 2008/01/06 22:28:13 krw Exp $"; +static char rcsid[] = "$OpenBSD: editor.c,v 1.144 2008/01/07 16:51:35 krw Exp $"; #endif /* not lint */ #include <sys/types.h> @@ -72,7 +72,6 @@ void editor_name(struct disklabel *, char **, char *); char *getstring(char *, char *, char *); u_int64_t getuint(struct disklabel *, int, char *, char *, u_int64_t, u_int64_t, u_int64_t, int); int has_overlap(struct disklabel *, u_int64_t *, int); -u_int64_t next_offset(struct disklabel *, u_int64_t *); int partition_cmp(const void *, const void *); struct partition **sort_partitions(struct disklabel *, u_int16_t *); void getdisktype(struct disklabel *, char *, char *); @@ -87,7 +86,7 @@ int get_bsize(struct disklabel *, int); int get_fsize(struct disklabel *, int); int get_fstype(struct disklabel *, int); int get_mp(struct disklabel *, char **, int); -int get_offset(struct disklabel *, int); +int get_offset(struct disklabel *, int, struct diskchunk *); int get_size(struct disklabel *, int, u_int64_t *, int); void get_geometry(int, struct disklabel **); void set_geometry(struct disklabel *, struct disklabel *, struct disklabel *, @@ -496,8 +495,20 @@ editor_add(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) /* Set defaults */ memset(pp, 0, sizeof(*pp)); - new_size = *freep; - new_offset = next_offset(lp, &new_size); + chunks = free_chunks(lp); + + /* + * Since we know there's free space, there must be at least one + * chunk. So find the largest chunk and assume we want to add the + * partition in that free space. + */ + new_size = new_offset = 0; + for (i = 0; chunks[i].start != 0 || chunks[i].stop != 0; i++) { + if (chunks[i].stop - chunks[i].start > new_size) { + new_size = chunks[i].stop - chunks[i].start; + new_offset = chunks[i].start; + } + } DL_SETPSIZE(pp, new_size); DL_SETPOFFSET(pp, new_offset); pp->p_fstype = partno == 1 ? FS_SWAP : FS_BSDFFS; @@ -511,24 +522,11 @@ editor_add(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) pp->p_cpg = 1; /* Get offset */ - if (get_offset(lp, partno) != 0) { + if (get_offset(lp, partno, chunks) != 0) { DL_SETPSIZE(pp, 0); /* effective delete */ return; } - /* Recompute recommended size based on new offset */ - ui = pp->p_fstype; - pp->p_fstype = FS_UNUSED; - chunks = free_chunks(lp); - for (i = 0; chunks[i].start != 0 || chunks[i].stop != 0; i++) { - if (DL_GETPOFFSET(pp) >= chunks[i].start && - DL_GETPOFFSET(pp) < chunks[i].stop) { - DL_SETPSIZE(pp, chunks[i].stop - DL_GETPOFFSET(pp)); - break; - } - } - pp->p_fstype = ui; - /* Get size */ if (get_size(lp, partno, freep, 1) != 0 || DL_GETPSIZE(pp) == 0) { DL_SETPSIZE(pp, 0); /* effective delete */ @@ -600,7 +598,8 @@ void editor_modify(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) { struct partition origpart, *pp; - int partno; + struct diskchunk *chunks; + int partno, i; /* Change which partition? */ if (p == NULL) { @@ -626,27 +625,13 @@ editor_modify(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) origpart = *pp; - /* Get filesystem type */ - if (get_fstype(lp, partno) != 0) { - *pp = origpart; /* undo changes */ - return; - } - - /* Ensure a newly enabled partition fits. */ - if (pp->p_fstype != FS_UNUSED && pp->p_fstype != FS_BOOT && - (origpart.p_fstype == FS_UNUSED || origpart.p_fstype == FS_BOOT)) { - if (DL_GETPSIZE(pp) > *freep) { - fprintf(stderr, - "Warning, need %llu sectors but there are only %llu " - "free. Setting size to %llu.\n", DL_GETPSIZE(pp), *freep, - *freep); - DL_SETPSIZE(pp, *freep); - } - } + pp->p_fstype = FS_UNUSED; + chunks = free_chunks(lp); editor_countfree(lp, freep); + pp->p_fstype = origpart.p_fstype; /* Get offset */ - if (get_offset(lp, partno) != 0) { + if (get_offset(lp, partno, chunks) != 0) { *pp = origpart; /* undo changes */ return; } @@ -657,6 +642,12 @@ editor_modify(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) return; } + /* Get filesystem type */ + if (get_fstype(lp, partno) != 0) { + *pp = origpart; /* undo changes */ + return; + } + /* get mount point */ if (get_mp(lp, mp, partno) != 0) { *pp = origpart; /* undo changes */ @@ -724,70 +715,6 @@ editor_delete(struct disklabel *lp, char **mp, u_int64_t *freep, char *p) } /* - * Find the next reasonable starting offset and returns it. - * Assumes there is a least one free sector left (returns 0 if not). - */ -u_int64_t -next_offset(struct disklabel *lp, u_int64_t *sizep) -{ - struct partition **spp; - struct diskchunk *chunks; - u_int16_t npartitions; - u_int64_t new_offset, new_size; - int i, good_offset; - - /* Get a sorted list of the partitions */ - if ((spp = sort_partitions(lp, &npartitions)) == NULL) - return(starting_sector); - - new_offset = starting_sector; - for (i = 0; i < npartitions; i++ ) { - u_int64_t pstart = DL_GETPOFFSET(spp[i]); - u_int64_t pend = pstart + DL_GETPSIZE(spp[i]); - u_int64_t newend = new_offset + *sizep; - - /* - * Is new_offset inside this partition? If so, - * make it the next sector after the partition ends. - */ - if (pend < ending_sector && - ((new_offset >= pstart && new_offset < pend) || - (newend > pstart && newend <= pend))) - new_offset = pend; - } - - /* Did we find a suitable offset? */ - for (good_offset = 1, i = 0; i < npartitions; i++ ) { - u_int64_t pstart = DL_GETPOFFSET(spp[i]); - u_int64_t pend = pstart + DL_GETPSIZE(spp[i]); - u_int64_t newend = new_offset + *sizep; - - if (newend > pstart && newend <= pend) { - /* Nope */ - good_offset = 0; - break; - } - } - - /* Specified size is too big, find something that fits */ - if (!good_offset) { - chunks = free_chunks(lp); - new_size = 0; - for (i = 0; chunks[i].start != 0 || chunks[i].stop != 0; i++) { - if (chunks[i].stop - chunks[i].start > new_size) { - new_size = chunks[i].stop - chunks[i].start; - new_offset = chunks[i].start; - } - } - /* XXX - should do something intelligent if new_size == 0 */ - *sizep = new_size; - } - - (void)free(spp); - return(new_offset); -} - -/* * Change the size of an existing partition. */ void @@ -1818,10 +1745,11 @@ mpsave(struct disklabel *lp, char **mp, char *cdev, char *fstabfile) } int -get_offset(struct disklabel *lp, int partno) +get_offset(struct disklabel *lp, int partno, struct diskchunk *chunks) { struct partition *pp = &lp->d_partitions[partno]; - u_int64_t ui; + u_int64_t ui, maxsize; + int i; for (;;) { ui = getuint(lp, partno, "offset", @@ -1845,11 +1773,21 @@ get_offset(struct disklabel *lp, int partno) fprintf(stderr, "This architecture requires that " "partition 'a' start at sector 0.\n"); #endif - else - break; + else { + for (i = 0; chunks[i].start != 0 || chunks[i].stop != 0; + i++) { + if (ui < chunks[i].start || + ui >= chunks[i].stop) + continue; + DL_SETPOFFSET(pp, ui); + maxsize = chunks[i].stop - DL_GETPOFFSET(pp); + if (DL_GETPSIZE(pp) > maxsize) + DL_SETPSIZE(pp, maxsize); + return (0); + } + fputs("The offset must be in a free area.\n", stderr); + } } - DL_SETPOFFSET(pp, ui); - return(0); } int |