12118f387SNathan Whitehorn /*- 22118f387SNathan Whitehorn * Copyright (c) 2011 Nathan Whitehorn 32118f387SNathan Whitehorn * All rights reserved. 42118f387SNathan Whitehorn * 52118f387SNathan Whitehorn * Redistribution and use in source and binary forms, with or without 62118f387SNathan Whitehorn * modification, are permitted provided that the following conditions 72118f387SNathan Whitehorn * are met: 82118f387SNathan Whitehorn * 1. Redistributions of source code must retain the above copyright 92118f387SNathan Whitehorn * notice, this list of conditions and the following disclaimer. 102118f387SNathan Whitehorn * 2. Redistributions in binary form must reproduce the above copyright 112118f387SNathan Whitehorn * notice, this list of conditions and the following disclaimer in the 122118f387SNathan Whitehorn * documentation and/or other materials provided with the distribution. 132118f387SNathan Whitehorn * 142118f387SNathan Whitehorn * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 152118f387SNathan Whitehorn * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 162118f387SNathan Whitehorn * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 172118f387SNathan Whitehorn * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 182118f387SNathan Whitehorn * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 192118f387SNathan Whitehorn * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 202118f387SNathan Whitehorn * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 212118f387SNathan Whitehorn * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 222118f387SNathan Whitehorn * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 232118f387SNathan Whitehorn * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 242118f387SNathan Whitehorn * SUCH DAMAGE. 252118f387SNathan Whitehorn * 262118f387SNathan Whitehorn * $FreeBSD$ 272118f387SNathan Whitehorn */ 282118f387SNathan Whitehorn 292118f387SNathan Whitehorn #include <sys/param.h> 302118f387SNathan Whitehorn #include <errno.h> 312118f387SNathan Whitehorn #include <libutil.h> 322118f387SNathan Whitehorn #include <inttypes.h> 332118f387SNathan Whitehorn 342118f387SNathan Whitehorn #include <libgeom.h> 352118f387SNathan Whitehorn #include <dialog.h> 362118f387SNathan Whitehorn #include <dlg_keys.h> 372118f387SNathan Whitehorn 382118f387SNathan Whitehorn #include "partedit.h" 392118f387SNathan Whitehorn 402118f387SNathan Whitehorn #define MIN_FREE_SPACE (1024*1024*1024) /* 1 GB */ 412118f387SNathan Whitehorn #define SWAP_SIZE(available) MIN(available/20, 4*1024*1024*1024LL) 422118f387SNathan Whitehorn 432118f387SNathan Whitehorn static char *boot_disk(struct gmesh *mesh); 442118f387SNathan Whitehorn static char *wizard_partition(struct gmesh *mesh, const char *disk); 452118f387SNathan Whitehorn 462118f387SNathan Whitehorn int 472118f387SNathan Whitehorn part_wizard(void) { 482118f387SNathan Whitehorn int error; 492118f387SNathan Whitehorn struct gmesh mesh; 502118f387SNathan Whitehorn char *disk, *schemeroot; 512118f387SNathan Whitehorn 522118f387SNathan Whitehorn startwizard: 532118f387SNathan Whitehorn error = geom_gettree(&mesh); 542118f387SNathan Whitehorn 552118f387SNathan Whitehorn dlg_put_backtitle(); 562118f387SNathan Whitehorn error = geom_gettree(&mesh); 572118f387SNathan Whitehorn disk = boot_disk(&mesh); 582118f387SNathan Whitehorn if (disk == NULL) 592118f387SNathan Whitehorn return (1); 602118f387SNathan Whitehorn 612118f387SNathan Whitehorn dlg_clear(); 622118f387SNathan Whitehorn dlg_put_backtitle(); 632118f387SNathan Whitehorn schemeroot = wizard_partition(&mesh, disk); 642118f387SNathan Whitehorn free(disk); 652118f387SNathan Whitehorn if (schemeroot == NULL) 662118f387SNathan Whitehorn return (1); 672118f387SNathan Whitehorn 682118f387SNathan Whitehorn geom_deletetree(&mesh); 692118f387SNathan Whitehorn dlg_clear(); 702118f387SNathan Whitehorn dlg_put_backtitle(); 712118f387SNathan Whitehorn error = geom_gettree(&mesh); 722118f387SNathan Whitehorn 73*a6b612e9SNathan Whitehorn error = wizard_makeparts(&mesh, schemeroot, 1); 742118f387SNathan Whitehorn if (error) 752118f387SNathan Whitehorn goto startwizard; 762118f387SNathan Whitehorn free(schemeroot); 772118f387SNathan Whitehorn 782118f387SNathan Whitehorn geom_deletetree(&mesh); 792118f387SNathan Whitehorn 802118f387SNathan Whitehorn return (0); 812118f387SNathan Whitehorn } 822118f387SNathan Whitehorn 832118f387SNathan Whitehorn static char * 842118f387SNathan Whitehorn boot_disk(struct gmesh *mesh) 852118f387SNathan Whitehorn { 862118f387SNathan Whitehorn struct gclass *classp; 872118f387SNathan Whitehorn struct gconfig *gc; 882118f387SNathan Whitehorn struct ggeom *gp; 892118f387SNathan Whitehorn struct gprovider *pp; 902118f387SNathan Whitehorn DIALOG_LISTITEM *disks = NULL; 919636f84bSNathan Whitehorn const char *type, *desc; 922118f387SNathan Whitehorn char diskdesc[512]; 932118f387SNathan Whitehorn char *chosen; 942118f387SNathan Whitehorn int i, err, selected, n = 0; 952118f387SNathan Whitehorn 962118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) { 972118f387SNathan Whitehorn if (strcmp(classp->lg_name, "DISK") != 0 && 98bcc25b7eSNathan Whitehorn strcmp(classp->lg_name, "RAID") != 0 && 992118f387SNathan Whitehorn strcmp(classp->lg_name, "MD") != 0) 1002118f387SNathan Whitehorn continue; 1012118f387SNathan Whitehorn 1022118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) { 1032118f387SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider)) 1042118f387SNathan Whitehorn continue; 1052118f387SNathan Whitehorn 1062118f387SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider) { 1079636f84bSNathan Whitehorn desc = type = NULL; 1089636f84bSNathan Whitehorn LIST_FOREACH(gc, &pp->lg_config, lg_config) { 1092118f387SNathan Whitehorn if (strcmp(gc->lg_name, "type") == 0) 1102118f387SNathan Whitehorn type = gc->lg_val; 1119636f84bSNathan Whitehorn if (strcmp(gc->lg_name, "descr") == 0) 1129636f84bSNathan Whitehorn desc = gc->lg_val; 1139636f84bSNathan Whitehorn } 1142118f387SNathan Whitehorn 1159636f84bSNathan Whitehorn /* Skip swap-backed md and WORM devices */ 1162118f387SNathan Whitehorn if (strcmp(classp->lg_name, "MD") == 0 && 1172118f387SNathan Whitehorn type != NULL && strcmp(type, "swap") == 0) 1182118f387SNathan Whitehorn continue; 1199636f84bSNathan Whitehorn if (strncmp(pp->lg_name, "cd", 2) == 0) 1209636f84bSNathan Whitehorn continue; 1212118f387SNathan Whitehorn 1222118f387SNathan Whitehorn disks = realloc(disks, (++n)*sizeof(disks[0])); 1232118f387SNathan Whitehorn disks[n-1].name = pp->lg_name; 1242118f387SNathan Whitehorn humanize_number(diskdesc, 7, pp->lg_mediasize, 1252118f387SNathan Whitehorn "B", HN_AUTOSCALE, HN_DECIMAL); 1262118f387SNathan Whitehorn if (strncmp(pp->lg_name, "ad", 2) == 0) 1272118f387SNathan Whitehorn strcat(diskdesc, " ATA Hard Disk"); 1282118f387SNathan Whitehorn else if (strncmp(pp->lg_name, "md", 2) == 0) 1292118f387SNathan Whitehorn strcat(diskdesc, " Memory Disk"); 1309636f84bSNathan Whitehorn else 1319636f84bSNathan Whitehorn strcat(diskdesc, " Disk"); 1329636f84bSNathan Whitehorn 1339636f84bSNathan Whitehorn if (desc != NULL) 1349636f84bSNathan Whitehorn snprintf(diskdesc, sizeof(diskdesc), 1359636f84bSNathan Whitehorn "%s <%s>", diskdesc, desc); 1369636f84bSNathan Whitehorn 1372118f387SNathan Whitehorn disks[n-1].text = strdup(diskdesc); 1382118f387SNathan Whitehorn disks[n-1].help = NULL; 1392118f387SNathan Whitehorn disks[n-1].state = 0; 1402118f387SNathan Whitehorn } 1412118f387SNathan Whitehorn } 1422118f387SNathan Whitehorn } 1432118f387SNathan Whitehorn 1442118f387SNathan Whitehorn if (n > 1) { 1452118f387SNathan Whitehorn err = dlg_menu("Partitioning", 1462118f387SNathan Whitehorn "Select the disk on which to install FreeBSD.", 0, 0, 0, 1472118f387SNathan Whitehorn n, disks, &selected, NULL); 1482118f387SNathan Whitehorn 1492118f387SNathan Whitehorn chosen = (err == 0) ? strdup(disks[selected].name) : NULL; 1502118f387SNathan Whitehorn } else if (n == 1) { 1512118f387SNathan Whitehorn chosen = strdup(disks[0].name); 1522118f387SNathan Whitehorn } else { 1532118f387SNathan Whitehorn chosen = NULL; 1542118f387SNathan Whitehorn } 1552118f387SNathan Whitehorn 1562118f387SNathan Whitehorn for (i = 0; i < n; i++) 1572118f387SNathan Whitehorn free(disks[i].text); 1582118f387SNathan Whitehorn 1592118f387SNathan Whitehorn return (chosen); 1602118f387SNathan Whitehorn } 1612118f387SNathan Whitehorn 1622118f387SNathan Whitehorn static struct gprovider * 1632118f387SNathan Whitehorn provider_for_name(struct gmesh *mesh, const char *name) 1642118f387SNathan Whitehorn { 1652118f387SNathan Whitehorn struct gclass *classp; 1662118f387SNathan Whitehorn struct gprovider *pp = NULL; 1672118f387SNathan Whitehorn struct ggeom *gp; 1682118f387SNathan Whitehorn 1692118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) { 1702118f387SNathan Whitehorn if (strcmp(classp->lg_name, "DISK") != 0 && 1712118f387SNathan Whitehorn strcmp(classp->lg_name, "PART") != 0 && 172bcc25b7eSNathan Whitehorn strcmp(classp->lg_name, "RAID") != 0 && 1732118f387SNathan Whitehorn strcmp(classp->lg_name, "MD") != 0) 1742118f387SNathan Whitehorn continue; 1752118f387SNathan Whitehorn 1762118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) { 1772118f387SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider)) 1782118f387SNathan Whitehorn continue; 1792118f387SNathan Whitehorn 1802118f387SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider) 1812118f387SNathan Whitehorn if (strcmp(pp->lg_name, name) == 0) 1822118f387SNathan Whitehorn break; 1832118f387SNathan Whitehorn 1842118f387SNathan Whitehorn if (pp != NULL) break; 1852118f387SNathan Whitehorn } 1862118f387SNathan Whitehorn 1872118f387SNathan Whitehorn if (pp != NULL) break; 1882118f387SNathan Whitehorn } 1892118f387SNathan Whitehorn 1902118f387SNathan Whitehorn return (pp); 1912118f387SNathan Whitehorn } 1922118f387SNathan Whitehorn 1932118f387SNathan Whitehorn static char * 1942118f387SNathan Whitehorn wizard_partition(struct gmesh *mesh, const char *disk) 1952118f387SNathan Whitehorn { 1962118f387SNathan Whitehorn struct gclass *classp; 1972118f387SNathan Whitehorn struct ggeom *gpart = NULL; 1982118f387SNathan Whitehorn struct gconfig *gc; 1992118f387SNathan Whitehorn char message[512]; 2002118f387SNathan Whitehorn const char *scheme = NULL; 2012118f387SNathan Whitehorn char *retval = NULL; 2022118f387SNathan Whitehorn int choice; 2032118f387SNathan Whitehorn 2042118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) 2052118f387SNathan Whitehorn if (strcmp(classp->lg_name, "PART") == 0) 2062118f387SNathan Whitehorn break; 2072118f387SNathan Whitehorn 2082118f387SNathan Whitehorn if (classp != NULL) { 2092118f387SNathan Whitehorn LIST_FOREACH(gpart, &classp->lg_geom, lg_geom) 2102118f387SNathan Whitehorn if (strcmp(gpart->lg_name, disk) == 0) 2112118f387SNathan Whitehorn break; 2122118f387SNathan Whitehorn } 2132118f387SNathan Whitehorn 2142118f387SNathan Whitehorn if (gpart != NULL) { 2152118f387SNathan Whitehorn LIST_FOREACH(gc, &gpart->lg_config, lg_config) { 2162118f387SNathan Whitehorn if (strcmp(gc->lg_name, "scheme") == 0) { 2172118f387SNathan Whitehorn scheme = gc->lg_val; 2182118f387SNathan Whitehorn break; 2192118f387SNathan Whitehorn } 2202118f387SNathan Whitehorn } 2212118f387SNathan Whitehorn } 2222118f387SNathan Whitehorn 223da42677dSNathan Whitehorn /* Treat uncommitted scheme deletions as no scheme */ 224da42677dSNathan Whitehorn if (scheme != NULL && strcmp(scheme, "(none)") == 0) 225da42677dSNathan Whitehorn scheme = NULL; 226da42677dSNathan Whitehorn 2272118f387SNathan Whitehorn query: 2282118f387SNathan Whitehorn dialog_vars.yes_label = "Entire Disk"; 2292118f387SNathan Whitehorn dialog_vars.no_label = "Partition"; 2302118f387SNathan Whitehorn if (gpart != NULL) 2312118f387SNathan Whitehorn dialog_vars.defaultno = TRUE; 2322118f387SNathan Whitehorn 2332118f387SNathan Whitehorn snprintf(message, sizeof(message), "Would you like to use this entire " 2342118f387SNathan Whitehorn "disk (%s) for FreeBSD or partition it to share it with other " 2352118f387SNathan Whitehorn "operating systems? Using the entire disk will erase any data " 2362118f387SNathan Whitehorn "currently stored there.", disk); 2372118f387SNathan Whitehorn choice = dialog_yesno("Partition", message, 0, 0); 2382118f387SNathan Whitehorn 2392118f387SNathan Whitehorn dialog_vars.yes_label = NULL; 2402118f387SNathan Whitehorn dialog_vars.no_label = NULL; 2412118f387SNathan Whitehorn dialog_vars.defaultno = FALSE; 2422118f387SNathan Whitehorn 2432118f387SNathan Whitehorn if (choice == 1 && scheme != NULL && !is_scheme_bootable(scheme)) { 2442118f387SNathan Whitehorn char warning[512]; 2452118f387SNathan Whitehorn int subchoice; 2462118f387SNathan Whitehorn 2472118f387SNathan Whitehorn sprintf(warning, "The existing partition scheme on this " 2482118f387SNathan Whitehorn "disk (%s) is not bootable on this platform. To install " 2492118f387SNathan Whitehorn "FreeBSD, it must be repartitioned. This will destroy all " 2502118f387SNathan Whitehorn "data on the disk. Are you sure you want to proceed?", 2512118f387SNathan Whitehorn scheme); 2522118f387SNathan Whitehorn subchoice = dialog_yesno("Non-bootable Disk", warning, 0, 0); 2532118f387SNathan Whitehorn if (subchoice != 0) 2542118f387SNathan Whitehorn goto query; 2552118f387SNathan Whitehorn 256f36a5e0fSNathan Whitehorn gpart_destroy(gpart); 2572118f387SNathan Whitehorn gpart_partition(disk, default_scheme()); 2582118f387SNathan Whitehorn scheme = default_scheme(); 2592118f387SNathan Whitehorn } 2602118f387SNathan Whitehorn 261da42677dSNathan Whitehorn if (scheme == NULL || choice == 0) { 262da42677dSNathan Whitehorn if (gpart != NULL && scheme != NULL) { 263da42677dSNathan Whitehorn /* Erase partitioned disk */ 2642118f387SNathan Whitehorn choice = dialog_yesno("Confirmation", "This will erase " 2652118f387SNathan Whitehorn "the disk. Are you sure you want to proceed?", 0, 0); 2662118f387SNathan Whitehorn if (choice != 0) 2672118f387SNathan Whitehorn goto query; 2682118f387SNathan Whitehorn 269f36a5e0fSNathan Whitehorn gpart_destroy(gpart); 2702118f387SNathan Whitehorn } 2712118f387SNathan Whitehorn 2722118f387SNathan Whitehorn gpart_partition(disk, default_scheme()); 2732118f387SNathan Whitehorn scheme = default_scheme(); 2742118f387SNathan Whitehorn } 2752118f387SNathan Whitehorn 2762118f387SNathan Whitehorn if (strcmp(scheme, "PC98") == 0 || strcmp(scheme, "MBR") == 0) { 2772118f387SNathan Whitehorn struct gmesh submesh; 2782118f387SNathan Whitehorn geom_gettree(&submesh); 2792118f387SNathan Whitehorn gpart_create(provider_for_name(&submesh, disk), 2802118f387SNathan Whitehorn "freebsd", NULL, NULL, &retval, 2812118f387SNathan Whitehorn choice /* Non-interactive for "Entire Disk" */); 2822118f387SNathan Whitehorn geom_deletetree(&submesh); 2832118f387SNathan Whitehorn } else { 2842118f387SNathan Whitehorn retval = strdup(disk); 2852118f387SNathan Whitehorn } 2862118f387SNathan Whitehorn 2872118f387SNathan Whitehorn return (retval); 2882118f387SNathan Whitehorn } 2892118f387SNathan Whitehorn 290*a6b612e9SNathan Whitehorn int 291*a6b612e9SNathan Whitehorn wizard_makeparts(struct gmesh *mesh, const char *disk, int interactive) 2922118f387SNathan Whitehorn { 2932118f387SNathan Whitehorn struct gmesh submesh; 2942118f387SNathan Whitehorn struct gclass *classp; 2952118f387SNathan Whitehorn struct ggeom *gp; 2962118f387SNathan Whitehorn struct gprovider *pp; 2972118f387SNathan Whitehorn intmax_t swapsize, available; 2982118f387SNathan Whitehorn char swapsizestr[10], rootsizestr[10]; 2992118f387SNathan Whitehorn int retval; 3002118f387SNathan Whitehorn 3012118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) 3022118f387SNathan Whitehorn if (strcmp(classp->lg_name, "PART") == 0) 3032118f387SNathan Whitehorn break; 3042118f387SNathan Whitehorn 3052118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) 3062118f387SNathan Whitehorn if (strcmp(gp->lg_name, disk) == 0) 3072118f387SNathan Whitehorn break; 3082118f387SNathan Whitehorn 3092118f387SNathan Whitehorn pp = provider_for_name(mesh, disk); 3102118f387SNathan Whitehorn 3112118f387SNathan Whitehorn available = gpart_max_free(gp, NULL)*pp->lg_sectorsize; 312*a6b612e9SNathan Whitehorn if (interactive && available < MIN_FREE_SPACE) { 3132118f387SNathan Whitehorn char availablestr[10], neededstr[10], message[512]; 3142118f387SNathan Whitehorn humanize_number(availablestr, 7, available, "B", HN_AUTOSCALE, 3152118f387SNathan Whitehorn HN_DECIMAL); 3162118f387SNathan Whitehorn humanize_number(neededstr, 7, MIN_FREE_SPACE, "B", HN_AUTOSCALE, 3172118f387SNathan Whitehorn HN_DECIMAL); 3182118f387SNathan Whitehorn sprintf(message, "There is not enough free space on %s to " 3192118f387SNathan Whitehorn "install FreeBSD (%s free, %s required). Would you like " 3202118f387SNathan Whitehorn "to choose another disk or to open the partition editor?", 3212118f387SNathan Whitehorn disk, availablestr, neededstr); 3222118f387SNathan Whitehorn 3232118f387SNathan Whitehorn dialog_vars.yes_label = "Another Disk"; 3242118f387SNathan Whitehorn dialog_vars.no_label = "Editor"; 3252118f387SNathan Whitehorn retval = dialog_yesno("Warning", message, 0, 0); 3262118f387SNathan Whitehorn dialog_vars.yes_label = NULL; 3272118f387SNathan Whitehorn dialog_vars.no_label = NULL; 3282118f387SNathan Whitehorn 3292118f387SNathan Whitehorn return (!retval); /* Editor -> return 0 */ 3302118f387SNathan Whitehorn } 3312118f387SNathan Whitehorn 3322118f387SNathan Whitehorn swapsize = SWAP_SIZE(available); 3332118f387SNathan Whitehorn humanize_number(swapsizestr, 7, swapsize, "B", HN_AUTOSCALE, 3342118f387SNathan Whitehorn HN_NOSPACE | HN_DECIMAL); 3352118f387SNathan Whitehorn humanize_number(rootsizestr, 7, available - swapsize - 1024*1024, 3362118f387SNathan Whitehorn "B", HN_AUTOSCALE, HN_NOSPACE | HN_DECIMAL); 3372118f387SNathan Whitehorn 3382118f387SNathan Whitehorn geom_gettree(&submesh); 3392118f387SNathan Whitehorn pp = provider_for_name(&submesh, disk); 3402118f387SNathan Whitehorn gpart_create(pp, "freebsd-ufs", rootsizestr, "/", NULL, 0); 3412118f387SNathan Whitehorn geom_deletetree(&submesh); 3422118f387SNathan Whitehorn 3432118f387SNathan Whitehorn geom_gettree(&submesh); 3442118f387SNathan Whitehorn pp = provider_for_name(&submesh, disk); 3452118f387SNathan Whitehorn gpart_create(pp, "freebsd-swap", swapsizestr, NULL, NULL, 0); 3462118f387SNathan Whitehorn geom_deletetree(&submesh); 3472118f387SNathan Whitehorn 3482118f387SNathan Whitehorn return (0); 3492118f387SNathan Whitehorn } 3502118f387SNathan Whitehorn 351