15eca7e06SNathan Whitehorn /*- 21de7b4b8SPedro F. Giffuni * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 31de7b4b8SPedro F. Giffuni * 45eca7e06SNathan Whitehorn * Copyright (c) 2013 Nathan Whitehorn 55eca7e06SNathan Whitehorn * All rights reserved. 65eca7e06SNathan Whitehorn * 75eca7e06SNathan Whitehorn * Redistribution and use in source and binary forms, with or without 85eca7e06SNathan Whitehorn * modification, are permitted provided that the following conditions 95eca7e06SNathan Whitehorn * are met: 105eca7e06SNathan Whitehorn * 1. Redistributions of source code must retain the above copyright 115eca7e06SNathan Whitehorn * notice, this list of conditions and the following disclaimer. 125eca7e06SNathan Whitehorn * 2. Redistributions in binary form must reproduce the above copyright 135eca7e06SNathan Whitehorn * notice, this list of conditions and the following disclaimer in the 145eca7e06SNathan Whitehorn * documentation and/or other materials provided with the distribution. 155eca7e06SNathan Whitehorn * 165eca7e06SNathan Whitehorn * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 175eca7e06SNathan Whitehorn * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 185eca7e06SNathan Whitehorn * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 195eca7e06SNathan Whitehorn * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 205eca7e06SNathan Whitehorn * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 215eca7e06SNathan Whitehorn * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 225eca7e06SNathan Whitehorn * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 235eca7e06SNathan Whitehorn * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 245eca7e06SNathan Whitehorn * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 255eca7e06SNathan Whitehorn * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 265eca7e06SNathan Whitehorn * SUCH DAMAGE. 275eca7e06SNathan Whitehorn * 285eca7e06SNathan Whitehorn * $FreeBSD$ 295eca7e06SNathan Whitehorn */ 305eca7e06SNathan Whitehorn 315eca7e06SNathan Whitehorn #include <sys/param.h> 325eca7e06SNathan Whitehorn 33*7d51283fSBaptiste Daroussin #include <ctype.h> 345eca7e06SNathan Whitehorn #include <libgeom.h> 35*7d51283fSBaptiste Daroussin #include <stdio.h> 36*7d51283fSBaptiste Daroussin #include <stdlib.h> 37*7d51283fSBaptiste Daroussin #include <string.h> 385eca7e06SNathan Whitehorn 395eca7e06SNathan Whitehorn #include "partedit.h" 405eca7e06SNathan Whitehorn 415eca7e06SNathan Whitehorn static struct gprovider * 425eca7e06SNathan Whitehorn provider_for_name(struct gmesh *mesh, const char *name) 435eca7e06SNathan Whitehorn { 445eca7e06SNathan Whitehorn struct gclass *classp; 455eca7e06SNathan Whitehorn struct gprovider *pp = NULL; 465eca7e06SNathan Whitehorn struct ggeom *gp; 475eca7e06SNathan Whitehorn 485eca7e06SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) { 495eca7e06SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) { 505eca7e06SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider)) 515eca7e06SNathan Whitehorn continue; 525eca7e06SNathan Whitehorn 535eca7e06SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider) 545eca7e06SNathan Whitehorn if (strcmp(pp->lg_name, name) == 0) 555eca7e06SNathan Whitehorn break; 565eca7e06SNathan Whitehorn 575eca7e06SNathan Whitehorn if (pp != NULL) break; 585eca7e06SNathan Whitehorn } 595eca7e06SNathan Whitehorn 605eca7e06SNathan Whitehorn if (pp != NULL) break; 615eca7e06SNathan Whitehorn } 625eca7e06SNathan Whitehorn 635eca7e06SNathan Whitehorn return (pp); 645eca7e06SNathan Whitehorn } 655eca7e06SNathan Whitehorn 665eca7e06SNathan Whitehorn static int 675eca7e06SNathan Whitehorn part_config(char *disk, const char *scheme, char *config) 685eca7e06SNathan Whitehorn { 695eca7e06SNathan Whitehorn char *partition, *ap, *size = NULL, *type = NULL, *mount = NULL; 705eca7e06SNathan Whitehorn struct gclass *classp; 715eca7e06SNathan Whitehorn struct gmesh mesh; 725eca7e06SNathan Whitehorn struct ggeom *gpart = NULL; 735eca7e06SNathan Whitehorn int error; 745eca7e06SNathan Whitehorn 755eca7e06SNathan Whitehorn if (scheme == NULL) 765eca7e06SNathan Whitehorn scheme = default_scheme(); 775eca7e06SNathan Whitehorn 785eca7e06SNathan Whitehorn error = geom_gettree(&mesh); 79f0ddc92dSNathan Whitehorn if (provider_for_name(&mesh, disk) == NULL) { 80f0ddc92dSNathan Whitehorn fprintf(stderr, "GEOM provider %s not found\n", disk); 81f0ddc92dSNathan Whitehorn geom_deletetree(&mesh); 82f0ddc92dSNathan Whitehorn return (-1); 83f0ddc92dSNathan Whitehorn } 845eca7e06SNathan Whitehorn 855eca7e06SNathan Whitehorn /* Remove any existing partitioning and create new scheme */ 865eca7e06SNathan Whitehorn LIST_FOREACH(classp, &mesh.lg_class, lg_class) 875eca7e06SNathan Whitehorn if (strcmp(classp->lg_name, "PART") == 0) 885eca7e06SNathan Whitehorn break; 895eca7e06SNathan Whitehorn if (classp != NULL) { 905eca7e06SNathan Whitehorn LIST_FOREACH(gpart, &classp->lg_geom, lg_geom) 915eca7e06SNathan Whitehorn if (strcmp(gpart->lg_name, disk) == 0) 925eca7e06SNathan Whitehorn break; 935eca7e06SNathan Whitehorn } 945eca7e06SNathan Whitehorn if (gpart != NULL) 955eca7e06SNathan Whitehorn gpart_destroy(gpart); 965eca7e06SNathan Whitehorn gpart_partition(disk, scheme); 975eca7e06SNathan Whitehorn 982b375b4eSYoshihiro Takahashi if (strcmp(scheme, "MBR") == 0) { 995eca7e06SNathan Whitehorn struct gmesh submesh; 1005eca7e06SNathan Whitehorn geom_gettree(&submesh); 1015eca7e06SNathan Whitehorn gpart_create(provider_for_name(&submesh, disk), 1025eca7e06SNathan Whitehorn "freebsd", NULL, NULL, &disk, 0); 1035eca7e06SNathan Whitehorn geom_deletetree(&submesh); 1045eca7e06SNathan Whitehorn } else { 1055eca7e06SNathan Whitehorn disk= strdup(disk); 1065eca7e06SNathan Whitehorn } 1075eca7e06SNathan Whitehorn 1085eca7e06SNathan Whitehorn geom_deletetree(&mesh); 1095eca7e06SNathan Whitehorn error = geom_gettree(&mesh); 1105eca7e06SNathan Whitehorn 1115eca7e06SNathan Whitehorn /* Create partitions */ 112a6b612e9SNathan Whitehorn if (config == NULL) { 1136e15678aSNathan Whitehorn wizard_makeparts(&mesh, disk, "ufs", 0); 114a6b612e9SNathan Whitehorn goto finished; 115a6b612e9SNathan Whitehorn } 116a6b612e9SNathan Whitehorn 1175eca7e06SNathan Whitehorn while ((partition = strsep(&config, ",")) != NULL) { 1185eca7e06SNathan Whitehorn while ((ap = strsep(&partition, " \t\n")) != NULL) { 1195eca7e06SNathan Whitehorn if (*ap == '\0') 1205eca7e06SNathan Whitehorn continue; 1215eca7e06SNathan Whitehorn if (size == NULL) 1225eca7e06SNathan Whitehorn size = ap; 1235eca7e06SNathan Whitehorn else if (type == NULL) 1245eca7e06SNathan Whitehorn type = ap; 1255eca7e06SNathan Whitehorn else if (mount == NULL) 1265eca7e06SNathan Whitehorn mount = ap; 1275eca7e06SNathan Whitehorn } 1285eca7e06SNathan Whitehorn if (size == NULL) 1295eca7e06SNathan Whitehorn continue; 1305eca7e06SNathan Whitehorn if (strcmp(size, "auto") == 0) 1315eca7e06SNathan Whitehorn size = NULL; 1325eca7e06SNathan Whitehorn gpart_create(provider_for_name(&mesh, disk), type, size, mount, 1335eca7e06SNathan Whitehorn NULL, 0); 1345eca7e06SNathan Whitehorn geom_deletetree(&mesh); 1355eca7e06SNathan Whitehorn error = geom_gettree(&mesh); 1365eca7e06SNathan Whitehorn size = type = mount = NULL; 1375eca7e06SNathan Whitehorn } 1385eca7e06SNathan Whitehorn 139a6b612e9SNathan Whitehorn finished: 1405eca7e06SNathan Whitehorn geom_deletetree(&mesh); 1415eca7e06SNathan Whitehorn free(disk); 1425eca7e06SNathan Whitehorn 1435eca7e06SNathan Whitehorn return (0); 1445eca7e06SNathan Whitehorn } 1455eca7e06SNathan Whitehorn 1465eca7e06SNathan Whitehorn static 1475eca7e06SNathan Whitehorn int parse_disk_config(char *input) 1485eca7e06SNathan Whitehorn { 1495eca7e06SNathan Whitehorn char *ap; 1505eca7e06SNathan Whitehorn char *disk = NULL, *scheme = NULL, *partconfig = NULL; 1515eca7e06SNathan Whitehorn 1525eca7e06SNathan Whitehorn while (input != NULL && *input != 0) { 1535eca7e06SNathan Whitehorn if (isspace(*input)) { 1545eca7e06SNathan Whitehorn input++; 1555eca7e06SNathan Whitehorn continue; 1565eca7e06SNathan Whitehorn } 1575eca7e06SNathan Whitehorn 1585eca7e06SNathan Whitehorn switch(*input) { 1595eca7e06SNathan Whitehorn case '{': 1605eca7e06SNathan Whitehorn input++; 1615eca7e06SNathan Whitehorn partconfig = strchr(input, '}'); 1625eca7e06SNathan Whitehorn if (partconfig == NULL) { 1635eca7e06SNathan Whitehorn fprintf(stderr, "Malformed partition setup " 1645eca7e06SNathan Whitehorn "string: %s\n", input); 1655eca7e06SNathan Whitehorn return (1); 1665eca7e06SNathan Whitehorn } 1675eca7e06SNathan Whitehorn *partconfig = '\0'; 1685eca7e06SNathan Whitehorn ap = partconfig+1; 1695eca7e06SNathan Whitehorn partconfig = input; 1705eca7e06SNathan Whitehorn input = ap; 1715eca7e06SNathan Whitehorn break; 1725eca7e06SNathan Whitehorn default: 1735eca7e06SNathan Whitehorn if (disk == NULL) 1745eca7e06SNathan Whitehorn disk = strsep(&input, " \t\n"); 1755eca7e06SNathan Whitehorn else if (scheme == NULL) 1765eca7e06SNathan Whitehorn scheme = strsep(&input, " \t\n"); 1775eca7e06SNathan Whitehorn else { 1785eca7e06SNathan Whitehorn fprintf(stderr, "Unknown directive: %s\n", 1795eca7e06SNathan Whitehorn strsep(&input, " \t\n")); 1805eca7e06SNathan Whitehorn return (1); 1815eca7e06SNathan Whitehorn } 1825eca7e06SNathan Whitehorn } 1835eca7e06SNathan Whitehorn } while (input != NULL && *input != 0); 1845eca7e06SNathan Whitehorn 1855140034cSNathan Whitehorn if (disk == NULL || strcmp(disk, "DEFAULT") == 0) { 1865140034cSNathan Whitehorn struct gmesh mesh; 1875140034cSNathan Whitehorn geom_gettree(&mesh); 1885140034cSNathan Whitehorn disk = boot_disk_select(&mesh); 1895140034cSNathan Whitehorn geom_deletetree(&mesh); 1905140034cSNathan Whitehorn } 1915eca7e06SNathan Whitehorn 1925140034cSNathan Whitehorn return (part_config(disk, scheme, partconfig)); 1935eca7e06SNathan Whitehorn } 1945eca7e06SNathan Whitehorn 1955eca7e06SNathan Whitehorn int 1965eca7e06SNathan Whitehorn scripted_editor(int argc, const char **argv) 1975eca7e06SNathan Whitehorn { 1985eca7e06SNathan Whitehorn char *token; 199f0ddc92dSNathan Whitehorn int i, error = 0, len = 0; 2005eca7e06SNathan Whitehorn 2015eca7e06SNathan Whitehorn for (i = 1; i < argc; i++) 2025eca7e06SNathan Whitehorn len += strlen(argv[i]) + 1; 2035eca7e06SNathan Whitehorn char inputbuf[len], *input = inputbuf; 2045eca7e06SNathan Whitehorn strcpy(input, argv[1]); 2055eca7e06SNathan Whitehorn for (i = 2; i < argc; i++) { 2065eca7e06SNathan Whitehorn strcat(input, " "); 2075eca7e06SNathan Whitehorn strcat(input, argv[i]); 2085eca7e06SNathan Whitehorn } 2095eca7e06SNathan Whitehorn 210f0ddc92dSNathan Whitehorn while ((token = strsep(&input, ";")) != NULL) { 211f0ddc92dSNathan Whitehorn error = parse_disk_config(token); 212f0ddc92dSNathan Whitehorn if (error != 0) 213f0ddc92dSNathan Whitehorn return (error); 214f0ddc92dSNathan Whitehorn } 2155eca7e06SNathan Whitehorn 2165eca7e06SNathan Whitehorn return (0); 2175eca7e06SNathan Whitehorn } 2185eca7e06SNathan Whitehorn 219