xref: /freebsd/usr.sbin/bsdinstall/partedit/scripted.c (revision 7d51283f700cb6f43f3293c40a8fcab3854c5fc1)
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