diff options
Diffstat (limited to 'drivers/ide/ide-iops.c')
-rw-r--r-- | drivers/ide/ide-iops.c | 68 |
1 files changed, 25 insertions, 43 deletions
diff --git a/drivers/ide/ide-iops.c b/drivers/ide/ide-iops.c index 7ec7fa2aef96..fbbbb30ae964 100644 --- a/drivers/ide/ide-iops.c +++ b/drivers/ide/ide-iops.c | |||
@@ -191,36 +191,47 @@ static void ata_vlb_sync(ide_drive_t *drive, unsigned long port) | |||
191 | 191 | ||
192 | /* | 192 | /* |
193 | * This is used for most PIO data transfers *from* the IDE interface | 193 | * This is used for most PIO data transfers *from* the IDE interface |
194 | * | ||
195 | * These routines will round up any request for an odd number of bytes, | ||
196 | * so if an odd len is specified, be sure that there's at least one | ||
197 | * extra byte allocated for the buffer. | ||
194 | */ | 198 | */ |
195 | static void ata_input_data(ide_drive_t *drive, struct request *rq, | 199 | static void ata_input_data(ide_drive_t *drive, struct request *rq, |
196 | void *buffer, u32 wcount) | 200 | void *buf, unsigned int len) |
197 | { | 201 | { |
198 | ide_hwif_t *hwif = drive->hwif; | 202 | ide_hwif_t *hwif = drive->hwif; |
199 | struct ide_io_ports *io_ports = &hwif->io_ports; | 203 | struct ide_io_ports *io_ports = &hwif->io_ports; |
204 | unsigned long data_addr = io_ports->data_addr; | ||
200 | u8 io_32bit = drive->io_32bit; | 205 | u8 io_32bit = drive->io_32bit; |
201 | 206 | ||
207 | len++; | ||
208 | |||
202 | if (io_32bit) { | 209 | if (io_32bit) { |
203 | if (io_32bit & 2) { | 210 | if (io_32bit & 2) { |
204 | unsigned long flags; | 211 | unsigned long flags; |
205 | 212 | ||
206 | local_irq_save(flags); | 213 | local_irq_save(flags); |
207 | ata_vlb_sync(drive, io_ports->nsect_addr); | 214 | ata_vlb_sync(drive, io_ports->nsect_addr); |
208 | hwif->INSL(io_ports->data_addr, buffer, wcount); | 215 | hwif->INSL(data_addr, buf, len / 4); |
209 | local_irq_restore(flags); | 216 | local_irq_restore(flags); |
210 | } else | 217 | } else |
211 | hwif->INSL(io_ports->data_addr, buffer, wcount); | 218 | hwif->INSL(data_addr, buf, len / 4); |
219 | |||
220 | if ((len & 3) >= 2) | ||
221 | hwif->INSW(data_addr, (u8 *)buf + (len & ~3), 1); | ||
212 | } else | 222 | } else |
213 | hwif->INSW(io_ports->data_addr, buffer, wcount << 1); | 223 | hwif->INSW(data_addr, buf, len / 2); |
214 | } | 224 | } |
215 | 225 | ||
216 | /* | 226 | /* |
217 | * This is used for most PIO data transfers *to* the IDE interface | 227 | * This is used for most PIO data transfers *to* the IDE interface |
218 | */ | 228 | */ |
219 | static void ata_output_data(ide_drive_t *drive, struct request *rq, | 229 | static void ata_output_data(ide_drive_t *drive, struct request *rq, |
220 | void *buffer, u32 wcount) | 230 | void *buf, unsigned int len) |
221 | { | 231 | { |
222 | ide_hwif_t *hwif = drive->hwif; | 232 | ide_hwif_t *hwif = drive->hwif; |
223 | struct ide_io_ports *io_ports = &hwif->io_ports; | 233 | struct ide_io_ports *io_ports = &hwif->io_ports; |
234 | unsigned long data_addr = io_ports->data_addr; | ||
224 | u8 io_32bit = drive->io_32bit; | 235 | u8 io_32bit = drive->io_32bit; |
225 | 236 | ||
226 | if (io_32bit) { | 237 | if (io_32bit) { |
@@ -229,50 +240,21 @@ static void ata_output_data(ide_drive_t *drive, struct request *rq, | |||
229 | 240 | ||
230 | local_irq_save(flags); | 241 | local_irq_save(flags); |
231 | ata_vlb_sync(drive, io_ports->nsect_addr); | 242 | ata_vlb_sync(drive, io_ports->nsect_addr); |
232 | hwif->OUTSL(io_ports->data_addr, buffer, wcount); | 243 | hwif->OUTSL(data_addr, buf, len / 4); |
233 | local_irq_restore(flags); | 244 | local_irq_restore(flags); |
234 | } else | 245 | } else |
235 | hwif->OUTSL(io_ports->data_addr, buffer, wcount); | 246 | hwif->OUTSL(data_addr, buf, len / 4); |
236 | } else | ||
237 | hwif->OUTSW(io_ports->data_addr, buffer, wcount << 1); | ||
238 | } | ||
239 | |||
240 | /* | ||
241 | * The following routines are mainly used by the ATAPI drivers. | ||
242 | * | ||
243 | * These routines will round up any request for an odd number of bytes, | ||
244 | * so if an odd bytecount is specified, be sure that there's at least one | ||
245 | * extra byte allocated for the buffer. | ||
246 | */ | ||
247 | |||
248 | static void atapi_input_bytes(ide_drive_t *drive, void *buffer, u32 bytecount) | ||
249 | { | ||
250 | ide_hwif_t *hwif = HWIF(drive); | ||
251 | 247 | ||
252 | ++bytecount; | 248 | if ((len & 3) >= 2) |
253 | hwif->ata_input_data(drive, NULL, buffer, bytecount / 4); | 249 | hwif->OUTSW(data_addr, (u8 *)buf + (len & ~3), 1); |
254 | if ((bytecount & 0x03) >= 2) | 250 | } else |
255 | hwif->INSW(hwif->io_ports.data_addr, | 251 | hwif->OUTSW(data_addr, buf, len / 2); |
256 | (u8 *)buffer + (bytecount & ~0x03), 1); | ||
257 | } | ||
258 | |||
259 | static void atapi_output_bytes(ide_drive_t *drive, void *buffer, u32 bytecount) | ||
260 | { | ||
261 | ide_hwif_t *hwif = HWIF(drive); | ||
262 | |||
263 | ++bytecount; | ||
264 | hwif->ata_output_data(drive, NULL, buffer, bytecount / 4); | ||
265 | if ((bytecount & 0x03) >= 2) | ||
266 | hwif->OUTSW(hwif->io_ports.data_addr, | ||
267 | (u8 *)buffer + (bytecount & ~0x03), 1); | ||
268 | } | 252 | } |
269 | 253 | ||
270 | void default_hwif_transport(ide_hwif_t *hwif) | 254 | void default_hwif_transport(ide_hwif_t *hwif) |
271 | { | 255 | { |
272 | hwif->ata_input_data = ata_input_data; | 256 | hwif->input_data = ata_input_data; |
273 | hwif->ata_output_data = ata_output_data; | 257 | hwif->output_data = ata_output_data; |
274 | hwif->atapi_input_bytes = atapi_input_bytes; | ||
275 | hwif->atapi_output_bytes = atapi_output_bytes; | ||
276 | } | 258 | } |
277 | 259 | ||
278 | void ide_fix_driveid (struct hd_driveid *id) | 260 | void ide_fix_driveid (struct hd_driveid *id) |
@@ -656,7 +638,7 @@ int ide_driveid_update(ide_drive_t *drive) | |||
656 | local_irq_restore(flags); | 638 | local_irq_restore(flags); |
657 | return 0; | 639 | return 0; |
658 | } | 640 | } |
659 | hwif->ata_input_data(drive, NULL, id, SECTOR_WORDS); | 641 | hwif->input_data(drive, NULL, id, SECTOR_SIZE); |
660 | (void)ide_read_status(drive); /* clear drive IRQ */ | 642 | (void)ide_read_status(drive); /* clear drive IRQ */ |
661 | local_irq_enable(); | 643 | local_irq_enable(); |
662 | local_irq_restore(flags); | 644 | local_irq_restore(flags); |