/* * * (c) 2004-2007 Laurent Vivier * */ #include #include #include #include #include #include #include #include #include #include #include #include "libemile.h" #include "libmap.h" #include "device.h" int verbose = 0; extern void scanbus(void); static enum { ACTION_NONE = 0x00000000, ACTION_SCANBUS = 0x00000001, ACTION_SET_HFS = 0x00000002, ACTION_RESTORE = 0x00000004, ACTION_BACKUP = 0x00000008, ACTION_TEST = 0x00000010, ACTION_CONFIG = 0x00000020, } action = ACTION_NONE; enum { ARG_NONE = 0, ARG_SCANBUS, ARG_SET_HFS, ARG_RESTORE, ARG_BACKUP, ARG_VERBOSE ='v', ARG_TEST = 't', ARG_HELP = 'h', ARG_CONFIG = 'c', }; static struct option long_options[] = { {"scanbus", 0, NULL, ARG_SCANBUS }, {"set-hfs", 0, NULL, ARG_SET_HFS }, {"restore", 2, NULL, ARG_RESTORE }, {"backup", 2, NULL, ARG_BACKUP }, {"verbose", 0, NULL, ARG_VERBOSE }, {"help", 0, NULL, ARG_HELP }, {"test", 0, NULL, ARG_TEST }, {"config", 1, NULL, ARG_CONFIG }, {NULL, 0, NULL, 0 } }; static void usage(int argc, char** argv) { 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"); fprintf(stderr," --restore[=FILE] restore boot block from FILE\n"); 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"); fprintf(stderr," -c, --config FILE use config file FILE\n"); fprintf(stderr, "!!! USE WITH CAUTION AND AT YOUR OWN RISK !!!\n"); fprintf(stderr, "\nbuild: \n%s\n", SIGNATURE); } static int open_map_of( char *dev_name, int flags, map_t **map, int *partition) { int ret; int disk; char disk_name[16]; int driver; device_io_t device; ret = emile_scsi_get_rdev(dev_name, &driver, &disk, partition); if (ret == -1) return -2; emile_get_dev_name(disk_name, driver, disk, 0); device_sector_size = 512; device.data = (void*)device_open(disk_name, flags); device.write_sector = (stream_read_sector_t)device_write_sector; device.read_sector = (stream_read_sector_t)device_read_sector; device.close = (stream_close_t)device_close; device.get_blocksize = (stream_get_blocksize_t)device_get_blocksize; *map = map_open(&device); if (*map == NULL) return -1; return 0; } static int check_has_apple_driver(char *dev_name) { map_t *map; int partition; int ret; ret = open_map_of(dev_name, O_RDONLY, &map, &partition); if (ret < 0) return ret; ret = map_has_apple_driver(map); map_close(map); return ret; } static int check_is_hfs(char *dev_name) { map_t *map; int ret; int partition; char *part_type; ret = open_map_of(dev_name, O_RDONLY, &map, &partition); if (ret < 0) return ret; ret = map_read(map, partition - 1); if (ret == -1) return -1; part_type = map_get_partition_type(map); ret = (strcmp("Apple_HFS", part_type) == 0); map_close(map); return ret; } static int check_is_EMILE_bootblock(char *dev_name) { map_t *map; int ret; int partition; char bootblock[BOOTBLOCK_SIZE]; int bootblock_type; ret = open_map_of(dev_name, O_RDONLY, &map, &partition); if (ret < 0) return ret; ret = map_read(map, partition - 1); if (ret == -1) return -1; ret = map_bootblock_read(map, bootblock); if (ret == -1) return -1; bootblock_type = map_bootblock_get_type(bootblock); map_close(map); return EMILE_BOOTBLOCK == bootblock_type; } static int backup_bootblock(char *dev_name, char *filename) { map_t *map; int ret; int partition; char bootblock[BOOTBLOCK_SIZE]; int fd; ret = open_map_of(dev_name, O_RDONLY, &map, &partition); if (ret < 0) return ret; ret = map_read(map, partition - 1); if (ret == -1) return -1; ret = map_bootblock_read(map, bootblock); if (ret == -1) return -1; 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; } static int restore_bootblock(char *dev_name, char *filename) { map_t *map; int ret; int partition; char bootblock[BOOTBLOCK_SIZE]; int fd; if (!check_is_EMILE_bootblock(dev_name)) { fprintf(stderr, "ERROR: cannot restore bootblock over non-EMILE bootblock\n"); return -1; } /* read bootblock */ fd = open(filename, O_RDONLY); if (fd == -1) return -1; ret = read(fd, bootblock, BOOTBLOCK_SIZE); if (ret != BOOTBLOCK_SIZE) return -1; close(fd); /* write bootblock */ ret = open_map_of(dev_name, O_RDWR, &map, &partition); if (ret < 0) return -1; ret = map_read(map, partition - 1); if (ret == -1) return -1; ret = map_bootblock_write(map, bootblock); if (ret == -1) return -1; map_close(map); return 0; } static int copy_file_to_bootblock(char* first_path, char* dev_name) { map_t *map; int ret; int partition; char bootblock[BOOTBLOCK_SIZE]; int fd; /* read first level */ fd = open(first_path, O_RDONLY); if (fd == -1) return -1; ret = read(fd, bootblock, BOOTBLOCK_SIZE); if (ret != BOOTBLOCK_SIZE) return -1; close(fd); /* write bootblock to partition */ ret = open_map_of(dev_name, O_RDWR, &map, &partition); if (ret < 0) return -1; ret = map_read(map, partition - 1); if (ret == -1) return -1; ret = map_bootblock_write(map, bootblock); if (ret == -1) return -1; map_close(map); return 0; } static int set_HFS(char *dev_name) { map_t *map; int ret; int partition; ret = open_map_of(dev_name, O_RDWR, &map, &partition); if (ret < 0) return -1; ret = map_read(map, partition - 1); if (ret == -1) return -1; ret = map_set_partition_type(map, "Apple_HFS"); if (ret == -1) return -1; ret = map_partition_set_bootable(map, 1); if (ret == -1) return -1; ret = map_write(map, partition - 1); if (ret == -1) return -1; map_close(map); return 0; } static char *get_map_name(char *filename) { char *a, *b; char *base, *dir; char *map_name; a = strdup(filename); base = basename(a); b = strdup(filename); dir = dirname(b); map_name = (char*)malloc(strlen(filename) + 6); if (map_name == NULL) return NULL; sprintf(map_name, "%s/.%s.map", dir, base); free(a); free(b); return map_name; } static int add_file(int8_t *configuration, char *index, char *property, char *path, char *map_path) { struct emile_container *container; unsigned short unit_id; char map_info[64]; if (emile_is_url(path)) { if (verbose) printf(" %s %s\n", property, path); config_set_indexed_property(configuration, "title", index, property, path); return 0; } if ((action & ACTION_TEST) != 0) { map_path = tempnam(NULL, "emile-map_path-"); } else if (map_path == NULL) { map_path = get_map_name(path); if (map_path == NULL) return -1; } else map_path = strdup(map_path); /* get block mapping of kernel in filesystem */ container = emile_second_create_mapfile(&unit_id, map_path, path); if (container == NULL) { free(map_path); return -1; } sprintf(map_info, "container:(sd%d)0x%x,0x%x", unit_id, container->blocks[0].offset, container->blocks[0].count); free(container); if (verbose) printf(" kernel %s (%s = %s)\n", path, map_path, map_info); config_set_indexed_property(configuration, "title", index, property, map_info); if ((action & ACTION_TEST) != 0) fprintf(stderr, "WARNING: test mode, you must remove manually %s\n", map_path); free(map_path); return 0; } static int8_t *set_config(char *config_path) { int8_t *configuration; int ret; configuration = (int8_t*)malloc(65536); if (configuration == NULL) { fprintf(stderr, "ERROR: cannot allocate memory for configuration\n"); return NULL; } configuration[0] = 0; ret = add_file(configuration, NULL, "configuration", config_path, NULL); if (ret == -1) { free(configuration); fprintf(stderr, "ERROR: cannot set configuration in %s\n", config_path); return NULL; } return configuration; } int main(int argc, char **argv) { char property[1024]; char *backup_path = PREFIX "/boot/emile/bootblock.backup"; char *config_path = PREFIX "/boot/emile/emile.conf"; char *first_path; char *second_path; int ret; int c; char *partition; int option_index = 0; int drive, second, size; int fd; int8_t *configuration; int8_t *config; struct stat st; while(1) { c = getopt_long(argc, argv, "vhtc:", long_options, &option_index); if (c == -1) break; switch(c) { case ARG_VERBOSE: verbose++; break; case ARG_HELP: usage(argc, argv); return 0; case ARG_SCANBUS: action |= ACTION_SCANBUS; break; case ARG_SET_HFS: action |= ACTION_SET_HFS; break; case ARG_RESTORE: action |= ACTION_RESTORE; if (optarg != NULL) backup_path = optarg; break; case ARG_BACKUP: action |= ACTION_BACKUP; if (optarg != NULL) backup_path = optarg; break; case ARG_TEST: action |= ACTION_TEST; break; case ARG_CONFIG: action |= ACTION_CONFIG; config_path = optarg; break; default: fprintf(stderr, "ERROR: unknown option %s (%d, %c)\n", argv[optind], c, c); return 1; } } if (action & ACTION_SCANBUS) { if (action & ~ACTION_SCANBUS) { fprintf(stderr, "ERROR: \"--scanbus\" cannot be used with other arguments\n"); return 1; } scanbus(); return 0; } /* read config file */ fd = open(config_path, O_RDONLY); if (fd == -1) { fprintf(stderr, "ERROR: cannot open config file %s\n", config_path); return 2; } if (fstat(fd, &st) == -1) { fprintf(stderr, "ERROR: cannot fstat file %s\n", config_path); return 2; } config = (int8_t*)malloc(st.st_size); if (config == NULL) { fprintf(stderr, "ERROR: cannot malloc %s\n", config_path); return 2; } if (read(fd, config, st.st_size) != st.st_size) { fprintf(stderr, "ERROR: cannot read() %s\n", config_path); return 5; } close(fd); /* get partition */ ret = config_get_property(config, "partition", property); if (ret == -1) { fprintf(stderr, "ERROR: you must specify in %s a partition to set\n" " EMILE bootblock\n", config_path); fprintf(stderr, " you can have the list of available partitions with \"--scanbus\".\n"); free(config); return 3; } partition = strdup(property); if (action & ACTION_RESTORE) { char* new_name; if (action & ~ACTION_RESTORE) { fprintf(stderr, "ERROR: \"--restore\" cannot be used with other arguments\n"); free(config); return 13; } ret = restore_bootblock(partition, backup_path); if (ret == -1) { fprintf(stderr, "ERROR: cannot restore bootblock %s from %s\n", partition, backup_path); free(config); return 14; } printf("Bootblock restore successfully done.\n"); /* 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); free(config); return 0; } ret = check_has_apple_driver(partition); if (ret != -2) { if (ret == -1) { fprintf(stderr, "ERROR: cannot check if Apple_Driver exists\n"); fprintf(stderr, " you should try as root\n"); if ((action & ACTION_TEST) == 0) { free(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) { free(config); return 5; } } ret = check_is_hfs(partition); if (ret == -1) { fprintf(stderr, "ERROR: cannot check if partition is Apple_HFS\n"); fprintf(stderr, " you should try as root\n"); if ((action & ACTION_TEST) == 0) { free(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) { free(config); return 7; } } 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) { free(config); return 8; } } } if ( (ret == 0) && ((action & ACTION_BACKUP) == 0) ) { fprintf(stderr, "ERROR: there is already a bootblock on \"%s\"\n", partition); fprintf(stderr, " you must use \"--backup\" to save it\n"); if ((action & ACTION_TEST) == 0) { free(config); return 9; } } if (action & ACTION_BACKUP) { if (action & ACTION_TEST) { fprintf(stderr, "ERROR: \"--backup\" cannot be used with \"--test\"\n"); free(config); return 13; } ret = backup_bootblock(partition, backup_path); if (ret == -1) { fprintf(stderr, "ERROR: cannot backup bootblock %s to %s\n", partition, backup_path); free(config); return 14; } printf("Bootblock backup successfully done.\n"); } ret = config_get_property(config, "first_level", property); if (ret == -1) return 2; first_path = strdup(property); ret = config_get_property(config, "second_level", property); if (ret == -1) return 2; second_path = strdup(property); fd = open(first_path, O_RDONLY); if (fd == -1) { fprintf(stderr, "ERROR: cannot open \"%s\".\n", first_path); free(config); return 20; } ret = emile_first_get_param(fd, &drive, &second, &size); close(fd); configuration = set_config(config_path); if (ret) return ret; if ((action & ACTION_TEST) == 0) { /* set configuration in second level */ fd = open(second_path, O_RDWR); emile_second_set_configuration(fd, configuration); close(fd); /* set second info in first level */ fd = open(first_path, O_RDWR); 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); free(first_path); free(second_path); free(config); free(configuration); return 21; } free(second_path); close(fd); /* 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)); free(first_path); free(config); free(configuration); return 22; } free(first_path); /* set HFS if needed */ if (action & ACTION_SET_HFS) { ret = set_HFS(partition); if (ret == -1) { fprintf( stderr, "ERROR: cannot set partition type of \"%s\" to Apple_HFS.\n" , partition); free(config); free(configuration); return 23; } } } free(partition); free(config); free(configuration); return 0; }