[e16e8f2] | 1 | /* ----------------------------------------------------------------------- * |
---|
| 2 | * |
---|
| 3 | * Copyright 2009 Pierre-Alexandre Meyer |
---|
| 4 | * |
---|
| 5 | * Some parts borrowed from chain.c32: |
---|
| 6 | * |
---|
| 7 | * Copyright 2003-2009 H. Peter Anvin - All Rights Reserved |
---|
| 8 | * Copyright 2009 Intel Corporation; author: H. Peter Anvin |
---|
| 9 | * |
---|
| 10 | * This file is part of Syslinux, and is made available under |
---|
| 11 | * the terms of the GNU General Public License version 2. |
---|
| 12 | * |
---|
| 13 | * ----------------------------------------------------------------------- */ |
---|
| 14 | |
---|
| 15 | #include <com32.h> |
---|
| 16 | #include <stdlib.h> |
---|
| 17 | #include <string.h> |
---|
| 18 | |
---|
| 19 | #include <disk/common.h> |
---|
| 20 | #include <disk/errno_disk.h> |
---|
| 21 | #include <disk/read.h> |
---|
| 22 | #include <disk/util.h> |
---|
| 23 | #include <disk/write.h> |
---|
| 24 | |
---|
| 25 | /** |
---|
| 26 | * write_sectors - write several sectors from disk |
---|
| 27 | * @drive_info: driveinfo struct describing the disk |
---|
| 28 | * @lba: Position to write |
---|
| 29 | * @data: Buffer to write |
---|
| 30 | * @size: Size of the buffer (number of sectors) |
---|
| 31 | * |
---|
| 32 | * Return the number of sectors write on success or -1 on failure. |
---|
| 33 | * errno_disk contains the error number. |
---|
| 34 | **/ |
---|
| 35 | int write_sectors(const struct driveinfo *drive_info, const unsigned int lba, |
---|
| 36 | const void *data, const int size) |
---|
| 37 | { |
---|
| 38 | com32sys_t inreg, outreg; |
---|
| 39 | struct ebios_dapa *dapa; |
---|
| 40 | void *buf; |
---|
| 41 | int rv = -1; |
---|
| 42 | |
---|
| 43 | buf = lmalloc(size); |
---|
| 44 | if (!buf) |
---|
| 45 | return -1; |
---|
| 46 | |
---|
| 47 | dapa = lmalloc(sizeof(*dapa)); |
---|
| 48 | if (!dapa) |
---|
| 49 | goto out; |
---|
| 50 | |
---|
| 51 | memcpy(buf, data, size); |
---|
| 52 | memset(&inreg, 0, sizeof inreg); |
---|
| 53 | |
---|
| 54 | if (drive_info->ebios) { |
---|
| 55 | dapa->len = sizeof(*dapa); |
---|
| 56 | dapa->count = size; |
---|
| 57 | dapa->off = OFFS(buf); |
---|
| 58 | dapa->seg = SEG(buf); |
---|
| 59 | dapa->lba = lba; |
---|
| 60 | |
---|
| 61 | inreg.esi.w[0] = OFFS(dapa); |
---|
| 62 | inreg.ds = SEG(dapa); |
---|
| 63 | inreg.edx.b[0] = drive_info->disk; |
---|
| 64 | inreg.eax.w[0] = 0x4300; /* Extended write */ |
---|
| 65 | } else { |
---|
| 66 | unsigned int c, h, s; |
---|
| 67 | |
---|
| 68 | if (!drive_info->cbios) { // XXX errno |
---|
| 69 | /* We failed to get the geometry */ |
---|
| 70 | if (lba) |
---|
| 71 | goto out; /* Can only write MBR */ |
---|
| 72 | |
---|
| 73 | s = 1; |
---|
| 74 | h = 0; |
---|
| 75 | c = 0; |
---|
| 76 | } else |
---|
| 77 | lba_to_chs(drive_info, lba, &s, &h, &c); |
---|
| 78 | |
---|
| 79 | // XXX errno |
---|
| 80 | if (s > 63 || h > 256 || c > 1023) |
---|
| 81 | goto out; |
---|
| 82 | |
---|
| 83 | inreg.eax.w[0] = 0x0301; /* Write one sector */ |
---|
| 84 | inreg.ecx.b[1] = c & 0xff; |
---|
| 85 | inreg.ecx.b[0] = s + (c >> 6); |
---|
| 86 | inreg.edx.b[1] = h; |
---|
| 87 | inreg.edx.b[0] = drive_info->disk; |
---|
| 88 | inreg.ebx.w[0] = OFFS(buf); |
---|
| 89 | inreg.es = SEG(buf); |
---|
| 90 | } |
---|
| 91 | |
---|
| 92 | /* Perform the write */ |
---|
| 93 | if (int13_retry(&inreg, &outreg)) { |
---|
| 94 | errno_disk = outreg.eax.b[1]; /* Give up */ |
---|
| 95 | } else |
---|
| 96 | rv = size; |
---|
| 97 | out: |
---|
| 98 | lfree(dapa); |
---|
| 99 | lfree(buf); |
---|
| 100 | return rv; |
---|
| 101 | } |
---|
| 102 | |
---|
| 103 | /** |
---|
| 104 | * write_verify_sectors - write several sectors from disk |
---|
| 105 | * @drive_info: driveinfo struct describing the disk |
---|
| 106 | * @lba: Position to write |
---|
| 107 | * @data: Buffer to write |
---|
| 108 | **/ |
---|
| 109 | int write_verify_sector(struct driveinfo *drive_info, |
---|
| 110 | const unsigned int lba, const void *data) |
---|
| 111 | { |
---|
| 112 | return write_verify_sectors(drive_info, lba, data, SECTOR); |
---|
| 113 | } |
---|
| 114 | |
---|
| 115 | /** |
---|
| 116 | * write_verify_sectors - write several sectors from disk |
---|
| 117 | * @drive_info: driveinfo struct describing the disk |
---|
| 118 | * @lba: Position to write |
---|
| 119 | * @data: Buffer to write |
---|
| 120 | * @size: Size of the buffer (number of sectors) |
---|
| 121 | **/ |
---|
| 122 | int write_verify_sectors(struct driveinfo *drive_info, |
---|
| 123 | const unsigned int lba, |
---|
| 124 | const void *data, const int size) |
---|
| 125 | { |
---|
| 126 | char *rb = malloc(SECTOR * size * sizeof(char)); |
---|
| 127 | int status; |
---|
| 128 | |
---|
| 129 | if (write_sectors(drive_info, lba, data, size) == -1) |
---|
| 130 | return -1; /* Write failure */ |
---|
| 131 | |
---|
| 132 | if (read_sectors(drive_info, rb, lba, size) == -1) |
---|
| 133 | return -1; /* Readback failure */ |
---|
| 134 | |
---|
| 135 | status = memcmp(data, rb, SECTOR * size); |
---|
| 136 | free(rb); |
---|
| 137 | return status ? -1 : 0; |
---|
| 138 | } |
---|