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 346e15678aSNathan Whitehorn #include <sys/sysctl.h> 356e15678aSNathan Whitehorn #include <string.h> 366e15678aSNathan Whitehorn 372118f387SNathan Whitehorn #include <libgeom.h> 382118f387SNathan Whitehorn #include <dialog.h> 392118f387SNathan Whitehorn #include <dlg_keys.h> 402118f387SNathan Whitehorn 412118f387SNathan Whitehorn #include "partedit.h" 422118f387SNathan Whitehorn 432118f387SNathan Whitehorn #define MIN_FREE_SPACE (1024*1024*1024) /* 1 GB */ 442118f387SNathan Whitehorn #define SWAP_SIZE(available) MIN(available/20, 4*1024*1024*1024LL) 452118f387SNathan Whitehorn 462118f387SNathan Whitehorn static char *boot_disk(struct gmesh *mesh); 472118f387SNathan Whitehorn static char *wizard_partition(struct gmesh *mesh, const char *disk); 482118f387SNathan Whitehorn 492118f387SNathan Whitehorn int 50*d547c56fSEnji Cooper part_wizard(const char *fsreq) 51*d547c56fSEnji Cooper { 522118f387SNathan Whitehorn char *disk, *schemeroot; 536e15678aSNathan Whitehorn const char *fstype; 54*d547c56fSEnji Cooper struct gmesh mesh; 55*d547c56fSEnji Cooper int error; 566e15678aSNathan Whitehorn 576e15678aSNathan Whitehorn if (fsreq != NULL) 586e15678aSNathan Whitehorn fstype = fsreq; 596e15678aSNathan Whitehorn else 606e15678aSNathan Whitehorn fstype = "ufs"; 612118f387SNathan Whitehorn 622118f387SNathan Whitehorn startwizard: 632118f387SNathan Whitehorn error = geom_gettree(&mesh); 642118f387SNathan Whitehorn 652118f387SNathan Whitehorn dlg_put_backtitle(); 662118f387SNathan Whitehorn error = geom_gettree(&mesh); 672118f387SNathan Whitehorn disk = boot_disk(&mesh); 682118f387SNathan Whitehorn if (disk == NULL) 692118f387SNathan Whitehorn return (1); 702118f387SNathan Whitehorn 712118f387SNathan Whitehorn dlg_clear(); 722118f387SNathan Whitehorn dlg_put_backtitle(); 732118f387SNathan Whitehorn schemeroot = wizard_partition(&mesh, disk); 742118f387SNathan Whitehorn free(disk); 752118f387SNathan Whitehorn if (schemeroot == NULL) 762118f387SNathan Whitehorn return (1); 772118f387SNathan Whitehorn 782118f387SNathan Whitehorn geom_deletetree(&mesh); 792118f387SNathan Whitehorn dlg_clear(); 802118f387SNathan Whitehorn dlg_put_backtitle(); 812118f387SNathan Whitehorn error = geom_gettree(&mesh); 822118f387SNathan Whitehorn 836e15678aSNathan Whitehorn error = wizard_makeparts(&mesh, schemeroot, fstype, 1); 842118f387SNathan Whitehorn if (error) 852118f387SNathan Whitehorn goto startwizard; 862118f387SNathan Whitehorn free(schemeroot); 872118f387SNathan Whitehorn 882118f387SNathan Whitehorn geom_deletetree(&mesh); 892118f387SNathan Whitehorn 902118f387SNathan Whitehorn return (0); 912118f387SNathan Whitehorn } 922118f387SNathan Whitehorn 932118f387SNathan Whitehorn static char * 942118f387SNathan Whitehorn boot_disk(struct gmesh *mesh) 952118f387SNathan Whitehorn { 962118f387SNathan Whitehorn struct gclass *classp; 972118f387SNathan Whitehorn struct gconfig *gc; 982118f387SNathan Whitehorn struct ggeom *gp; 992118f387SNathan Whitehorn struct gprovider *pp; 1002118f387SNathan Whitehorn DIALOG_LISTITEM *disks = NULL; 1019636f84bSNathan Whitehorn const char *type, *desc; 1022118f387SNathan Whitehorn char diskdesc[512]; 1032118f387SNathan Whitehorn char *chosen; 1042118f387SNathan Whitehorn int i, err, selected, n = 0; 1052118f387SNathan Whitehorn 1062118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) { 1072118f387SNathan Whitehorn if (strcmp(classp->lg_name, "DISK") != 0 && 108bcc25b7eSNathan Whitehorn strcmp(classp->lg_name, "RAID") != 0 && 1092118f387SNathan Whitehorn strcmp(classp->lg_name, "MD") != 0) 1102118f387SNathan Whitehorn continue; 1112118f387SNathan Whitehorn 1122118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) { 1132118f387SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider)) 1142118f387SNathan Whitehorn continue; 1152118f387SNathan Whitehorn 1162118f387SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider) { 1179636f84bSNathan Whitehorn desc = type = NULL; 1189636f84bSNathan Whitehorn LIST_FOREACH(gc, &pp->lg_config, lg_config) { 1192118f387SNathan Whitehorn if (strcmp(gc->lg_name, "type") == 0) 1202118f387SNathan Whitehorn type = gc->lg_val; 1219636f84bSNathan Whitehorn if (strcmp(gc->lg_name, "descr") == 0) 1229636f84bSNathan Whitehorn desc = gc->lg_val; 1239636f84bSNathan Whitehorn } 1242118f387SNathan Whitehorn 1259636f84bSNathan Whitehorn /* Skip swap-backed md and WORM devices */ 1262118f387SNathan Whitehorn if (strcmp(classp->lg_name, "MD") == 0 && 1272118f387SNathan Whitehorn type != NULL && strcmp(type, "swap") == 0) 1282118f387SNathan Whitehorn continue; 1299636f84bSNathan Whitehorn if (strncmp(pp->lg_name, "cd", 2) == 0) 1309636f84bSNathan Whitehorn continue; 1312118f387SNathan Whitehorn 1322118f387SNathan Whitehorn disks = realloc(disks, (++n)*sizeof(disks[0])); 1332118f387SNathan Whitehorn disks[n-1].name = pp->lg_name; 1342118f387SNathan Whitehorn humanize_number(diskdesc, 7, pp->lg_mediasize, 1352118f387SNathan Whitehorn "B", HN_AUTOSCALE, HN_DECIMAL); 1362118f387SNathan Whitehorn if (strncmp(pp->lg_name, "ad", 2) == 0) 1372118f387SNathan Whitehorn strcat(diskdesc, " ATA Hard Disk"); 1382118f387SNathan Whitehorn else if (strncmp(pp->lg_name, "md", 2) == 0) 1392118f387SNathan Whitehorn strcat(diskdesc, " Memory Disk"); 1409636f84bSNathan Whitehorn else 1419636f84bSNathan Whitehorn strcat(diskdesc, " Disk"); 1429636f84bSNathan Whitehorn 1439636f84bSNathan Whitehorn if (desc != NULL) 1449636f84bSNathan Whitehorn snprintf(diskdesc, sizeof(diskdesc), 1459636f84bSNathan Whitehorn "%s <%s>", diskdesc, desc); 1469636f84bSNathan Whitehorn 1472118f387SNathan Whitehorn disks[n-1].text = strdup(diskdesc); 1482118f387SNathan Whitehorn disks[n-1].help = NULL; 1492118f387SNathan Whitehorn disks[n-1].state = 0; 1502118f387SNathan Whitehorn } 1512118f387SNathan Whitehorn } 1522118f387SNathan Whitehorn } 1532118f387SNathan Whitehorn 1542118f387SNathan Whitehorn if (n > 1) { 1552118f387SNathan Whitehorn err = dlg_menu("Partitioning", 1562118f387SNathan Whitehorn "Select the disk on which to install FreeBSD.", 0, 0, 0, 1572118f387SNathan Whitehorn n, disks, &selected, NULL); 1582118f387SNathan Whitehorn 1592118f387SNathan Whitehorn chosen = (err == 0) ? strdup(disks[selected].name) : NULL; 1602118f387SNathan Whitehorn } else if (n == 1) { 1612118f387SNathan Whitehorn chosen = strdup(disks[0].name); 1622118f387SNathan Whitehorn } else { 1632118f387SNathan Whitehorn chosen = NULL; 1642118f387SNathan Whitehorn } 1652118f387SNathan Whitehorn 1662118f387SNathan Whitehorn for (i = 0; i < n; i++) 1672118f387SNathan Whitehorn free(disks[i].text); 1682118f387SNathan Whitehorn 1692118f387SNathan Whitehorn return (chosen); 1702118f387SNathan Whitehorn } 1712118f387SNathan Whitehorn 1722118f387SNathan Whitehorn static struct gprovider * 1732118f387SNathan Whitehorn provider_for_name(struct gmesh *mesh, const char *name) 1742118f387SNathan Whitehorn { 1752118f387SNathan Whitehorn struct gclass *classp; 1762118f387SNathan Whitehorn struct gprovider *pp = NULL; 1772118f387SNathan Whitehorn struct ggeom *gp; 1782118f387SNathan Whitehorn 1792118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) { 1802118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) { 1812118f387SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider)) 1822118f387SNathan Whitehorn continue; 1832118f387SNathan Whitehorn 1842118f387SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider) 1852118f387SNathan Whitehorn if (strcmp(pp->lg_name, name) == 0) 1862118f387SNathan Whitehorn break; 1872118f387SNathan Whitehorn 1882118f387SNathan Whitehorn if (pp != NULL) break; 1892118f387SNathan Whitehorn } 1902118f387SNathan Whitehorn 1912118f387SNathan Whitehorn if (pp != NULL) break; 1922118f387SNathan Whitehorn } 1932118f387SNathan Whitehorn 1942118f387SNathan Whitehorn return (pp); 1952118f387SNathan Whitehorn } 1962118f387SNathan Whitehorn 1972118f387SNathan Whitehorn static char * 1982118f387SNathan Whitehorn wizard_partition(struct gmesh *mesh, const char *disk) 1992118f387SNathan Whitehorn { 2002118f387SNathan Whitehorn struct gclass *classp; 2012118f387SNathan Whitehorn struct ggeom *gpart = NULL; 2022118f387SNathan Whitehorn struct gconfig *gc; 2032118f387SNathan Whitehorn char *retval = NULL; 204*d547c56fSEnji Cooper const char *scheme = NULL; 205*d547c56fSEnji Cooper char message[512]; 2062118f387SNathan Whitehorn int choice; 2072118f387SNathan Whitehorn 2082118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) 2092118f387SNathan Whitehorn if (strcmp(classp->lg_name, "PART") == 0) 2102118f387SNathan Whitehorn break; 2112118f387SNathan Whitehorn 2122118f387SNathan Whitehorn if (classp != NULL) { 2132118f387SNathan Whitehorn LIST_FOREACH(gpart, &classp->lg_geom, lg_geom) 2142118f387SNathan Whitehorn if (strcmp(gpart->lg_name, disk) == 0) 2152118f387SNathan Whitehorn break; 2162118f387SNathan Whitehorn } 2172118f387SNathan Whitehorn 2182118f387SNathan Whitehorn if (gpart != NULL) { 2192118f387SNathan Whitehorn LIST_FOREACH(gc, &gpart->lg_config, lg_config) { 2202118f387SNathan Whitehorn if (strcmp(gc->lg_name, "scheme") == 0) { 2212118f387SNathan Whitehorn scheme = gc->lg_val; 2222118f387SNathan Whitehorn break; 2232118f387SNathan Whitehorn } 2242118f387SNathan Whitehorn } 2252118f387SNathan Whitehorn } 2262118f387SNathan Whitehorn 227da42677dSNathan Whitehorn /* Treat uncommitted scheme deletions as no scheme */ 228da42677dSNathan Whitehorn if (scheme != NULL && strcmp(scheme, "(none)") == 0) 229da42677dSNathan Whitehorn scheme = NULL; 230da42677dSNathan Whitehorn 2312118f387SNathan Whitehorn query: 2322118f387SNathan Whitehorn dialog_vars.yes_label = "Entire Disk"; 2332118f387SNathan Whitehorn dialog_vars.no_label = "Partition"; 2342118f387SNathan Whitehorn if (gpart != NULL) 2352118f387SNathan Whitehorn dialog_vars.defaultno = TRUE; 2362118f387SNathan Whitehorn 2372118f387SNathan Whitehorn snprintf(message, sizeof(message), "Would you like to use this entire " 2382118f387SNathan Whitehorn "disk (%s) for FreeBSD or partition it to share it with other " 2392118f387SNathan Whitehorn "operating systems? Using the entire disk will erase any data " 2402118f387SNathan Whitehorn "currently stored there.", disk); 2412118f387SNathan Whitehorn choice = dialog_yesno("Partition", message, 0, 0); 2422118f387SNathan Whitehorn 2432118f387SNathan Whitehorn dialog_vars.yes_label = NULL; 2442118f387SNathan Whitehorn dialog_vars.no_label = NULL; 2452118f387SNathan Whitehorn dialog_vars.defaultno = FALSE; 2462118f387SNathan Whitehorn 2472118f387SNathan Whitehorn if (choice == 1 && scheme != NULL && !is_scheme_bootable(scheme)) { 2482118f387SNathan Whitehorn char warning[512]; 2492118f387SNathan Whitehorn int subchoice; 2502118f387SNathan Whitehorn 2512118f387SNathan Whitehorn sprintf(warning, "The existing partition scheme on this " 2522118f387SNathan Whitehorn "disk (%s) is not bootable on this platform. To install " 2532118f387SNathan Whitehorn "FreeBSD, it must be repartitioned. This will destroy all " 2542118f387SNathan Whitehorn "data on the disk. Are you sure you want to proceed?", 2552118f387SNathan Whitehorn scheme); 2562118f387SNathan Whitehorn subchoice = dialog_yesno("Non-bootable Disk", warning, 0, 0); 2572118f387SNathan Whitehorn if (subchoice != 0) 2582118f387SNathan Whitehorn goto query; 2592118f387SNathan Whitehorn 260f36a5e0fSNathan Whitehorn gpart_destroy(gpart); 2617059fa6fSAllan Jude scheme = choose_part_type(default_scheme()); 2627059fa6fSAllan Jude if (scheme == NULL) 2637059fa6fSAllan Jude return NULL; 2647059fa6fSAllan Jude gpart_partition(disk, scheme); 2652118f387SNathan Whitehorn } 2662118f387SNathan Whitehorn 267da42677dSNathan Whitehorn if (scheme == NULL || choice == 0) { 268da42677dSNathan Whitehorn if (gpart != NULL && scheme != NULL) { 269da42677dSNathan Whitehorn /* Erase partitioned disk */ 2702118f387SNathan Whitehorn choice = dialog_yesno("Confirmation", "This will erase " 2712118f387SNathan Whitehorn "the disk. Are you sure you want to proceed?", 0, 0); 2722118f387SNathan Whitehorn if (choice != 0) 2732118f387SNathan Whitehorn goto query; 2742118f387SNathan Whitehorn 275f36a5e0fSNathan Whitehorn gpart_destroy(gpart); 2762118f387SNathan Whitehorn } 2772118f387SNathan Whitehorn 2787059fa6fSAllan Jude scheme = choose_part_type(default_scheme()); 2797059fa6fSAllan Jude if (scheme == NULL) 2807059fa6fSAllan Jude return NULL; 2817059fa6fSAllan Jude gpart_partition(disk, scheme); 2822118f387SNathan Whitehorn } 2832118f387SNathan Whitehorn 2842b375b4eSYoshihiro Takahashi if (strcmp(scheme, "MBR") == 0) { 2852118f387SNathan Whitehorn struct gmesh submesh; 2862118f387SNathan Whitehorn geom_gettree(&submesh); 2872118f387SNathan Whitehorn gpart_create(provider_for_name(&submesh, disk), 2882118f387SNathan Whitehorn "freebsd", NULL, NULL, &retval, 2892118f387SNathan Whitehorn choice /* Non-interactive for "Entire Disk" */); 2902118f387SNathan Whitehorn geom_deletetree(&submesh); 2912118f387SNathan Whitehorn } else { 2922118f387SNathan Whitehorn retval = strdup(disk); 2932118f387SNathan Whitehorn } 2942118f387SNathan Whitehorn 2952118f387SNathan Whitehorn return (retval); 2962118f387SNathan Whitehorn } 2972118f387SNathan Whitehorn 298a6b612e9SNathan Whitehorn int 299*d547c56fSEnji Cooper wizard_makeparts(struct gmesh *mesh, const char *disk, const char *fstype, 300*d547c56fSEnji Cooper int interactive) 3012118f387SNathan Whitehorn { 3022118f387SNathan Whitehorn struct gclass *classp; 3032118f387SNathan Whitehorn struct ggeom *gp; 3042118f387SNathan Whitehorn struct gprovider *pp; 3056e15678aSNathan Whitehorn char *fsnames[] = {"freebsd-ufs", "freebsd-zfs"}; 306*d547c56fSEnji Cooper char *fsname; 307*d547c56fSEnji Cooper struct gmesh submesh; 308*d547c56fSEnji Cooper char swapsizestr[10], rootsizestr[10]; 309*d547c56fSEnji Cooper intmax_t swapsize, available; 3102118f387SNathan Whitehorn int retval; 3112118f387SNathan Whitehorn 3126e15678aSNathan Whitehorn if (strcmp(fstype, "zfs") == 0) { 3136e15678aSNathan Whitehorn fsname = fsnames[1]; 3146e15678aSNathan Whitehorn } else { 3156e15678aSNathan Whitehorn /* default to UFS */ 3166e15678aSNathan Whitehorn fsname = fsnames[0]; 3176e15678aSNathan Whitehorn } 3186e15678aSNathan Whitehorn 3192118f387SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) 3202118f387SNathan Whitehorn if (strcmp(classp->lg_name, "PART") == 0) 3212118f387SNathan Whitehorn break; 3222118f387SNathan Whitehorn 3232118f387SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) 3242118f387SNathan Whitehorn if (strcmp(gp->lg_name, disk) == 0) 3252118f387SNathan Whitehorn break; 3262118f387SNathan Whitehorn 3272118f387SNathan Whitehorn pp = provider_for_name(mesh, disk); 3282118f387SNathan Whitehorn 3292118f387SNathan Whitehorn available = gpart_max_free(gp, NULL)*pp->lg_sectorsize; 330a6b612e9SNathan Whitehorn if (interactive && available < MIN_FREE_SPACE) { 3312118f387SNathan Whitehorn char availablestr[10], neededstr[10], message[512]; 3322118f387SNathan Whitehorn humanize_number(availablestr, 7, available, "B", HN_AUTOSCALE, 3332118f387SNathan Whitehorn HN_DECIMAL); 3342118f387SNathan Whitehorn humanize_number(neededstr, 7, MIN_FREE_SPACE, "B", HN_AUTOSCALE, 3352118f387SNathan Whitehorn HN_DECIMAL); 3362118f387SNathan Whitehorn sprintf(message, "There is not enough free space on %s to " 3372118f387SNathan Whitehorn "install FreeBSD (%s free, %s required). Would you like " 3382118f387SNathan Whitehorn "to choose another disk or to open the partition editor?", 3392118f387SNathan Whitehorn disk, availablestr, neededstr); 3402118f387SNathan Whitehorn 3412118f387SNathan Whitehorn dialog_vars.yes_label = "Another Disk"; 3422118f387SNathan Whitehorn dialog_vars.no_label = "Editor"; 3432118f387SNathan Whitehorn retval = dialog_yesno("Warning", message, 0, 0); 3442118f387SNathan Whitehorn dialog_vars.yes_label = NULL; 3452118f387SNathan Whitehorn dialog_vars.no_label = NULL; 3462118f387SNathan Whitehorn 3472118f387SNathan Whitehorn return (!retval); /* Editor -> return 0 */ 3482118f387SNathan Whitehorn } 3492118f387SNathan Whitehorn 3502118f387SNathan Whitehorn swapsize = SWAP_SIZE(available); 3512118f387SNathan Whitehorn humanize_number(swapsizestr, 7, swapsize, "B", HN_AUTOSCALE, 3522118f387SNathan Whitehorn HN_NOSPACE | HN_DECIMAL); 3532118f387SNathan Whitehorn humanize_number(rootsizestr, 7, available - swapsize - 1024*1024, 3542118f387SNathan Whitehorn "B", HN_AUTOSCALE, HN_NOSPACE | HN_DECIMAL); 3552118f387SNathan Whitehorn 3562118f387SNathan Whitehorn geom_gettree(&submesh); 3572118f387SNathan Whitehorn pp = provider_for_name(&submesh, disk); 3586e15678aSNathan Whitehorn gpart_create(pp, fsname, rootsizestr, "/", NULL, 0); 3592118f387SNathan Whitehorn geom_deletetree(&submesh); 3602118f387SNathan Whitehorn 3612118f387SNathan Whitehorn geom_gettree(&submesh); 3622118f387SNathan Whitehorn pp = provider_for_name(&submesh, disk); 3632118f387SNathan Whitehorn gpart_create(pp, "freebsd-swap", swapsizestr, NULL, NULL, 0); 3642118f387SNathan Whitehorn geom_deletetree(&submesh); 3652118f387SNathan Whitehorn 3662118f387SNathan Whitehorn return (0); 3672118f387SNathan Whitehorn } 368