2004-12-03 00:29:29 +00:00
|
|
|
/*
|
|
|
|
*
|
2007-05-04 09:19:57 +00:00
|
|
|
* (c) 2004-2007 Laurent Vivier <Laurent@lvivier.info>
|
2004-12-03 00:29:29 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
2004-12-16 23:27:50 +00:00
|
|
|
#include <getopt.h>
|
2007-05-06 23:03:55 +00:00
|
|
|
#include <libgen.h>
|
2004-12-03 00:29:29 +00:00
|
|
|
|
2004-12-10 00:28:35 +00:00
|
|
|
#include "libemile.h"
|
2007-05-11 12:21:58 +00:00
|
|
|
#include "emile_config.h"
|
2004-12-07 00:39:39 +00:00
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
int verbose = 0;
|
|
|
|
|
2004-12-22 00:29:25 +00:00
|
|
|
extern void scanbus(void);
|
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
enum {
|
|
|
|
ACTION_NONE = 0x00000000,
|
|
|
|
ACTION_SCANBUS = 0x00000001,
|
|
|
|
ACTION_SET_HFS = 0x00000002,
|
|
|
|
ACTION_RESTORE = 0x00000004,
|
|
|
|
ACTION_BACKUP = 0x00000008,
|
2007-05-11 12:21:58 +00:00
|
|
|
ACTION_TEST = 0x00000010,
|
|
|
|
ACTION_CONFIG = 0x00000020,
|
2004-12-27 21:39:19 +00:00
|
|
|
};
|
|
|
|
|
2004-12-22 00:29:25 +00:00
|
|
|
enum {
|
2004-12-23 23:24:06 +00:00
|
|
|
ARG_NONE = 0,
|
|
|
|
ARG_SCANBUS,
|
|
|
|
ARG_SET_HFS,
|
2004-12-27 21:39:19 +00:00
|
|
|
ARG_RESTORE,
|
2004-12-23 23:24:06 +00:00
|
|
|
ARG_BACKUP,
|
2004-12-22 00:29:25 +00:00
|
|
|
ARG_VERBOSE ='v',
|
2004-12-24 00:10:33 +00:00
|
|
|
ARG_TEST = 't',
|
2004-12-22 00:29:25 +00:00
|
|
|
ARG_HELP = 'h',
|
2007-05-11 12:21:58 +00:00
|
|
|
ARG_CONFIG = 'c',
|
2004-12-22 00:29:25 +00:00
|
|
|
};
|
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
static struct option long_options[] =
|
|
|
|
{
|
2004-12-23 23:24:06 +00:00
|
|
|
{"scanbus", 0, NULL, ARG_SCANBUS },
|
2004-12-27 01:32:03 +00:00
|
|
|
{"set-hfs", 0, NULL, ARG_SET_HFS },
|
2004-12-27 21:39:19 +00:00
|
|
|
{"restore", 2, NULL, ARG_RESTORE },
|
2004-12-23 23:24:06 +00:00
|
|
|
{"backup", 2, NULL, ARG_BACKUP },
|
2007-05-11 12:21:58 +00:00
|
|
|
{"verbose", 0, NULL, ARG_VERBOSE },
|
|
|
|
{"help", 0, NULL, ARG_HELP },
|
2004-12-23 23:24:06 +00:00
|
|
|
{"test", 0, NULL, ARG_TEST },
|
2007-05-11 12:21:58 +00:00
|
|
|
{"config", 1, NULL, ARG_CONFIG },
|
2004-12-23 23:24:06 +00:00
|
|
|
{NULL, 0, NULL, 0 }
|
2004-12-16 23:27:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void usage(int argc, char** argv)
|
|
|
|
{
|
2004-12-24 00:10:33 +00:00
|
|
|
fprintf(stderr, "Usage: %s [OPTION]\n", argv[0]);
|
|
|
|
fprintf(stderr, "Update and install EMILE stuff on your SCSI disk.\n");
|
|
|
|
fprintf(stderr, "EMILE allows to boot linux directly from linux partition\n");
|
|
|
|
fprintf(stderr," -h, --help display this text\n");
|
|
|
|
fprintf(stderr," -v, --verbose active verbose mode\n");
|
|
|
|
fprintf(stderr," -t, --test active test mode (don't write to disk)\n");
|
|
|
|
fprintf(stderr," --scanbus display information about all disks and partitions\n");
|
2004-12-27 21:39:19 +00:00
|
|
|
fprintf(stderr," --restore[=FILE] save current boot block from FILE\n");
|
2004-12-24 00:10:33 +00:00
|
|
|
fprintf(stderr," --backup[=FILE] save current boot block to FILE\n");
|
|
|
|
fprintf(stderr," --set-hfs set type of partition DEV to Apple_HFS (needed to be bootable)\n");
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr," -c, --config FILE use config file FILE\n");
|
2004-12-24 00:10:33 +00:00
|
|
|
fprintf(stderr, "!!! USE WITH CAUTION AND AT YOUR OWN RISK !!!\n");
|
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
fprintf(stderr, "\nbuild: \n%s\n", SIGNATURE);
|
|
|
|
}
|
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
static int open_map_of( char *dev_name, int flags,
|
|
|
|
emile_map_t **map, int *partition)
|
2004-12-22 00:29:25 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int disk;
|
|
|
|
char disk_name[16];
|
2006-06-01 23:36:57 +00:00
|
|
|
int driver;
|
2004-12-22 00:29:25 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_scsi_get_rdev(dev_name, &driver, &disk, partition);
|
2004-12-23 22:35:36 +00:00
|
|
|
if (ret == -1)
|
2006-06-01 23:36:57 +00:00
|
|
|
return -2;
|
2004-12-23 22:35:36 +00:00
|
|
|
|
2006-06-01 23:36:57 +00:00
|
|
|
emile_get_dev_name(disk_name, driver, disk, 0);
|
2004-12-22 00:29:25 +00:00
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
*map = emile_map_open(disk_name, flags);
|
2004-12-27 22:32:18 +00:00
|
|
|
if (*map == NULL)
|
|
|
|
return -1;
|
2004-12-23 22:35:36 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int check_has_apple_driver(char *dev_name)
|
|
|
|
{
|
|
|
|
emile_map_t *map;
|
|
|
|
int partition;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = open_map_of(dev_name, O_RDONLY, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2004-12-22 00:29:25 +00:00
|
|
|
|
|
|
|
ret = emile_map_has_apple_driver(map);
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int check_is_hfs(char *dev_name)
|
|
|
|
{
|
|
|
|
emile_map_t *map;
|
|
|
|
int ret;
|
|
|
|
int partition;
|
|
|
|
char *part_type;
|
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
ret = open_map_of(dev_name, O_RDONLY, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2004-12-22 00:29:25 +00:00
|
|
|
|
|
|
|
ret = emile_map_read(map, partition - 1);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
part_type = emile_map_get_partition_type(map);
|
|
|
|
ret = (strcmp("Apple_HFS", part_type) == 0);
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-22 23:14:07 +00:00
|
|
|
static int check_is_EMILE_bootblock(char *dev_name)
|
|
|
|
{
|
|
|
|
emile_map_t *map;
|
|
|
|
int ret;
|
|
|
|
int partition;
|
|
|
|
char bootblock[BOOTBLOCK_SIZE];
|
|
|
|
int bootblock_type;
|
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
ret = open_map_of(dev_name, O_RDONLY, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2004-12-22 23:14:07 +00:00
|
|
|
|
|
|
|
ret = emile_map_read(map, partition - 1);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = emile_map_bootblock_read(map, bootblock);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
bootblock_type = emile_map_bootblock_get_type(bootblock);
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
return EMILE_BOOTBLOCK == bootblock_type;
|
|
|
|
}
|
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
static int backup_bootblock(char *dev_name, char *filename)
|
2004-12-22 23:14:07 +00:00
|
|
|
{
|
2004-12-23 22:35:36 +00:00
|
|
|
emile_map_t *map;
|
|
|
|
int ret;
|
|
|
|
int partition;
|
|
|
|
char bootblock[BOOTBLOCK_SIZE];
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
ret = open_map_of(dev_name, O_RDONLY, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2004-12-23 22:35:36 +00:00
|
|
|
|
|
|
|
ret = emile_map_read(map, partition - 1);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = emile_map_bootblock_read(map, bootblock);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
/* save bootblock */
|
|
|
|
|
|
|
|
fd = open(filename, O_CREAT | O_EXCL | O_WRONLY, S_IRUSR | S_IWUSR);
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
if (errno == EEXIST)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: \"%s\" already exists.\n",
|
|
|
|
filename);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = write(fd, bootblock, BOOTBLOCK_SIZE);
|
|
|
|
if (ret != BOOTBLOCK_SIZE)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
close(fd);
|
|
|
|
|
|
|
|
return 0;
|
2004-12-22 23:14:07 +00:00
|
|
|
}
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
static int restore_bootblock(char *dev_name, char *filename)
|
2004-12-23 23:24:06 +00:00
|
|
|
{
|
|
|
|
emile_map_t *map;
|
|
|
|
int ret;
|
|
|
|
int partition;
|
|
|
|
char bootblock[BOOTBLOCK_SIZE];
|
|
|
|
int fd;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (!check_is_EMILE_bootblock(dev_name))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot restore bootblock over non-EMILE bootblock\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2004-12-23 23:24:06 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
/* read bootblock */
|
|
|
|
|
|
|
|
fd = open(filename, O_RDONLY);
|
2004-12-23 23:24:06 +00:00
|
|
|
if (fd == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = read(fd, bootblock, BOOTBLOCK_SIZE);
|
|
|
|
if (ret != BOOTBLOCK_SIZE)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
close(fd);
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
/* write bootblock */
|
2004-12-23 23:24:06 +00:00
|
|
|
|
2004-12-27 22:32:18 +00:00
|
|
|
ret = open_map_of(dev_name, O_RDWR, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
2004-12-23 23:24:06 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = emile_map_read(map, partition - 1);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = emile_map_bootblock_write(map, bootblock);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2004-12-23 23:48:35 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
static int copy_file_to_bootblock(char* first_path, char* dev_name)
|
2004-12-23 23:48:35 +00:00
|
|
|
{
|
|
|
|
emile_map_t *map;
|
|
|
|
int ret;
|
|
|
|
int partition;
|
2004-12-27 21:39:19 +00:00
|
|
|
char bootblock[BOOTBLOCK_SIZE];
|
|
|
|
int fd;
|
2004-12-23 23:48:35 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
/* read first level */
|
|
|
|
|
|
|
|
fd = open(first_path, O_RDONLY);
|
|
|
|
if (fd == -1)
|
2004-12-23 23:48:35 +00:00
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = read(fd, bootblock, BOOTBLOCK_SIZE);
|
|
|
|
if (ret != BOOTBLOCK_SIZE)
|
2004-12-23 23:48:35 +00:00
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
close(fd);
|
|
|
|
|
|
|
|
/* write bootblock to partition */
|
|
|
|
|
|
|
|
ret = open_map_of(dev_name, O_RDWR, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
2004-12-23 23:48:35 +00:00
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_map_read(map, partition - 1);
|
2004-12-24 00:10:33 +00:00
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_map_bootblock_write(map, bootblock);
|
2004-12-23 23:48:35 +00:00
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
static int set_HFS(char *dev_name)
|
2004-12-23 23:48:35 +00:00
|
|
|
{
|
2004-12-27 21:39:19 +00:00
|
|
|
emile_map_t *map;
|
2004-12-23 23:48:35 +00:00
|
|
|
int ret;
|
|
|
|
int partition;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = open_map_of(dev_name, O_RDWR, &map, &partition);
|
2006-06-01 23:36:57 +00:00
|
|
|
if (ret < 0)
|
2004-12-27 21:39:19 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = emile_map_read(map, partition - 1);
|
2004-12-23 23:48:35 +00:00
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_map_set_partition_type(map, "Apple_HFS");
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
2004-12-23 23:48:35 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_map_partition_set_bootable(map, 1);
|
2004-12-23 23:48:35 +00:00
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
ret = emile_map_write(map, partition - 1);
|
|
|
|
if (ret == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
emile_map_close(map);
|
|
|
|
|
2004-12-23 23:48:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-12-07 00:39:39 +00:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2007-05-12 19:31:52 +00:00
|
|
|
char *backup_path = PREFIX "/boot/emile/bootblock.backup";
|
|
|
|
char *config_path = PREFIX "/boot/emile/emile.conf";
|
|
|
|
char *partition;
|
|
|
|
char *first_path;
|
|
|
|
char *second_path;
|
|
|
|
char *kernel_path;
|
|
|
|
char *initrd_path;
|
|
|
|
char *kernel_map_path;
|
|
|
|
char *initrd_map_path;
|
|
|
|
char *append_string;
|
2004-12-22 00:29:25 +00:00
|
|
|
int ret;
|
2004-12-16 23:27:50 +00:00
|
|
|
int c;
|
|
|
|
int option_index = 0;
|
2004-12-23 23:24:06 +00:00
|
|
|
int fd;
|
2004-12-27 21:39:19 +00:00
|
|
|
int action = ACTION_NONE;
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config *config;
|
2004-12-16 23:27:50 +00:00
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
c = getopt_long(argc, argv, "vhtc:", long_options, &option_index);
|
2004-12-16 23:27:50 +00:00
|
|
|
if (c == -1)
|
|
|
|
break;
|
|
|
|
switch(c)
|
|
|
|
{
|
2004-12-22 00:29:25 +00:00
|
|
|
case ARG_VERBOSE:
|
2004-12-16 23:27:50 +00:00
|
|
|
verbose++;
|
|
|
|
break;
|
2004-12-22 00:29:25 +00:00
|
|
|
case ARG_HELP:
|
2004-12-16 23:27:50 +00:00
|
|
|
usage(argc, argv);
|
|
|
|
return 0;
|
2004-12-27 21:39:19 +00:00
|
|
|
case ARG_SCANBUS:
|
|
|
|
action |= ACTION_SCANBUS;
|
|
|
|
break;
|
2004-12-22 00:29:25 +00:00
|
|
|
case ARG_SET_HFS:
|
2004-12-27 21:39:19 +00:00
|
|
|
action |= ACTION_SET_HFS;
|
2004-12-22 00:29:25 +00:00
|
|
|
break;
|
2004-12-27 21:39:19 +00:00
|
|
|
case ARG_RESTORE:
|
|
|
|
action |= ACTION_RESTORE;
|
|
|
|
if (optarg != NULL)
|
|
|
|
backup_path = optarg;
|
2004-12-22 00:29:25 +00:00
|
|
|
break;
|
2004-12-23 23:24:06 +00:00
|
|
|
case ARG_BACKUP:
|
2004-12-27 21:39:19 +00:00
|
|
|
action |= ACTION_BACKUP;
|
2004-12-22 23:14:07 +00:00
|
|
|
if (optarg != NULL)
|
|
|
|
backup_path = optarg;
|
|
|
|
break;
|
|
|
|
case ARG_TEST:
|
2004-12-27 21:39:19 +00:00
|
|
|
action |= ACTION_TEST;
|
2004-12-22 00:29:25 +00:00
|
|
|
break;
|
2007-05-11 12:21:58 +00:00
|
|
|
case ARG_CONFIG:
|
|
|
|
action |= ACTION_CONFIG;
|
|
|
|
config_path = optarg;
|
|
|
|
break;
|
2004-12-27 01:32:03 +00:00
|
|
|
default:
|
|
|
|
fprintf(stderr, "ERROR: unknown option %s (%d, %c)\n",
|
|
|
|
argv[optind], c, c);
|
|
|
|
return 1;
|
2004-12-16 23:27:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ACTION_SCANBUS) {
|
2004-12-23 23:24:06 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ~ACTION_SCANBUS) {
|
2004-12-23 23:24:06 +00:00
|
|
|
fprintf(stderr,
|
2004-12-27 21:39:19 +00:00
|
|
|
"ERROR: \"--scanbus\" cannot be used with other arguments\n");
|
2004-12-23 23:24:06 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
scanbus();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
/* read config file */
|
2004-12-27 21:39:19 +00:00
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
config = emile_config_open(config_path);
|
|
|
|
if (config == NULL)
|
2004-12-22 00:29:25 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr, "ERROR: cannot open config file %s\n", config_path);
|
|
|
|
return 2;
|
2004-12-27 21:39:19 +00:00
|
|
|
}
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = emile_config_get(config, CONFIG_PARTITION, &partition);
|
|
|
|
if (ret == -1)
|
2004-12-23 23:24:06 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr, "ERROR: you must specify in %s a partition to set\n"
|
|
|
|
" EMILE bootblock\n", config_path);
|
2004-12-23 23:24:06 +00:00
|
|
|
fprintf(stderr,
|
|
|
|
" you can have the list of available partitions with \"--scanbus\".\n");
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-23 23:24:06 +00:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ACTION_RESTORE)
|
|
|
|
{
|
2007-05-09 19:26:48 +00:00
|
|
|
char* new_name;
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
if (action & ~ACTION_RESTORE)
|
2004-12-27 21:39:19 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: \"--restore\" cannot be used with other arguments\n");
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-27 21:39:19 +00:00
|
|
|
return 13;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = restore_bootblock(partition, backup_path);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot restore bootblock %s from %s\n",
|
|
|
|
partition, backup_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-27 21:39:19 +00:00
|
|
|
return 14;
|
|
|
|
}
|
|
|
|
printf("Bootblock restore successfully done.\n");
|
|
|
|
|
2007-05-09 19:26:48 +00:00
|
|
|
/* rename backup file to .old */
|
|
|
|
|
|
|
|
new_name = (char*)malloc(strlen(backup_path) + 4 + 1);
|
|
|
|
|
|
|
|
sprintf(new_name, "%s.old", new_name);
|
|
|
|
|
|
|
|
unlink(new_name);
|
|
|
|
rename(backup_path, new_name);
|
|
|
|
|
|
|
|
free(new_name);
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-27 21:39:19 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = check_has_apple_driver(partition);
|
|
|
|
if (ret != -2)
|
2004-12-23 23:24:06 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
if (ret == -1)
|
2004-12-27 21:39:19 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr, "ERROR: cannot check if Apple_Driver exists\n");
|
|
|
|
fprintf(stderr, " you should try as root\n");
|
|
|
|
if ((action & ACTION_TEST) == 0)
|
|
|
|
{
|
|
|
|
emile_config_close(config);
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ret == 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: to be bootable a disk must have an Apple Driver on it\n");
|
|
|
|
fprintf(stderr,
|
|
|
|
" You must partition this disk with Apple Disk utility\n");
|
|
|
|
fprintf(stderr,
|
|
|
|
" or wait a release of EMILE allowing you to add this driver\n");
|
|
|
|
if ((action & ACTION_TEST) == 0)
|
|
|
|
{
|
|
|
|
emile_config_close(config);
|
|
|
|
return 5;
|
|
|
|
}
|
2004-12-27 21:39:19 +00:00
|
|
|
}
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = check_is_hfs(partition);
|
2004-12-27 21:39:19 +00:00
|
|
|
if (ret == -1)
|
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot check if partition is Apple_HFS\n");
|
|
|
|
fprintf(stderr, " you should try as root\n");
|
|
|
|
if ((action & ACTION_TEST) == 0)
|
|
|
|
{
|
|
|
|
emile_config_close(config);
|
|
|
|
return 6;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( (ret == 0) && !(action & ACTION_SET_HFS) )
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: to be bootable a partition must be of type Apple_HFS\n");
|
|
|
|
fprintf(stderr,
|
|
|
|
" you can change it to Apple_HFS using \"--set-hfs\" argument\n");
|
|
|
|
if ((action & ACTION_TEST) == 0)
|
|
|
|
{
|
|
|
|
emile_config_close(config);
|
|
|
|
return 7;
|
|
|
|
}
|
2004-12-27 21:39:19 +00:00
|
|
|
}
|
2004-12-22 23:14:07 +00:00
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = check_is_EMILE_bootblock(partition);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot check bootblock type\n");
|
|
|
|
fprintf(stderr, " you should try as root\n");
|
|
|
|
if ((action & ACTION_TEST) == 0)
|
|
|
|
{
|
|
|
|
emile_config_close(config);
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
}
|
2006-06-01 23:36:57 +00:00
|
|
|
}
|
2004-12-27 22:32:18 +00:00
|
|
|
if ( (ret == 0) && ((action & ACTION_BACKUP) == 0) )
|
2004-12-22 23:14:07 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: there is already a bootblock on \"%s\"\n", partition);
|
|
|
|
fprintf(stderr,
|
2004-12-23 23:24:06 +00:00
|
|
|
" you must use \"--backup\" to save it\n");
|
2004-12-27 21:39:19 +00:00
|
|
|
if ((action & ACTION_TEST) == 0)
|
2007-05-11 12:21:58 +00:00
|
|
|
{
|
|
|
|
emile_config_close(config);
|
2004-12-23 23:24:06 +00:00
|
|
|
return 9;
|
2007-05-11 12:21:58 +00:00
|
|
|
}
|
2004-12-22 23:14:07 +00:00
|
|
|
}
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ACTION_BACKUP)
|
2004-12-22 23:14:07 +00:00
|
|
|
{
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ACTION_TEST)
|
2004-12-22 23:14:07 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr,
|
2004-12-23 23:24:06 +00:00
|
|
|
"ERROR: \"--backup\" cannot be used with \"--test\"\n");
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-23 23:24:06 +00:00
|
|
|
return 13;
|
2004-12-22 23:14:07 +00:00
|
|
|
}
|
|
|
|
|
2004-12-23 22:35:36 +00:00
|
|
|
ret = backup_bootblock(partition, backup_path);
|
2004-12-22 23:14:07 +00:00
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot backup bootblock %s to %s\n",
|
|
|
|
partition, backup_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-23 23:24:06 +00:00
|
|
|
return 14;
|
2004-12-22 23:14:07 +00:00
|
|
|
}
|
|
|
|
printf("Bootblock backup successfully done.\n");
|
2004-12-22 00:29:25 +00:00
|
|
|
}
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = emile_config_get(config, CONFIG_FIRST_LEVEL, &first_path);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot read first level path from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = emile_config_get(config, CONFIG_SECOND_LEVEL, &second_path);
|
|
|
|
if (ret == -1)
|
2005-11-29 23:30:50 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr, "ERROR: cannot read second level path from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("partition: %s\n", partition);
|
|
|
|
printf("first: %s\n", first_path);
|
|
|
|
printf("second: %s\n", second_path);
|
|
|
|
|
|
|
|
ret = emile_config_read_first_entry(config);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot read first kernel entry from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = emile_config_get(config, CONFIG_KERNEL, &kernel_path);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot read kernel path from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!emile_is_url(kernel_path))
|
|
|
|
{
|
|
|
|
ret = emile_config_get(config, CONFIG_KERNEL_MAP, &kernel_map_path);
|
|
|
|
if (ret == -1)
|
2005-11-29 23:30:50 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
char *a = strdup(kernel_path);
|
|
|
|
char *b = strdup(kernel_path);
|
|
|
|
char *base = basename(a);
|
|
|
|
char *dir = dirname(b);
|
|
|
|
kernel_map_path = (char*)malloc(strlen(kernel_path) + 6);
|
|
|
|
if (kernel_map_path == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot allocate memory\n");
|
|
|
|
emile_config_close(config);
|
|
|
|
return 15;
|
|
|
|
}
|
|
|
|
sprintf(kernel_map_path, "%s/.%s.map", dir, base);
|
|
|
|
free(a);
|
|
|
|
free(b);
|
2005-11-29 23:30:50 +00:00
|
|
|
}
|
2007-05-04 20:45:21 +00:00
|
|
|
} else
|
|
|
|
kernel_map_path = kernel_path;
|
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = emile_config_get(config, CONFIG_INITRD, &initrd_path);
|
|
|
|
if (ret == -1)
|
2007-05-04 20:45:21 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
fprintf(stderr, "ERROR: cannot read initrd path from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!emile_is_url(initrd_path))
|
|
|
|
{
|
|
|
|
ret = emile_config_get(config, CONFIG_INITRD_MAP, &initrd_map_path);
|
|
|
|
if (ret == -1)
|
2007-05-04 20:45:21 +00:00
|
|
|
{
|
2007-05-11 12:21:58 +00:00
|
|
|
char *a = strdup(initrd_path);
|
|
|
|
char *b = strdup(initrd_path);
|
|
|
|
char *base = basename(a);
|
|
|
|
char *dir = dirname(b);
|
|
|
|
initrd_map_path = (char*)malloc(strlen(initrd_path) + 6);
|
|
|
|
if (initrd_map_path == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot allocate memory\n");
|
|
|
|
emile_config_close(config);
|
|
|
|
return 15;
|
|
|
|
}
|
|
|
|
sprintf(initrd_map_path, "%s/.%s.map", dir, base);
|
|
|
|
free(a);
|
|
|
|
free(b);
|
2007-05-04 20:45:21 +00:00
|
|
|
}
|
|
|
|
} else
|
|
|
|
initrd_map_path = initrd_path;
|
2005-11-29 23:30:50 +00:00
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
ret = emile_config_get(config, CONFIG_ARGS, &append_string);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot read kernel parameters from %s\n",
|
|
|
|
config_path);
|
|
|
|
emile_config_close(config);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2004-12-16 23:27:50 +00:00
|
|
|
printf("kernel: %s\n", kernel_path);
|
2004-12-23 23:24:06 +00:00
|
|
|
printf("append: %s\n", append_string);
|
2007-05-11 12:21:58 +00:00
|
|
|
printf("kernel map file: %s\n", kernel_map_path);
|
|
|
|
printf("initrd: %s\n", initrd_path);
|
|
|
|
printf("initrd map file: %s\n", initrd_map_path);
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-26 20:12:44 +00:00
|
|
|
/* set kernel info into second level */
|
2004-12-07 00:39:39 +00:00
|
|
|
|
2004-12-26 20:12:44 +00:00
|
|
|
fd = open(second_path, O_RDWR);
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "ERROR: cannot open \"%s\"\n",
|
|
|
|
second_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 16;
|
2004-12-26 20:12:44 +00:00
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if ((action & ACTION_TEST) == 0)
|
2004-12-26 20:12:44 +00:00
|
|
|
{
|
2005-11-29 23:30:50 +00:00
|
|
|
struct emile_container *container;
|
2007-02-24 14:02:40 +00:00
|
|
|
unsigned short unit_id;
|
2007-05-04 20:45:21 +00:00
|
|
|
char kernel_map_info[64];
|
|
|
|
char initrd_map_info[64];
|
2006-09-20 23:28:53 +00:00
|
|
|
int drive, second, size;
|
2005-11-29 00:15:30 +00:00
|
|
|
|
2007-05-04 20:45:21 +00:00
|
|
|
/* get block mapping of kernel in filesystem */
|
|
|
|
|
|
|
|
container = emile_second_create_mapfile(&unit_id, kernel_map_path, kernel_path);
|
2005-11-29 23:30:50 +00:00
|
|
|
if (container == NULL)
|
2004-12-23 23:24:06 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot set \"%s\" information in \"%s\".\n",
|
2007-05-04 20:45:21 +00:00
|
|
|
kernel_path, kernel_map_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 17;
|
2004-12-23 23:24:06 +00:00
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2005-11-29 23:30:50 +00:00
|
|
|
/* set kernel info */
|
|
|
|
|
2007-05-04 20:45:21 +00:00
|
|
|
sprintf(kernel_map_info, "container:(sd%d)0x%x,0x%x", unit_id,
|
2006-06-01 23:36:57 +00:00
|
|
|
container->blocks[0].offset,
|
|
|
|
container->blocks[0].count);
|
2007-05-04 20:45:21 +00:00
|
|
|
free(container);
|
2005-11-29 23:30:50 +00:00
|
|
|
|
2006-06-01 23:36:57 +00:00
|
|
|
/* set second configuration */
|
2005-11-29 23:30:50 +00:00
|
|
|
|
2006-09-20 23:28:53 +00:00
|
|
|
ret = emile_first_get_param(fd, &drive, &second, &size);
|
2006-10-28 22:35:44 +00:00
|
|
|
if (ret == EEMILE_UNKNOWN_FIRST)
|
|
|
|
lseek(fd, 0, SEEK_SET);
|
2007-05-04 20:45:21 +00:00
|
|
|
|
|
|
|
/* get block mapping of initrd */
|
|
|
|
|
|
|
|
container = emile_second_create_mapfile(&unit_id, initrd_map_path, initrd_path);
|
|
|
|
if (container == NULL)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot set \"%s\" information in \"%s\".\n",
|
|
|
|
initrd_path, initrd_map_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2007-05-04 20:45:21 +00:00
|
|
|
return 17;
|
|
|
|
}
|
|
|
|
sprintf(initrd_map_info, "container:(sd%d)0x%x,0x%x", unit_id,
|
|
|
|
container->blocks[0].offset,
|
|
|
|
container->blocks[0].count);
|
|
|
|
free(container);
|
|
|
|
ret = emile_second_set_param(fd, kernel_map_info,
|
|
|
|
append_string, initrd_map_info);
|
2006-10-28 22:35:44 +00:00
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot set \"%s\" information in \"%s\".\n",
|
2007-05-04 20:45:21 +00:00
|
|
|
initrd_path, initrd_map_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2006-10-28 22:35:44 +00:00
|
|
|
return 18;
|
|
|
|
}
|
2004-12-26 20:12:44 +00:00
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-26 20:12:44 +00:00
|
|
|
close(fd);
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-26 20:12:44 +00:00
|
|
|
/* set second info in first level */
|
|
|
|
|
|
|
|
fd = open(first_path, O_RDWR);
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot open \"%s\".\n", first_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 20;
|
2004-12-26 20:12:44 +00:00
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if ((action & ACTION_TEST) == 0)
|
2004-12-26 20:12:44 +00:00
|
|
|
{
|
2004-12-23 23:24:06 +00:00
|
|
|
ret = emile_first_set_param_scsi(fd, second_path);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot set \"%s\" information into \"%s\".\n",
|
|
|
|
second_path, first_path);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 21;
|
2004-12-23 23:24:06 +00:00
|
|
|
}
|
2004-12-26 20:12:44 +00:00
|
|
|
}
|
2004-12-23 23:24:06 +00:00
|
|
|
|
2004-12-26 20:12:44 +00:00
|
|
|
close(fd);
|
2004-12-23 23:24:06 +00:00
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if ((action & ACTION_TEST) == 0)
|
2004-12-26 20:12:44 +00:00
|
|
|
{
|
2004-12-23 23:24:06 +00:00
|
|
|
/* copy first level to boot block */
|
|
|
|
|
|
|
|
ret = copy_file_to_bootblock(first_path, partition);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"ERROR: cannot write \"%s\" to bootblock of \"%s\".\n",
|
|
|
|
first_path, partition);
|
|
|
|
fprintf(stderr,
|
|
|
|
" %s\n", strerror(errno));
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 22;
|
2004-12-23 23:24:06 +00:00
|
|
|
}
|
2004-12-23 23:48:35 +00:00
|
|
|
|
|
|
|
/* set HFS if needed */
|
|
|
|
|
2004-12-27 21:39:19 +00:00
|
|
|
if (action & ACTION_SET_HFS)
|
2004-12-23 23:48:35 +00:00
|
|
|
{
|
|
|
|
ret = set_HFS(partition);
|
|
|
|
if (ret == -1)
|
|
|
|
{
|
|
|
|
fprintf( stderr,
|
|
|
|
"ERROR: cannot set partition type of \"%s\" to Apple_HFS.\n"
|
|
|
|
, partition);
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2005-11-29 23:30:50 +00:00
|
|
|
return 23;
|
2004-12-23 23:48:35 +00:00
|
|
|
}
|
|
|
|
}
|
2004-12-23 23:24:06 +00:00
|
|
|
}
|
2004-12-13 02:12:27 +00:00
|
|
|
|
2007-05-11 12:21:58 +00:00
|
|
|
emile_config_close(config);
|
2004-12-03 00:29:29 +00:00
|
|
|
return 0;
|
|
|
|
}
|