#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/param.h>
#include <sys/mnttab.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/openpromio.h>
#include <stgcom.h>
#define MAXPROPSIZE 128
#define MAXVALSIZE (8192 - MAXPROPSIZE - sizeof (uint_t))
#define BOOTDEV_PROP_NAME "boot-device"
static int getbootdevname(char *, char *);
static int setprom(unsigned, unsigned, char *);
extern int devfs_dev_to_prom_name(char *, char *);
int
setboot(unsigned int yes, unsigned int verbose, char *fname)
{
char bdev[MAXPATHLEN];
if (!getbootdevname(fname, bdev)) {
(void) fprintf(stderr, MSGSTR(6000,
"Cannot determine device name for %s\n"),
fname);
return (errno);
}
return (setprom(yes, verbose, bdev));
}
static int
getbootdevname(char *bootfs, char *bdev)
{
FILE *f;
char *fname;
char *devname;
struct mnttab m;
struct stat sbuf;
int mountpt = 0;
int found = 0;
devname = bootfs;
if (stat(bootfs, &sbuf) < 0) {
perror(MSGSTR(6001, "stat"));
return (0);
}
switch (sbuf.st_mode & S_IFMT) {
case S_IFBLK:
break;
default:
mountpt = 1;
break;
}
if (mountpt) {
fname = MNTTAB;
f = fopen(fname, "r");
if (f == NULL) {
perror(fname);
return (0);
}
while (getmntent(f, &m) == 0) {
if (strcmp(m.mnt_mountp, bootfs))
continue;
else {
found = 1;
break;
}
}
(void) fclose(f);
if (!found) {
return (0);
}
devname = m.mnt_special;
}
if (devfs_dev_to_prom_name(devname, bdev) != 0) {
perror(devname);
return (0);
}
return (1);
}
static int
setprom(unsigned yes, unsigned verbose, char *bdev)
{
struct openpromio *pio;
int fd;
char save_bootdev[MAXVALSIZE];
if ((fd = open("/dev/openprom", O_RDWR)) < 0) {
perror(MSGSTR(6002, "Could not open openprom dev"));
return (errno);
}
pio = (struct openpromio *)malloc(sizeof (struct openpromio) +
MAXVALSIZE + MAXPROPSIZE);
if (pio == (struct openpromio *)NULL) {
perror(MSGSTR(6003, " Error: Unable to allocate memory."));
return (errno);
}
pio->oprom_size = MAXVALSIZE;
(void) strcpy(pio->oprom_array, BOOTDEV_PROP_NAME);
if (ioctl(fd, OPROMGETOPT, pio) < 0) {
perror(MSGSTR(6004, "openprom getopt ioctl"));
return (errno);
}
(void) strcpy(save_bootdev, pio->oprom_array);
if (verbose) {
(void) fprintf(stdout,
MSGSTR(6005,
"Current boot-device = %s\n"), pio->oprom_array);
(void) fprintf(stdout, MSGSTR(6006,
"New boot-device = %s\n"), bdev);
}
if (!yes) {
(void) fprintf(stdout, MSGSTR(6007,
"Do you want to change boot-device "
"to the new setting? (y/n) "));
switch (getchar()) {
case 'Y':
case 'y':
break;
default:
return (0);
}
}
pio->oprom_size = (int)strlen(BOOTDEV_PROP_NAME) + 1 +
(int)strlen(bdev);
(void) strcpy(pio->oprom_array, BOOTDEV_PROP_NAME);
(void) strcpy(pio->oprom_array + (int)strlen(BOOTDEV_PROP_NAME) + 1,
bdev);
if (ioctl(fd, OPROMSETOPT, pio) < 0) {
perror(MSGSTR(6008, "openprom setopt ioctl"));
return (errno);
}
pio->oprom_size = MAXVALSIZE;
(void) strcpy(pio->oprom_array, BOOTDEV_PROP_NAME);
if (ioctl(fd, OPROMGETOPT, pio) < 0) {
perror(MSGSTR(6009, "openprom getopt ioctl"));
return (errno);
}
if (strcmp(bdev, pio->oprom_array)) {
perror(MSGSTR(6010,
"Could not set boot-device, reverting to old value"));
pio->oprom_size = (int)strlen(BOOTDEV_PROP_NAME) + 1 +
(int)strlen(save_bootdev);
(void) strcpy(pio->oprom_array, BOOTDEV_PROP_NAME);
(void) strcpy(pio->oprom_array +
(int)strlen(BOOTDEV_PROP_NAME) + 1,
save_bootdev);
if (ioctl(fd, OPROMSETOPT, pio) < 0) {
perror(MSGSTR(6011, "openprom setopt ioctl"));
return (errno);
}
}
(void) close(fd);
return (0);
}