VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS-new/ata.c@ 41513

Last change on this file since 41513 was 39651, checked in by vboxsync, 13 years ago

Use indirect calls to avoid conditionals; renamed constants for clarity.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.0 KB
Line 
1/*
2 * Copyright (C) 2006-2011 Oracle Corporation
3 *
4 * This file is part of VirtualBox Open Source Edition (OSE), as
5 * available from http://www.virtualbox.org. This file is free software;
6 * you can redistribute it and/or modify it under the terms of the GNU
7 * General Public License (GPL) as published by the Free Software
8 * Foundation, in version 2 as it comes in the "COPYING" file of the
9 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
10 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
11 * --------------------------------------------------------------------
12 *
13 * This code is based on:
14 *
15 * ROM BIOS for use with Bochs/Plex86/QEMU emulation environment
16 *
17 * Copyright (C) 2002 MandrakeSoft S.A.
18 *
19 * MandrakeSoft S.A.
20 * 43, rue d'Aboukir
21 * 75002 Paris - France
22 * http://www.linux-mandrake.com/
23 * http://www.mandrakesoft.com/
24 *
25 * This library is free software; you can redistribute it and/or
26 * modify it under the terms of the GNU Lesser General Public
27 * License as published by the Free Software Foundation; either
28 * version 2 of the License, or (at your option) any later version.
29 *
30 * This library is distributed in the hope that it will be useful,
31 * but WITHOUT ANY WARRANTY; without even the implied warranty of
32 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
33 * Lesser General Public License for more details.
34 *
35 * You should have received a copy of the GNU Lesser General Public
36 * License along with this library; if not, write to the Free Software
37 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
38 *
39 */
40
41
42#include <stdint.h>
43#include <stdarg.h>
44#include "inlines.h"
45#include "biosint.h"
46#include "ebda.h"
47#include "ata.h"
48
49#if DEBUG_ATA
50# define BX_DEBUG_ATA(...) BX_DEBUG(__VA_ARGS__)
51#else
52# define BX_DEBUG_ATA(...)
53#endif
54
55
56// ---------------------------------------------------------------------------
57// Start of ATA/ATAPI Driver
58// ---------------------------------------------------------------------------
59
60void insw_discard(unsigned nwords, unsigned port);
61#pragma aux insw_discard = \
62 ".286" \
63 "again:" \
64 "in ax,dx" \
65 "loop again" \
66 parm [cx] [dx] modify exact [cx ax] nomemory;
67
68void insd_discard(unsigned ndwords, unsigned port);
69#pragma aux insd_discard = \
70 ".386" \
71 "push eax" \
72 "again:" \
73 "in eax,dx" \
74 "loop again" \
75 "pop eax" \
76 parm [cx] [dx] modify exact [cx] nomemory;
77
78// ---------------------------------------------------------------------------
79// ATA/ATAPI driver : initialization
80// ---------------------------------------------------------------------------
81void BIOSCALL ata_init(void)
82{
83 uint8_t channel, device;
84 bio_dsk_t __far *bios_dsk;
85
86 bios_dsk = read_word(0x0040, 0x000E) :> &EbdaData->bdisk;
87
88 // Channels info init.
89 for (channel=0; channel<BX_MAX_ATA_INTERFACES; channel++) {
90 bios_dsk->channels[channel].iface = ATA_IFACE_NONE;
91 bios_dsk->channels[channel].iobase1 = 0x0;
92 bios_dsk->channels[channel].iobase2 = 0x0;
93 bios_dsk->channels[channel].irq = 0;
94 }
95
96 // Devices info init.
97 for (device=0; device<BX_MAX_ATA_DEVICES; device++) {
98 bios_dsk->devices[device].type = DSK_TYPE_NONE;
99 bios_dsk->devices[device].device = DSK_DEVICE_NONE;
100 bios_dsk->devices[device].removable = 0;
101 bios_dsk->devices[device].lock = 0;
102 bios_dsk->devices[device].mode = ATA_MODE_NONE;
103 bios_dsk->devices[device].blksize = 0x200;
104 bios_dsk->devices[device].translation = GEO_TRANSLATION_NONE;
105 bios_dsk->devices[device].lchs.heads = 0;
106 bios_dsk->devices[device].lchs.cylinders = 0;
107 bios_dsk->devices[device].lchs.spt = 0;
108 bios_dsk->devices[device].pchs.heads = 0;
109 bios_dsk->devices[device].pchs.cylinders = 0;
110 bios_dsk->devices[device].pchs.spt = 0;
111 bios_dsk->devices[device].sectors = 0;
112 }
113
114 // hdidmap and cdidmap init.
115 for (device=0; device<BX_MAX_STORAGE_DEVICES; device++) {
116 bios_dsk->hdidmap[device] = BX_MAX_STORAGE_DEVICES;
117 bios_dsk->cdidmap[device] = BX_MAX_STORAGE_DEVICES;
118 }
119
120 bios_dsk->hdcount = 0;
121 bios_dsk->cdcount = 0;
122}
123
124// ---------------------------------------------------------------------------
125// ATA/ATAPI driver : software reset
126// ---------------------------------------------------------------------------
127// ATA-3
128// 8.2.1 Software reset - Device 0
129
130void ata_reset(uint16_t device)
131{
132 uint16_t iobase1, iobase2;
133 uint8_t channel, slave, sn, sc;
134 uint16_t max;
135 uint16_t pdelay;
136 bio_dsk_t __far *bios_dsk;
137
138 bios_dsk = read_word(0x0040, 0x000E) :> &EbdaData->bdisk;
139 channel = device / 2;
140 slave = device % 2;
141
142 iobase1 = bios_dsk->channels[channel].iobase1;
143 iobase2 = bios_dsk->channels[channel].iobase2;
144
145 // Reset
146
147 // 8.2.1 (a) -- set SRST in DC
148 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN | ATA_CB_DC_SRST);
149
150 // 8.2.1 (b) -- wait for BSY
151 max=0xff;
152 while(--max>0) {
153 uint8_t status = inb(iobase1+ATA_CB_STAT);
154 if ((status & ATA_CB_STAT_BSY) != 0)
155 break;
156 }
157
158 // 8.2.1 (f) -- clear SRST
159 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN);
160
161 if (bios_dsk->devices[device].type != DSK_TYPE_NONE) {
162 // 8.2.1 (g) -- check for sc==sn==0x01
163 // select device
164 outb(iobase1+ATA_CB_DH, slave?ATA_CB_DH_DEV1:ATA_CB_DH_DEV0);
165 sc = inb(iobase1+ATA_CB_SC);
166 sn = inb(iobase1+ATA_CB_SN);
167
168 if ( (sc==0x01) && (sn==0x01) ) {
169 // 8.2.1 (h) -- wait for not BSY
170 max=0xffff; /* The ATA specification says that the drive may be busy for up to 30 seconds. */
171 while(--max>0) {
172 uint8_t status = inb(iobase1+ATA_CB_STAT);
173 if ((status & ATA_CB_STAT_BSY) == 0)
174 break;
175 pdelay=0xffff;
176 while (--pdelay>0) {
177 /* nothing */
178 }
179 }
180 }
181 }
182
183 // 8.2.1 (i) -- wait for DRDY
184 max = 0x10; /* Speed up for virtual drives. Disks are immediately ready, CDs never */
185 while(--max>0) {
186 uint8_t status = inb(iobase1+ATA_CB_STAT);
187 if ((status & ATA_CB_STAT_RDY) != 0)
188 break;
189 }
190
191 // Enable interrupts
192 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
193}
194
195// ---------------------------------------------------------------------------
196// ATA/ATAPI driver : execute a data-in command
197// ---------------------------------------------------------------------------
198 // returns
199 // 0 : no error
200 // 1 : BUSY bit set
201 // 2 : read error
202 // 3 : expected DRQ=1
203 // 4 : no sectors left to read/verify
204 // 5 : more sectors to read/verify
205 // 6 : no sectors left to write
206 // 7 : more sectors to write
207uint16_t ata_cmd_data_in(bio_dsk_t __far *bios_dsk, uint16_t command, uint16_t count)
208{
209 uint32_t lba;
210 uint16_t iobase1, iobase2, blksize, mult_blk_cnt;
211 uint16_t cylinder;
212 uint16_t head;
213 uint16_t sector;
214 uint16_t device;
215 uint8_t channel, slave;
216 uint8_t status, mode;
217 char __far *buffer;
218
219 device = bios_dsk->drqp.dev_id;
220 channel = device / 2;
221 slave = device % 2;
222
223 iobase1 = bios_dsk->channels[channel].iobase1;
224 iobase2 = bios_dsk->channels[channel].iobase2;
225 mode = bios_dsk->devices[device].mode;
226 blksize = bios_dsk->devices[device].blksize;
227 if (blksize == 0) { /* If transfer size is exactly 64K */
228 if (mode == ATA_MODE_PIO32)
229 blksize = 0x4000;
230 else
231 blksize = 0x8000;
232 } else {
233 if (mode == ATA_MODE_PIO32)
234 blksize >>= 2;
235 else
236 blksize >>= 1;
237 }
238
239 status = inb(iobase1 + ATA_CB_STAT);
240 if (status & ATA_CB_STAT_BSY)
241 {
242 BX_DEBUG_ATA("%s: disk busy\n", __func__);
243 // Enable interrupts
244 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
245 return 1;
246 }
247
248 lba = bios_dsk->drqp.lba;
249 buffer = bios_dsk->drqp.buffer;
250 sector = bios_dsk->drqp.sector;
251 cylinder = bios_dsk->drqp.cylinder;
252 head = bios_dsk->drqp.head;
253
254 // sector will be 0 only on lba access. Convert to lba-chs
255 if (sector == 0) {
256 if (lba + count >= 268435456)
257 {
258 sector = (lba & 0xff000000L) >> 24;
259 cylinder = 0; /* The parameter lba is just a 32 bit value. */
260 outb(iobase1 + ATA_CB_SC, (count & 0xff00) >> 8);
261 outb(iobase1 + ATA_CB_SN, sector);
262 outb(iobase1 + ATA_CB_CL, cylinder & 0x00ff);
263 outb(iobase1 + ATA_CB_CH, cylinder >> 8);
264 /* Leave the bottom 24 bits as is, they are treated correctly by the
265 * LBA28 code path. */
266 lba &= 0xffffff;
267 }
268 sector = (uint16_t) (lba & 0x000000ffL);
269 lba >>= 8;
270 cylinder = (uint16_t) (lba & 0x0000ffffL);
271 lba >>= 16;
272 head = ((uint16_t) (lba & 0x0000000fL)) | 0x40;
273 }
274
275 outb(iobase2 + ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN);
276 outb(iobase1 + ATA_CB_FR, 0x00);
277 outb(iobase1 + ATA_CB_SC, count);
278 outb(iobase1 + ATA_CB_SN, sector);
279 outb(iobase1 + ATA_CB_CL, cylinder & 0x00ff);
280 outb(iobase1 + ATA_CB_CH, cylinder >> 8);
281 outb(iobase1 + ATA_CB_DH, (slave ? ATA_CB_DH_DEV1 : ATA_CB_DH_DEV0) | (uint8_t) head );
282 outb(iobase1 + ATA_CB_CMD, command);
283
284 if (command == ATA_CMD_READ_MULTIPLE || command == ATA_CMD_READ_MULTIPLE_EXT) {
285 mult_blk_cnt = count;
286 count = 1;
287 } else {
288 mult_blk_cnt = 1;
289 }
290
291 while (1) {
292 status = inb(iobase1 + ATA_CB_STAT);
293 if ( !(status & ATA_CB_STAT_BSY) )
294 break;
295 }
296
297 if (status & ATA_CB_STAT_ERR) {
298 BX_DEBUG_ATA("%s: read error\n", __func__);
299 // Enable interrupts
300 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
301 return 2;
302 } else if ( !(status & ATA_CB_STAT_DRQ) ) {
303 BX_DEBUG_ATA("%s: DRQ not set (status %02x)\n", __func__, (unsigned) status);
304 // Enable interrupts
305 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
306 return 3;
307 }
308
309 // FIXME : move seg/off translation here
310
311 int_enable(); // enable higher priority interrupts
312
313 while (1) {
314
315 // adjust if there will be an overrun. 2K max sector size
316 if (FP_OFF(buffer) >= 0xF800)
317 buffer = MK_FP(FP_SEG(buffer) + 0x80, FP_OFF(buffer) - 0x800);
318
319 if (mode == ATA_MODE_PIO32) {
320 buffer = rep_insd(buffer, blksize, iobase1);
321 } else {
322 buffer = rep_insw(buffer, blksize, iobase1);
323 }
324 bios_dsk->drqp.trsfsectors += mult_blk_cnt;
325 count--;
326 while (1) {
327 status = inb(iobase1 + ATA_CB_STAT);
328 if ( !(status & ATA_CB_STAT_BSY) )
329 break;
330 }
331 if (count == 0) {
332 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
333 != ATA_CB_STAT_RDY ) {
334 BX_DEBUG_ATA("%s: no sectors left (status %02x)\n", __func__, (unsigned) status);
335 // Enable interrupts
336 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
337 return 4;
338 }
339 break;
340 }
341 else {
342 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
343 != (ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ) ) {
344 BX_DEBUG_ATA("%s: more sectors left (status %02x)\n", __func__, (unsigned) status);
345 // Enable interrupts
346 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
347 return 5;
348 }
349 continue;
350 }
351 }
352 // Enable interrupts
353 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
354 return 0;
355}
356
357// ---------------------------------------------------------------------------
358// ATA/ATAPI driver : device detection
359// ---------------------------------------------------------------------------
360
361void BIOSCALL ata_detect(void)
362{
363 uint16_t ebda_seg = read_word(0x0040,0x000E);
364 uint8_t hdcount, cdcount, device, type;
365 uint8_t buffer[0x0200];
366 bio_dsk_t __far *bios_dsk;
367
368 bios_dsk = ebda_seg :> &EbdaData->bdisk;
369
370#if BX_MAX_ATA_INTERFACES > 0
371 bios_dsk->channels[0].iface = ATA_IFACE_ISA;
372 bios_dsk->channels[0].iobase1 = 0x1f0;
373 bios_dsk->channels[0].iobase2 = 0x3f0;
374 bios_dsk->channels[0].irq = 14;
375#endif
376#if BX_MAX_ATA_INTERFACES > 1
377 bios_dsk->channels[1].iface = ATA_IFACE_ISA;
378 bios_dsk->channels[1].iobase1 = 0x170;
379 bios_dsk->channels[1].iobase2 = 0x370;
380 bios_dsk->channels[1].irq = 15;
381#endif
382#if 0 //@todo - temporarily removed to avoid conflict with AHCI
383#if BX_MAX_ATA_INTERFACES > 2
384 bios_dsk->channels[2].iface = ATA_IFACE_ISA;
385 bios_dsk->channels[2].iobase1 = 0x1e8;
386 bios_dsk->channels[2].iobase2 = 0x3e0;
387 bios_dsk->channels[2].irq = 12;
388#endif
389#if BX_MAX_ATA_INTERFACES > 3
390 bios_dsk->channels[3].iface = ATA_IFACE_ISA;
391 bios_dsk->channels[3].iobase1 = 0x168;
392 bios_dsk->channels[3].iobase2 = 0x360;
393 bios_dsk->channels[3].irq = 11;
394#endif
395#endif
396#if BX_MAX_ATA_INTERFACES > 4
397#error Please fill the ATA interface informations
398#endif
399
400 // Device detection
401 hdcount = cdcount = 0;
402
403 for (device = 0; device < BX_MAX_ATA_DEVICES; device++) {
404 uint16_t iobase1, iobase2;
405 uint8_t channel, slave;
406 uint8_t sc, sn, cl, ch, st;
407
408 channel = device / 2;
409 slave = device % 2;
410
411 iobase1 = bios_dsk->channels[channel].iobase1;
412 iobase2 = bios_dsk->channels[channel].iobase2;
413
414 // Disable interrupts
415 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN);
416
417 // Look for device
418 outb(iobase1+ATA_CB_DH, slave ? ATA_CB_DH_DEV1 : ATA_CB_DH_DEV0);
419 outb(iobase1+ATA_CB_SC, 0x55);
420 outb(iobase1+ATA_CB_SN, 0xaa);
421 outb(iobase1+ATA_CB_SC, 0xaa);
422 outb(iobase1+ATA_CB_SN, 0x55);
423 outb(iobase1+ATA_CB_SC, 0x55);
424 outb(iobase1+ATA_CB_SN, 0xaa);
425
426 // If we found something
427 sc = inb(iobase1+ATA_CB_SC);
428 sn = inb(iobase1+ATA_CB_SN);
429
430 if ( (sc == 0x55) && (sn == 0xaa) ) {
431 bios_dsk->devices[device].type = DSK_TYPE_UNKNOWN;
432
433 // reset the channel
434 ata_reset(device);
435
436 // check for ATA or ATAPI
437 outb(iobase1+ATA_CB_DH, slave ? ATA_CB_DH_DEV1 : ATA_CB_DH_DEV0);
438 sc = inb(iobase1+ATA_CB_SC);
439 sn = inb(iobase1+ATA_CB_SN);
440 if ((sc==0x01) && (sn==0x01)) {
441 cl = inb(iobase1+ATA_CB_CL);
442 ch = inb(iobase1+ATA_CB_CH);
443 st = inb(iobase1+ATA_CB_STAT);
444
445 if ((cl==0x14) && (ch==0xeb)) {
446 bios_dsk->devices[device].type = DSK_TYPE_ATAPI;
447 } else if ((cl==0x00) && (ch==0x00) && (st!=0x00)) {
448 bios_dsk->devices[device].type = DSK_TYPE_ATA;
449 } else if ((cl==0xff) && (ch==0xff)) {
450 bios_dsk->devices[device].type = DSK_TYPE_NONE;
451 }
452 }
453 }
454
455 // Enable interrupts
456 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
457
458 type = bios_dsk->devices[device].type;
459
460 // Now we send a IDENTIFY command to ATA device
461 if (type == DSK_TYPE_ATA) {
462 uint32_t sectors;
463 uint16_t cylinders, heads, spt, blksize;
464 uint16_t lcylinders, lheads, lspt;
465 uint8_t chsgeo_base;
466 uint8_t removable, mode;
467
468 //Temporary values to do the transfer
469 bios_dsk->devices[device].device = DSK_DEVICE_HD;
470 bios_dsk->devices[device].mode = ATA_MODE_PIO16;
471 bios_dsk->drqp.buffer = buffer;
472 bios_dsk->drqp.dev_id = device;
473
474 if (ata_cmd_data_in(bios_dsk, ATA_CMD_IDENTIFY_DEVICE, 1) !=0 )
475 BX_PANIC("ata-detect: Failed to detect ATA device\n");
476
477 removable = (*(buffer+0) & 0x80) ? 1 : 0;
478 mode = *(buffer+96) ? ATA_MODE_PIO32 : ATA_MODE_PIO16;
479 blksize = 512; /* There is no sector size field any more. */
480
481 cylinders = *(uint16_t *)(buffer+(1*2)); // word 1
482 heads = *(uint16_t *)(buffer+(3*2)); // word 3
483 spt = *(uint16_t *)(buffer+(6*2)); // word 6
484
485 sectors = *(uint32_t *)(buffer+(60*2)); // word 60 and word 61
486 /** @todo update sectors to be a 64 bit number (also lba...). */
487 if (sectors == 268435455)
488 sectors = *(uint32_t *)(buffer+(100*2)); // words 100 to 103 (someday)
489 switch (device)
490 {
491 case 0:
492 chsgeo_base = 0x1e;
493 break;
494 case 1:
495 chsgeo_base = 0x26;
496 break;
497 case 2:
498 chsgeo_base = 0x67;
499 break;
500 case 3:
501 chsgeo_base = 0x70;
502 break;
503 case 4:
504 chsgeo_base = 0x40;
505 break;
506 case 5:
507 chsgeo_base = 0x48;
508 break;
509 case 6:
510 chsgeo_base = 0x50;
511 break;
512 case 7:
513 chsgeo_base = 0x58;
514 break;
515 default:
516 chsgeo_base = 0;
517 }
518 if (chsgeo_base != 0)
519 {
520 lcylinders = inb_cmos(chsgeo_base) + (inb_cmos(chsgeo_base+1) << 8);
521 lheads = inb_cmos(chsgeo_base+2);
522 lspt = inb_cmos(chsgeo_base+7);
523 }
524 else
525 {
526 lcylinders = 0;
527 lheads = 0;
528 lspt = 0;
529 }
530 BX_INFO("ata%d-%d: PCHS=%u/%d/%d LCHS=%u/%u/%u\n", channel, slave,
531 cylinders,heads, spt, lcylinders, lheads, lspt);
532
533 bios_dsk->devices[device].device = DSK_DEVICE_HD;
534 bios_dsk->devices[device].removable = removable;
535 bios_dsk->devices[device].mode = mode;
536 bios_dsk->devices[device].blksize = blksize;
537 bios_dsk->devices[device].pchs.heads = heads;
538 bios_dsk->devices[device].pchs.cylinders = cylinders;
539 bios_dsk->devices[device].pchs.spt = spt;
540 bios_dsk->devices[device].sectors = sectors;
541 bios_dsk->devices[device].lchs.heads = lheads;
542 bios_dsk->devices[device].lchs.cylinders = lcylinders;
543 bios_dsk->devices[device].lchs.spt = lspt;
544 if (device < 2)
545 {
546 uint8_t sum, i;
547 fdpt_t __far *fdpt;
548
549 if (device == 0)
550 fdpt = ebda_seg :> &EbdaData->fdpt0;
551 else
552 fdpt = ebda_seg :> &EbdaData->fdpt1;
553
554 /* Update the DPT for drive 0/1 pointed to by Int41/46. This used
555 * to be done at POST time with lots of ugly assembler code, which
556 * isn't worth the effort of converting from AMI to Award CMOS
557 * format. Just do it here. */
558 fdpt->lcyl = lcylinders;
559 fdpt->lhead = lheads;
560 fdpt->sig = 0xa0;
561 fdpt->spt = spt;
562 fdpt->cyl = cylinders;
563 fdpt->head = heads;
564 fdpt->lspt = spt;
565 sum = 0;
566 for (i = 0; i < 0xf; i++)
567 sum += *((uint8_t __far *)fdpt + i);
568 sum = -sum;
569 fdpt->csum = sum;
570 }
571
572 // fill hdidmap
573 bios_dsk->hdidmap[hdcount] = device;
574 hdcount++;
575 }
576
577 // Now we send an IDENTIFY command to ATAPI device
578 if (type == DSK_TYPE_ATAPI) {
579 uint8_t type, removable, mode;
580 uint16_t blksize;
581
582 // Temporary values to do the transfer
583 bios_dsk->devices[device].device = DSK_DEVICE_CDROM;
584 bios_dsk->devices[device].mode = ATA_MODE_PIO16;
585 bios_dsk->drqp.buffer = buffer;
586 bios_dsk->drqp.dev_id = device;
587
588 if (ata_cmd_data_in(bios_dsk, ATA_CMD_IDENTIFY_PACKET, 1) != 0)
589 BX_PANIC("ata-detect: Failed to detect ATAPI device\n");
590
591 type = *(buffer+1) & 0x1f;
592 removable = (*(buffer+0) & 0x80) ? 1 : 0;
593 mode = *(buffer+96) ? ATA_MODE_PIO32 : ATA_MODE_PIO16;
594 blksize = 2048;
595
596 bios_dsk->devices[device].device = type;
597 bios_dsk->devices[device].removable = removable;
598 bios_dsk->devices[device].mode = mode;
599 bios_dsk->devices[device].blksize = blksize;
600
601 // fill cdidmap
602 bios_dsk->cdidmap[cdcount] = device;
603 cdcount++;
604 }
605
606 {
607 uint32_t sizeinmb;
608 uint16_t ataversion;
609 uint8_t version, model[41];
610 int i;
611
612 switch (type) {
613 case DSK_TYPE_ATA:
614 sizeinmb = bios_dsk->devices[device].sectors;
615 sizeinmb >>= 11;
616 case DSK_TYPE_ATAPI:
617 // Read ATA/ATAPI version
618 ataversion = ((uint16_t)(*(buffer+161))<<8) | *(buffer+160);
619 for (version = 15; version > 0; version--) {
620 if ((ataversion & (1 << version)) !=0 )
621 break;
622 }
623
624 // Read model name
625 for (i = 0; i < 20; i++ ) {
626 *(model+(i*2)) = *(buffer+(i*2)+54+1);
627 *(model+(i*2)+1) = *(buffer+(i*2)+54);
628 }
629
630 // Reformat
631 *(model+40) = 0x00;
632 for ( i = 39; i > 0; i-- ){
633 if (*(model+i) == 0x20)
634 *(model+i) = 0x00;
635 else
636 break;
637 }
638 break;
639 }
640
641#ifdef VBOXz
642 // we don't want any noisy output for now
643#else /* !VBOX */
644 switch (type) {
645 int c;
646 case DSK_TYPE_ATA:
647 printf("ata%d %s: ", channel, slave ? " slave" : "master");
648 i=0;
649 while(c=*(model+i++))
650 printf("%c", c);
651 printf(" ATA-%d Hard-Disk (%lu MBytes)\n", version, sizeinmb);
652 break;
653 case DSK_TYPE_ATAPI:
654 printf("ata%d %s: ", channel, slave ? " slave" : "master");
655 i=0;
656 while(c=*(model+i++))
657 printf("%c", c);
658 if (bios_dsk->devices[device].device == DSK_DEVICE_CDROM)
659 printf(" ATAPI-%d CD-ROM/DVD-ROM\n", version);
660 else
661 printf(" ATAPI-%d Device\n", version);
662 break;
663 case DSK_TYPE_UNKNOWN:
664 printf("ata%d %s: Unknown device\n", channel , slave ? " slave" : "master");
665 break;
666 }
667#endif /* !VBOX */
668 }
669 }
670
671 // Store the devices counts
672 bios_dsk->hdcount = hdcount;
673 bios_dsk->cdcount = cdcount;
674 write_byte(0x40,0x75, hdcount);
675
676#ifdef VBOX
677 // we don't want any noisy output for now
678#else /* !VBOX */
679 printf("\n");
680#endif /* !VBOX */
681
682 // FIXME : should use bios=cmos|auto|disable bits
683 // FIXME : should know about translation bits
684 // FIXME : move hard_drive_post here
685
686}
687
688// ---------------------------------------------------------------------------
689// ATA/ATAPI driver : execute a data-out command
690// ---------------------------------------------------------------------------
691 // returns
692 // 0 : no error
693 // 1 : BUSY bit set
694 // 2 : read error
695 // 3 : expected DRQ=1
696 // 4 : no sectors left to read/verify
697 // 5 : more sectors to read/verify
698 // 6 : no sectors left to write
699 // 7 : more sectors to write
700uint16_t ata_cmd_data_out(bio_dsk_t __far *bios_dsk, uint16_t command, uint16_t count)
701{
702 uint32_t lba;
703 char __far *buffer;
704 uint16_t iobase1, iobase2, blksize;
705 uint16_t cylinder;
706 uint16_t head;
707 uint16_t sector;
708 uint16_t device;
709 uint8_t channel, slave;
710 uint8_t status, mode;
711
712 device = bios_dsk->drqp.dev_id;
713 channel = device / 2;
714 slave = device % 2;
715
716 iobase1 = bios_dsk->channels[channel].iobase1;
717 iobase2 = bios_dsk->channels[channel].iobase2;
718 mode = bios_dsk->devices[device].mode;
719 blksize = 0x200; // was = bios_dsk->devices[device].blksize;
720 if (mode == ATA_MODE_PIO32)
721 blksize >>= 2;
722 else
723 blksize >>= 1;
724
725 status = inb(iobase1 + ATA_CB_STAT);
726 if (status & ATA_CB_STAT_BSY)
727 {
728 // Enable interrupts
729 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
730 return 1;
731 }
732
733 lba = bios_dsk->drqp.lba;
734 buffer = bios_dsk->drqp.buffer;
735 sector = bios_dsk->drqp.sector;
736 cylinder = bios_dsk->drqp.cylinder;
737 head = bios_dsk->drqp.head;
738
739 // sector will be 0 only on lba access. Convert to lba-chs
740 if (sector == 0) {
741 if (lba + count >= 268435456)
742 {
743 sector = (lba & 0xff000000L) >> 24;
744 cylinder = 0; /* The parameter lba is just a 32 bit value. */
745 outb(iobase1 + ATA_CB_SC, (count & 0xff00) >> 8);
746 outb(iobase1 + ATA_CB_SN, sector);
747 outb(iobase1 + ATA_CB_CL, cylinder & 0x00ff);
748 outb(iobase1 + ATA_CB_CH, cylinder >> 8);
749 /* Leave the bottom 24 bits as is, they are treated correctly by the
750 * LBA28 code path. */
751 lba &= 0xffffff;
752 }
753 sector = (uint16_t) (lba & 0x000000ffL);
754 lba >>= 8;
755 cylinder = (uint16_t) (lba & 0x0000ffffL);
756 lba >>= 16;
757 head = ((uint16_t) (lba & 0x0000000fL)) | 0x40;
758 }
759
760 outb(iobase2 + ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN);
761 outb(iobase1 + ATA_CB_FR, 0x00);
762 outb(iobase1 + ATA_CB_SC, count);
763 outb(iobase1 + ATA_CB_SN, sector);
764 outb(iobase1 + ATA_CB_CL, cylinder & 0x00ff);
765 outb(iobase1 + ATA_CB_CH, cylinder >> 8);
766 outb(iobase1 + ATA_CB_DH, (slave ? ATA_CB_DH_DEV1 : ATA_CB_DH_DEV0) | (uint8_t) head );
767 outb(iobase1 + ATA_CB_CMD, command);
768
769 while (1) {
770 status = inb(iobase1 + ATA_CB_STAT);
771 if ( !(status & ATA_CB_STAT_BSY) )
772 break;
773 }
774
775 if (status & ATA_CB_STAT_ERR) {
776 BX_DEBUG_ATA("%s: read error\n", __func__);
777 // Enable interrupts
778 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
779 return 2;
780 } else if ( !(status & ATA_CB_STAT_DRQ) ) {
781 BX_DEBUG_ATA("%s: DRQ not set (status %02x)\n", __func__, (unsigned) status);
782 // Enable interrupts
783 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
784 return 3;
785 }
786
787 // FIXME : move seg/off translation here
788
789 int_enable(); // enable higher priority interrupts
790
791 while (1) {
792
793 // adjust if there will be an overrun. 2K max sector size
794 if (FP_OFF(buffer) >= 0xF800)
795 buffer = MK_FP(FP_SEG(buffer) + 0x80, FP_OFF(buffer) - 0x800);
796
797 if (mode == ATA_MODE_PIO32) {
798 buffer = rep_outsd(buffer, blksize, iobase1);
799 } else {
800 buffer = rep_outsw(buffer, blksize, iobase1);
801 }
802
803 bios_dsk->drqp.trsfsectors++;
804 count--;
805 while (1) {
806 status = inb(iobase1 + ATA_CB_STAT);
807 if ( !(status & ATA_CB_STAT_BSY) )
808 break;
809 }
810 if (count == 0) {
811 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DF | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
812 != ATA_CB_STAT_RDY ) {
813 BX_DEBUG_ATA("%s: no sectors left (status %02x)\n", __func__, (unsigned) status);
814 // Enable interrupts
815 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
816 return 6;
817 }
818 break;
819 }
820 else {
821 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
822 != (ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ) ) {
823 BX_DEBUG_ATA("%s: more sectors left (status %02x)\n", __func__, (unsigned) status);
824 // Enable interrupts
825 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
826 return 7;
827 }
828 continue;
829 }
830 }
831 // Enable interrupts
832 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
833 return 0;
834}
835
836
837/**
838 * Read sectors from an attached ATA device.
839 *
840 * @returns status code.
841 * @param bios_dsk Pointer to disk request packet (in the
842 * EBDA).
843 */
844int ata_read_sectors(bio_dsk_t __far *bios_dsk)
845{
846 uint16_t n_sect;
847 int status;
848 uint8_t device_id;
849
850 device_id = bios_dsk->drqp.dev_id;
851 n_sect = bios_dsk->drqp.nsect;
852
853 if (bios_dsk->drqp.sector) {
854 /* CHS addressing. */
855 bios_dsk->devices[device_id].blksize = n_sect * 0x200;
856 status = ata_cmd_data_in(bios_dsk, ATA_CMD_READ_MULTIPLE, n_sect);
857 bios_dsk->devices[device_id].blksize = 0x200;
858 } else {
859 /* LBA addressing. */
860 if (bios_dsk->drqp.lba + n_sect >= 268435456)
861 status = ata_cmd_data_in(bios_dsk, ATA_CMD_READ_SECTORS_EXT, n_sect);
862 else {
863 bios_dsk->devices[device_id].blksize = n_sect * 0x200;
864 status = ata_cmd_data_in(bios_dsk, ATA_CMD_READ_MULTIPLE, n_sect);
865 bios_dsk->devices[device_id].blksize = 0x200;
866 }
867 }
868 return status;
869}
870
871/**
872 * Write sectors to an attached ATA device.
873 *
874 * @returns status code.
875 * @param bios_dsk Pointer to disk request packet (in the
876 * EBDA).
877 */
878int ata_write_sectors(bio_dsk_t __far *bios_dsk)
879{
880 uint16_t n_sect;
881
882 n_sect = bios_dsk->drqp.nsect;
883
884 if (bios_dsk->drqp.sector) {
885 /* CHS addressing. */
886 return ata_cmd_data_out(bios_dsk, ATA_CMD_WRITE_SECTORS, n_sect);
887 } else {
888 /* LBA addressing. */
889 if (bios_dsk->drqp.lba + n_sect >= 268435456)
890 return ata_cmd_data_out(bios_dsk, ATA_CMD_WRITE_SECTORS_EXT, n_sect);
891 else
892 return ata_cmd_data_out(bios_dsk, ATA_CMD_WRITE_SECTORS, n_sect);
893 }
894}
895
896
897// ---------------------------------------------------------------------------
898// ATA/ATAPI driver : execute a packet command
899// ---------------------------------------------------------------------------
900 // returns
901 // 0 : no error
902 // 1 : error in parameters
903 // 2 : BUSY bit set
904 // 3 : error
905 // 4 : not ready
906uint16_t ata_cmd_packet(uint16_t device, uint8_t cmdlen, char __far *cmdbuf,
907 uint16_t header, uint32_t length, uint8_t inout, char __far *buffer)
908{
909 uint16_t iobase1, iobase2;
910 uint16_t lcount, lbefore, lafter, count;
911 uint8_t channel, slave;
912 uint8_t status, mode, lmode;
913 uint32_t transfer;
914 bio_dsk_t __far *bios_dsk;
915
916 bios_dsk = read_word(0x0040, 0x000E) :> &EbdaData->bdisk;
917
918 channel = device / 2;
919 slave = device % 2;
920
921 // Data out is not supported yet
922 if (inout == ATA_DATA_OUT) {
923 BX_INFO("%s: DATA_OUT not supported yet\n", __func__);
924 return 1;
925 }
926
927 // The header length must be even
928 if (header & 1) {
929 BX_DEBUG_ATA("%s: header must be even (%04x)\n", __func__, header);
930 return 1;
931 }
932
933 iobase1 = bios_dsk->channels[channel].iobase1;
934 iobase2 = bios_dsk->channels[channel].iobase2;
935 mode = bios_dsk->devices[device].mode;
936 transfer = 0L;
937
938 if (cmdlen < 12)
939 cmdlen = 12;
940 if (cmdlen > 12)
941 cmdlen = 16;
942 cmdlen >>= 1;
943
944 // Reset count of transferred data
945 // @todo: clear in calling code?
946 bios_dsk->drqp.trsfsectors = 0;
947 bios_dsk->drqp.trsfbytes = 0;
948
949 status = inb(iobase1 + ATA_CB_STAT);
950 if (status & ATA_CB_STAT_BSY)
951 return 2;
952
953 outb(iobase2 + ATA_CB_DC, ATA_CB_DC_HD15 | ATA_CB_DC_NIEN);
954 // outb(iobase1 + ATA_CB_FR, 0x00);
955 // outb(iobase1 + ATA_CB_SC, 0x00);
956 // outb(iobase1 + ATA_CB_SN, 0x00);
957 outb(iobase1 + ATA_CB_CL, 0xfff0 & 0x00ff);
958 outb(iobase1 + ATA_CB_CH, 0xfff0 >> 8);
959 outb(iobase1 + ATA_CB_DH, slave ? ATA_CB_DH_DEV1 : ATA_CB_DH_DEV0);
960 outb(iobase1 + ATA_CB_CMD, ATA_CMD_PACKET);
961
962 // Device should ok to receive command
963 while (1) {
964 status = inb(iobase1 + ATA_CB_STAT);
965 if ( !(status & ATA_CB_STAT_BSY) ) break;
966 }
967
968 if (status & ATA_CB_STAT_ERR) {
969 BX_DEBUG_ATA("%s: error, status is %02x\n", __func__, status);
970 // Enable interrupts
971 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
972 return 3;
973 } else if ( !(status & ATA_CB_STAT_DRQ) ) {
974 BX_DEBUG_ATA("%s: DRQ not set (status %02x)\n", __func__, (unsigned) status);
975 // Enable interrupts
976 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
977 return 4;
978 }
979
980 int_enable(); // enable higher priority interrupts
981
982 // Normalize address
983 BX_DEBUG_ATA("acp1 buffer ptr: %04x:%04x wlen %04x\n", FP_SEG(cmdbuf), FP_OFF(cmdbuf), cmdlen);
984 cmdbuf = MK_FP(FP_SEG(cmdbuf) + FP_OFF(cmdbuf) / 16 , FP_OFF(cmdbuf) % 16);
985 // cmdseg += (cmdoff / 16);
986 // cmdoff %= 16;
987
988 // Send command to device
989 rep_outsw(cmdbuf, cmdlen, iobase1);
990
991 if (inout == ATA_DATA_NO) {
992 status = inb(iobase1 + ATA_CB_STAT);
993 }
994 else {
995 while (1) {
996
997 while (1) {
998 status = inb(iobase1 + ATA_CB_STAT);
999 if ( !(status & ATA_CB_STAT_BSY) )
1000 break;
1001 }
1002
1003 // Check if command completed
1004 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_DRQ) ) ==0 )
1005 break;
1006
1007 if (status & ATA_CB_STAT_ERR) {
1008 BX_DEBUG_ATA("%s: error (status %02x)\n", __func__, status);
1009 // Enable interrupts
1010 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
1011 return 3;
1012 }
1013
1014 // Device must be ready to send data
1015 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
1016 != (ATA_CB_STAT_RDY | ATA_CB_STAT_DRQ) ) {
1017 BX_DEBUG_ATA("%s: not ready (status %02x)\n", __func__, status);
1018 // Enable interrupts
1019 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
1020 return 4;
1021 }
1022
1023 // Normalize address
1024 BX_DEBUG_ATA("acp2 buffer ptr: %04x:%04x\n", FP_SEG(buffer), FP_OFF(buffer));
1025 buffer = MK_FP(FP_SEG(buffer) + FP_OFF(buffer) / 16 , FP_OFF(buffer) % 16);
1026 // bufseg += (bufoff / 16);
1027 // bufoff %= 16;
1028
1029 // Get the byte count
1030 lcount = ((uint16_t)(inb(iobase1 + ATA_CB_CH))<<8)+inb(iobase1 + ATA_CB_CL);
1031
1032 // adjust to read what we want
1033 if (header>lcount) {
1034 lbefore = lcount;
1035 header -= lcount;
1036 lcount = 0;
1037 }
1038 else {
1039 lbefore = header;
1040 header = 0;
1041 lcount -= lbefore;
1042 }
1043
1044 if (lcount>length) {
1045 lafter = lcount - length;
1046 lcount = length;
1047 length = 0;
1048 }
1049 else {
1050 lafter = 0;
1051 length -= lcount;
1052 }
1053
1054 // Save byte count
1055 count = lcount;
1056
1057 BX_DEBUG_ATA("Trying to read %04x bytes (%04x %04x %04x) ",lbefore+lcount+lafter,lbefore,lcount,lafter);
1058 BX_DEBUG_ATA("to 0x%04x:0x%04x\n",FP_SEG(buffer),FP_OFF(buffer));
1059
1060 // If counts not dividable by 4, use 16bits mode
1061 lmode = mode;
1062 if (lbefore & 0x03)
1063 lmode = ATA_MODE_PIO16;
1064 if (lcount & 0x03)
1065 lmode = ATA_MODE_PIO16;
1066 if (lafter & 0x03)
1067 lmode = ATA_MODE_PIO16;
1068
1069 // adds an extra byte if count are odd. before is always even
1070 if (lcount & 0x01) {
1071 lcount += 1;
1072 if ((lafter > 0) && (lafter & 0x01)) {
1073 lafter -= 1;
1074 }
1075 }
1076
1077 if (lmode == ATA_MODE_PIO32) {
1078 lcount >>= 2;
1079 lbefore >>= 2;
1080 lafter >>= 2;
1081 }
1082 else {
1083 lcount >>= 1;
1084 lbefore >>= 1;
1085 lafter >>= 1;
1086 }
1087
1088 if (lmode == ATA_MODE_PIO32) {
1089 if (lbefore)
1090 insd_discard(lbefore, iobase1);
1091 rep_insd(buffer, lcount, iobase1);
1092 if (lafter)
1093 insd_discard(lafter, iobase1);
1094 } else {
1095 if (lbefore)
1096 insw_discard(lbefore, iobase1);
1097 rep_insw(buffer, lcount, iobase1);
1098 if (lafter)
1099 insw_discard(lafter, iobase1);
1100 }
1101
1102
1103 // Compute new buffer address
1104 buffer += count;
1105
1106 // Save transferred bytes count
1107 transfer += count;
1108 bios_dsk->drqp.trsfbytes = transfer;
1109 }
1110 }
1111
1112 // Final check, device must be ready
1113 if ( (status & (ATA_CB_STAT_BSY | ATA_CB_STAT_RDY | ATA_CB_STAT_DF | ATA_CB_STAT_DRQ | ATA_CB_STAT_ERR) )
1114 != ATA_CB_STAT_RDY ) {
1115 BX_DEBUG_ATA("%s: not ready (status %02x)\n", __func__, (unsigned) status);
1116 // Enable interrupts
1117 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
1118 return 4;
1119 }
1120
1121 // Enable interrupts
1122 outb(iobase2+ATA_CB_DC, ATA_CB_DC_HD15);
1123 return 0;
1124}
1125
1126
1127// ---------------------------------------------------------------------------
1128// End of ATA/ATAPI Driver
1129// ---------------------------------------------------------------------------
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette