/* ----------------------------------------------------------------------- * * * Copyright 2009 Pierre-Alexandre Meyer * * Some parts borrowed from chain.c32: * * Copyright 2003-2009 H. Peter Anvin - All Rights Reserved * Copyright 2009 Intel Corporation; author: H. Peter Anvin * * This file is part of Syslinux, and is made available under * the terms of the GNU General Public License version 2. * * ----------------------------------------------------------------------- */ #include <stdlib.h> #include <disk/common.h> #include <disk/geom.h> #include <disk/msdos.h> #include <disk/partition.h> #include <disk/read.h> static int is_extended_partition(struct part_entry *ptab) { return (ptab->ostype == 0x05 || ptab->ostype == 0x0f || ptab->ostype == 0x85); } static int msdos_magic_present(const char *ptab) { return (*(uint16_t *) (ptab + 0x1fe) == 0xaa55); } /** * process_extended_partition - execute a callback for each partition contained listed in an ebr * @drive_info: driveinfo struct describing the drive * @partition_offset: Absolute start (lba) of the extended partition * @ebr_offset: Relative start (lba) of the current ebr processed within * the extended partition * @callback: Callback to execute * @error: Buffer for I/O errors * @nb_part_seen: Number of partitions found on the disk so far **/ static int process_extended_partition(struct driveinfo *drive_info, const int partition_offset, const int ebr_offset, p_callback callback, int nb_part_seen) { int status = 0; /* The ebr is located at the first sector of the extended partition */ char *ebr = malloc(SECTOR * sizeof(char)); if (read_sectors(drive_info, ebr, partition_offset + ebr_offset, 1) == -1) goto abort; /* Check msdos magic signature */ if (!msdos_magic_present(ebr)) goto abort; struct part_entry *ptab = (struct part_entry *)(ebr + PARTITION_TABLES_OFFSET); for (int i = 0; i < 4; i++) { if (status == -1) goto abort; if (!is_extended_partition(&ptab[i])) { /* * This EBR partition table entry points to the * logical partition associated to that EBR */ int logical_partition_start = ebr_offset + ptab[i].start_lba; /* Last EBR in the extended partition? */ if (!logical_partition_start) continue; /* * Check for garbage: * 3rd and 4th entries in an EBR should be zero * Some (malformed) partitioning software still add some * data partitions there. */ if (ptab[i].start_lba <= 0 || ptab[i].length <= 0) continue; nb_part_seen++; callback(drive_info, &ptab[i], partition_offset + logical_partition_start, nb_part_seen); } else status = process_extended_partition(drive_info, partition_offset, ptab[i].start_lba, callback, nb_part_seen); } free(ebr); return 0; abort: free(ebr); return -1; } /** * process_mbr - execute a callback for each partition contained in an {m,e}br * @drive_info: driveinfo struct describing the drive * @ptab: Pointer to the partition table * @callback: Callback to execute **/ static int process_mbr(struct driveinfo *drive_info, struct part_entry *ptab, p_callback callback) { int status = 0; for (int i = 0; i < 4; i++) { if (status == -1) return -1; if (ptab[i].start_sect > 0) { if (is_extended_partition(&ptab[i])) { callback(drive_info, &ptab[i], ptab[i].start_lba, i + 1); status = process_extended_partition(drive_info, ptab[i].start_lba, 0, callback, 4); } else callback(drive_info, &ptab[i], ptab[i].start_lba, i + 1); } } return 0; } /** * parse_partition_table - execute a callback for each partition entry * @d: driveinfo struct describing the drive * @callback: Callback to execute * * The signature of the callback should be the following: * * void callback(struct driveinfo *drive_info, * struct part_entry *ptab, * int offset_root, * int nb_part_seen) **/ int parse_partition_table(struct driveinfo *d, p_callback callback) { char *mbr = malloc(SECTOR * sizeof(char)); if (read_mbr(d->disk, mbr) == -1) return -1; else { /* Check msdos magic signature */ if (!msdos_magic_present(mbr)) return -1; struct part_entry *ptab = (struct part_entry *)(mbr + PARTITION_TABLES_OFFSET); return process_mbr(d, ptab, callback); } }