2020-06-30 22:02:55 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
int main(int argc, char *argv[]) {
|
2020-09-15 13:16:31 +03:00
|
|
|
FILE *bootloader_file, *device;
|
|
|
|
uint8_t *bootloader_img;
|
|
|
|
uint8_t orig_mbr[70], timestamp[6];
|
2020-11-05 19:00:15 +03:00
|
|
|
uint32_t stage2_sect, sect_size;
|
2020-09-15 13:16:31 +03:00
|
|
|
|
|
|
|
if (argc < 3) {
|
2020-11-05 19:00:15 +03:00
|
|
|
printf("Usage: %s <bootloader image> <device> [<stage2 start sector> <sector size>]\n", argv[0]);
|
2020-09-15 13:16:31 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bootloader_file = fopen(argv[1], "rb");
|
|
|
|
if (bootloader_file == NULL) {
|
|
|
|
perror("Error: ");
|
2020-06-30 22:02:55 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-09-15 13:16:31 +03:00
|
|
|
// The bootloader image is 64 sectors (32k)
|
|
|
|
bootloader_img = malloc(64 * 512);
|
|
|
|
if (bootloader_img == NULL) {
|
|
|
|
perror("Error: ");
|
|
|
|
fclose(bootloader_file);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load in bootloader image
|
|
|
|
fseek(bootloader_file, 0, SEEK_SET);
|
|
|
|
fread(bootloader_img, 64, 512, bootloader_file);
|
|
|
|
fclose(bootloader_file);
|
|
|
|
|
|
|
|
device = fopen(argv[2], "r+b");
|
2020-06-30 22:02:55 +03:00
|
|
|
if (device == NULL) {
|
|
|
|
perror("Error: ");
|
2020-09-15 13:16:31 +03:00
|
|
|
free(bootloader_img);
|
2020-06-30 22:02:55 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-11-05 19:00:15 +03:00
|
|
|
stage2_sect = 0;
|
|
|
|
sect_size = 512;
|
2020-09-15 13:16:31 +03:00
|
|
|
if (argc >= 4)
|
|
|
|
sscanf(argv[3], "%" SCNu32, &stage2_sect);
|
2020-11-05 19:00:15 +03:00
|
|
|
if (argc >= 5)
|
|
|
|
sscanf(argv[4], "%" SCNu32, §_size);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
2020-09-06 09:52:50 +03:00
|
|
|
// Save original timestamp
|
|
|
|
fseek(device, 218, SEEK_SET);
|
|
|
|
fread(timestamp, 1, 6, device);
|
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
// Save the original partition table of the device
|
2020-09-06 09:52:50 +03:00
|
|
|
fseek(device, 440, SEEK_SET);
|
|
|
|
fread(orig_mbr, 1, 70, device);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
// Write the bootsector from the bootloader to the device
|
2020-06-30 22:02:55 +03:00
|
|
|
fseek(device, 0, SEEK_SET);
|
2020-09-15 13:16:31 +03:00
|
|
|
fwrite(&bootloader_img[0], 1, 512, device);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
// Write the rest of stage 2 to the device
|
2020-11-05 19:00:15 +03:00
|
|
|
fseek(device, stage2_sect * sect_size, SEEK_SET);
|
|
|
|
fwrite(&bootloader_img[0], 64, 512, device);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
// Hardcode in the bootsector the location of stage 2
|
2020-06-30 22:02:55 +03:00
|
|
|
fseek(device, 0x1b0, SEEK_SET);
|
2020-09-04 06:21:58 +03:00
|
|
|
fwrite(&stage2_sect, 1, sizeof(uint32_t), device);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
2020-09-06 09:52:50 +03:00
|
|
|
// Write back timestamp
|
|
|
|
fseek(device, 218, SEEK_SET);
|
|
|
|
fwrite(timestamp, 1, 6, device);
|
|
|
|
|
2020-09-04 06:21:58 +03:00
|
|
|
// Write back the saved partition table to the device
|
2020-09-06 09:52:50 +03:00
|
|
|
fseek(device, 440, SEEK_SET);
|
|
|
|
fwrite(orig_mbr, 1, 70, device);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
|
|
|
fclose(device);
|
2020-09-15 13:16:31 +03:00
|
|
|
free(bootloader_img);
|
2020-06-30 22:02:55 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|