#include <sys/param.h>
#include <ctype.h>
#include <libgeom.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "partedit.h"
static struct gprovider *
provider_for_name(struct gmesh *mesh, const char *name)
{
struct gclass *classp;
struct gprovider *pp = NULL;
struct ggeom *gp;
LIST_FOREACH(classp, &mesh->lg_class, lg_class) {
LIST_FOREACH(gp, &classp->lg_geom, lg_geom) {
if (LIST_EMPTY(&gp->lg_provider))
continue;
LIST_FOREACH(pp, &gp->lg_provider, lg_provider)
if (strcmp(pp->lg_name, name) == 0)
break;
if (pp != NULL) break;
}
if (pp != NULL) break;
}
return (pp);
}
static int
part_config(char *disk, const char *scheme, char *config)
{
char *partition, *ap, *size = NULL, *type = NULL, *mount = NULL;
struct gclass *classp;
struct gmesh mesh;
struct ggeom *gpart = NULL;
int error;
if (scheme == NULL)
scheme = default_scheme();
error = geom_gettree(&mesh);
if (error != 0)
return (-1);
if (provider_for_name(&mesh, disk) == NULL) {
fprintf(stderr, "GEOM provider %s not found\n", disk);
geom_deletetree(&mesh);
return (-1);
}
LIST_FOREACH(classp, &mesh.lg_class, lg_class)
if (strcmp(classp->lg_name, "PART") == 0)
break;
if (classp != NULL) {
LIST_FOREACH(gpart, &classp->lg_geom, lg_geom)
if (strcmp(gpart->lg_name, disk) == 0)
break;
}
if (gpart != NULL)
gpart_destroy(gpart);
gpart_partition(disk, scheme);
if (strcmp(scheme, "MBR") == 0) {
struct gmesh submesh;
if (geom_gettree(&submesh) == 0) {
gpart_create(provider_for_name(&submesh, disk),
"freebsd", NULL, NULL, &disk, 0);
geom_deletetree(&submesh);
}
} else {
disk = strdup(disk);
}
geom_deletetree(&mesh);
error = geom_gettree(&mesh);
if (error != 0) {
free(disk);
return (-1);
}
if (config == NULL) {
wizard_makeparts(&mesh, disk, "ufs", 0);
goto finished;
}
while ((partition = strsep(&config, ",")) != NULL) {
while ((ap = strsep(&partition, " \t\n")) != NULL) {
if (*ap == '\0')
continue;
if (size == NULL)
size = ap;
else if (type == NULL)
type = ap;
else if (mount == NULL)
mount = ap;
}
if (size == NULL)
continue;
if (strcmp(size, "auto") == 0)
size = NULL;
gpart_create(provider_for_name(&mesh, disk), type, size, mount,
NULL, 0);
geom_deletetree(&mesh);
error = geom_gettree(&mesh);
if (error != 0) {
free(disk);
return (-1);
}
size = type = mount = NULL;
}
finished:
geom_deletetree(&mesh);
free(disk);
return (0);
}
static int
parse_disk_config(char *input)
{
char *ap;
char *disk = NULL, *scheme = NULL, *partconfig = NULL;
while (input != NULL && *input != 0) {
if (isspace(*input)) {
input++;
continue;
}
switch(*input) {
case '{':
input++;
partconfig = strchr(input, '}');
if (partconfig == NULL) {
fprintf(stderr, "Malformed partition setup "
"string: %s\n", input);
return (1);
}
*partconfig = '\0';
ap = partconfig+1;
partconfig = input;
input = ap;
break;
default:
if (disk == NULL)
disk = strsep(&input, " \t\n");
else if (scheme == NULL)
scheme = strsep(&input, " \t\n");
else {
fprintf(stderr, "Unknown directive: %s\n",
strsep(&input, " \t\n"));
return (1);
}
}
} while (input != NULL && *input != 0);
if (disk == NULL || strcmp(disk, "DEFAULT") == 0) {
struct gmesh mesh;
if (geom_gettree(&mesh) == 0) {
disk = boot_disk_select(&mesh);
geom_deletetree(&mesh);
}
}
return (part_config(disk, scheme, partconfig));
}
int
scripted_editor(int argc, const char **argv)
{
FILE *fp;
char *input, *token;
size_t len;
int i, error = 0;
fp = open_memstream(&input, &len);
fputs(argv[1], fp);
for (i = 2; i < argc; i++) {
fprintf(fp, " %s", argv[i]);
}
fclose(fp);
while ((token = strsep(&input, ";")) != NULL) {
error = parse_disk_config(token);
if (error != 0) {
free(input);
return (error);
}
}
free(input);
return (0);
}