15eca7e06SNathan Whitehorn /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
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
295eca7e06SNathan Whitehorn #include <sys/param.h>
305eca7e06SNathan Whitehorn
317d51283fSBaptiste Daroussin #include <ctype.h>
325eca7e06SNathan Whitehorn #include <libgeom.h>
337d51283fSBaptiste Daroussin #include <stdio.h>
347d51283fSBaptiste Daroussin #include <stdlib.h>
357d51283fSBaptiste Daroussin #include <string.h>
365eca7e06SNathan Whitehorn
375eca7e06SNathan Whitehorn #include "partedit.h"
385eca7e06SNathan Whitehorn
395eca7e06SNathan Whitehorn static struct gprovider *
provider_for_name(struct gmesh * mesh,const char * name)405eca7e06SNathan Whitehorn provider_for_name(struct gmesh *mesh, const char *name)
415eca7e06SNathan Whitehorn {
425eca7e06SNathan Whitehorn struct gclass *classp;
435eca7e06SNathan Whitehorn struct gprovider *pp = NULL;
445eca7e06SNathan Whitehorn struct ggeom *gp;
455eca7e06SNathan Whitehorn
465eca7e06SNathan Whitehorn LIST_FOREACH(classp, &mesh->lg_class, lg_class) {
475eca7e06SNathan Whitehorn LIST_FOREACH(gp, &classp->lg_geom, lg_geom) {
485eca7e06SNathan Whitehorn if (LIST_EMPTY(&gp->lg_provider))
495eca7e06SNathan Whitehorn continue;
505eca7e06SNathan Whitehorn
515eca7e06SNathan Whitehorn LIST_FOREACH(pp, &gp->lg_provider, lg_provider)
525eca7e06SNathan Whitehorn if (strcmp(pp->lg_name, name) == 0)
535eca7e06SNathan Whitehorn break;
545eca7e06SNathan Whitehorn
555eca7e06SNathan Whitehorn if (pp != NULL) break;
565eca7e06SNathan Whitehorn }
575eca7e06SNathan Whitehorn
585eca7e06SNathan Whitehorn if (pp != NULL) break;
595eca7e06SNathan Whitehorn }
605eca7e06SNathan Whitehorn
615eca7e06SNathan Whitehorn return (pp);
625eca7e06SNathan Whitehorn }
635eca7e06SNathan Whitehorn
645eca7e06SNathan Whitehorn static int
part_config(char * disk,const char * scheme,char * config)655eca7e06SNathan Whitehorn part_config(char *disk, const char *scheme, char *config)
665eca7e06SNathan Whitehorn {
675eca7e06SNathan Whitehorn char *partition, *ap, *size = NULL, *type = NULL, *mount = NULL;
685eca7e06SNathan Whitehorn struct gclass *classp;
695eca7e06SNathan Whitehorn struct gmesh mesh;
705eca7e06SNathan Whitehorn struct ggeom *gpart = NULL;
715eca7e06SNathan Whitehorn int error;
725eca7e06SNathan Whitehorn
735eca7e06SNathan Whitehorn if (scheme == NULL)
745eca7e06SNathan Whitehorn scheme = default_scheme();
755eca7e06SNathan Whitehorn
765eca7e06SNathan Whitehorn error = geom_gettree(&mesh);
77*23099099SJohn Baldwin if (error != 0)
78*23099099SJohn Baldwin return (-1);
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;
100*23099099SJohn Baldwin
101*23099099SJohn Baldwin if (geom_gettree(&submesh) == 0) {
1025eca7e06SNathan Whitehorn gpart_create(provider_for_name(&submesh, disk),
1035eca7e06SNathan Whitehorn "freebsd", NULL, NULL, &disk, 0);
1045eca7e06SNathan Whitehorn geom_deletetree(&submesh);
105*23099099SJohn Baldwin }
1065eca7e06SNathan Whitehorn } else {
1075eca7e06SNathan Whitehorn disk = strdup(disk);
1085eca7e06SNathan Whitehorn }
1095eca7e06SNathan Whitehorn
1105eca7e06SNathan Whitehorn geom_deletetree(&mesh);
1115eca7e06SNathan Whitehorn error = geom_gettree(&mesh);
112*23099099SJohn Baldwin if (error != 0) {
113*23099099SJohn Baldwin free(disk);
114*23099099SJohn Baldwin return (-1);
115*23099099SJohn Baldwin }
1165eca7e06SNathan Whitehorn
1175eca7e06SNathan Whitehorn /* Create partitions */
118a6b612e9SNathan Whitehorn if (config == NULL) {
1196e15678aSNathan Whitehorn wizard_makeparts(&mesh, disk, "ufs", 0);
120a6b612e9SNathan Whitehorn goto finished;
121a6b612e9SNathan Whitehorn }
122a6b612e9SNathan Whitehorn
1235eca7e06SNathan Whitehorn while ((partition = strsep(&config, ",")) != NULL) {
1245eca7e06SNathan Whitehorn while ((ap = strsep(&partition, " \t\n")) != NULL) {
1255eca7e06SNathan Whitehorn if (*ap == '\0')
1265eca7e06SNathan Whitehorn continue;
1275eca7e06SNathan Whitehorn if (size == NULL)
1285eca7e06SNathan Whitehorn size = ap;
1295eca7e06SNathan Whitehorn else if (type == NULL)
1305eca7e06SNathan Whitehorn type = ap;
1315eca7e06SNathan Whitehorn else if (mount == NULL)
1325eca7e06SNathan Whitehorn mount = ap;
1335eca7e06SNathan Whitehorn }
1345eca7e06SNathan Whitehorn if (size == NULL)
1355eca7e06SNathan Whitehorn continue;
1365eca7e06SNathan Whitehorn if (strcmp(size, "auto") == 0)
1375eca7e06SNathan Whitehorn size = NULL;
1385eca7e06SNathan Whitehorn gpart_create(provider_for_name(&mesh, disk), type, size, mount,
1395eca7e06SNathan Whitehorn NULL, 0);
1405eca7e06SNathan Whitehorn geom_deletetree(&mesh);
1415eca7e06SNathan Whitehorn error = geom_gettree(&mesh);
142*23099099SJohn Baldwin if (error != 0) {
143*23099099SJohn Baldwin free(disk);
144*23099099SJohn Baldwin return (-1);
145*23099099SJohn Baldwin }
1465eca7e06SNathan Whitehorn size = type = mount = NULL;
1475eca7e06SNathan Whitehorn }
1485eca7e06SNathan Whitehorn
149a6b612e9SNathan Whitehorn finished:
1505eca7e06SNathan Whitehorn geom_deletetree(&mesh);
1515eca7e06SNathan Whitehorn free(disk);
1525eca7e06SNathan Whitehorn
1535eca7e06SNathan Whitehorn return (0);
1545eca7e06SNathan Whitehorn }
1555eca7e06SNathan Whitehorn
156*23099099SJohn Baldwin static int
parse_disk_config(char * input)157*23099099SJohn Baldwin parse_disk_config(char *input)
1585eca7e06SNathan Whitehorn {
1595eca7e06SNathan Whitehorn char *ap;
1605eca7e06SNathan Whitehorn char *disk = NULL, *scheme = NULL, *partconfig = NULL;
1615eca7e06SNathan Whitehorn
1625eca7e06SNathan Whitehorn while (input != NULL && *input != 0) {
1635eca7e06SNathan Whitehorn if (isspace(*input)) {
1645eca7e06SNathan Whitehorn input++;
1655eca7e06SNathan Whitehorn continue;
1665eca7e06SNathan Whitehorn }
1675eca7e06SNathan Whitehorn
1685eca7e06SNathan Whitehorn switch(*input) {
1695eca7e06SNathan Whitehorn case '{':
1705eca7e06SNathan Whitehorn input++;
1715eca7e06SNathan Whitehorn partconfig = strchr(input, '}');
1725eca7e06SNathan Whitehorn if (partconfig == NULL) {
1735eca7e06SNathan Whitehorn fprintf(stderr, "Malformed partition setup "
1745eca7e06SNathan Whitehorn "string: %s\n", input);
1755eca7e06SNathan Whitehorn return (1);
1765eca7e06SNathan Whitehorn }
1775eca7e06SNathan Whitehorn *partconfig = '\0';
1785eca7e06SNathan Whitehorn ap = partconfig+1;
1795eca7e06SNathan Whitehorn partconfig = input;
1805eca7e06SNathan Whitehorn input = ap;
1815eca7e06SNathan Whitehorn break;
1825eca7e06SNathan Whitehorn default:
1835eca7e06SNathan Whitehorn if (disk == NULL)
1845eca7e06SNathan Whitehorn disk = strsep(&input, " \t\n");
1855eca7e06SNathan Whitehorn else if (scheme == NULL)
1865eca7e06SNathan Whitehorn scheme = strsep(&input, " \t\n");
1875eca7e06SNathan Whitehorn else {
1885eca7e06SNathan Whitehorn fprintf(stderr, "Unknown directive: %s\n",
1895eca7e06SNathan Whitehorn strsep(&input, " \t\n"));
1905eca7e06SNathan Whitehorn return (1);
1915eca7e06SNathan Whitehorn }
1925eca7e06SNathan Whitehorn }
1935eca7e06SNathan Whitehorn } while (input != NULL && *input != 0);
1945eca7e06SNathan Whitehorn
1955140034cSNathan Whitehorn if (disk == NULL || strcmp(disk, "DEFAULT") == 0) {
1965140034cSNathan Whitehorn struct gmesh mesh;
197*23099099SJohn Baldwin
198*23099099SJohn Baldwin if (geom_gettree(&mesh) == 0) {
1995140034cSNathan Whitehorn disk = boot_disk_select(&mesh);
2005140034cSNathan Whitehorn geom_deletetree(&mesh);
2015140034cSNathan Whitehorn }
202*23099099SJohn Baldwin }
2035eca7e06SNathan Whitehorn
2045140034cSNathan Whitehorn return (part_config(disk, scheme, partconfig));
2055eca7e06SNathan Whitehorn }
2065eca7e06SNathan Whitehorn
2075eca7e06SNathan Whitehorn int
scripted_editor(int argc,const char ** argv)2085eca7e06SNathan Whitehorn scripted_editor(int argc, const char **argv)
2095eca7e06SNathan Whitehorn {
210f66a8328SJohn Baldwin FILE *fp;
211f66a8328SJohn Baldwin char *input, *token;
212f66a8328SJohn Baldwin size_t len;
213f66a8328SJohn Baldwin int i, error = 0;
2145eca7e06SNathan Whitehorn
215f66a8328SJohn Baldwin fp = open_memstream(&input, &len);
216f66a8328SJohn Baldwin fputs(argv[1], fp);
2175eca7e06SNathan Whitehorn for (i = 2; i < argc; i++) {
218f66a8328SJohn Baldwin fprintf(fp, " %s", argv[i]);
2195eca7e06SNathan Whitehorn }
220f66a8328SJohn Baldwin fclose(fp);
2215eca7e06SNathan Whitehorn
222f0ddc92dSNathan Whitehorn while ((token = strsep(&input, ";")) != NULL) {
223f0ddc92dSNathan Whitehorn error = parse_disk_config(token);
224f66a8328SJohn Baldwin if (error != 0) {
225f66a8328SJohn Baldwin free(input);
226f0ddc92dSNathan Whitehorn return (error);
227f0ddc92dSNathan Whitehorn }
228f66a8328SJohn Baldwin }
229f66a8328SJohn Baldwin free(input);
2305eca7e06SNathan Whitehorn
2315eca7e06SNathan Whitehorn return (0);
2325eca7e06SNathan Whitehorn }
2335eca7e06SNathan Whitehorn
234