VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS/floppy.c@ 58450

Last change on this file since 58450 was 56316, checked in by vboxsync, 10 years ago

whitespace

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 43.0 KB
Line 
1/*
2 * Copyright (C) 2006-2015 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 "inlines.h"
44#include "biosint.h"
45
46extern uint16_t get_floppy_dpt(uint8_t drive_type);
47
48//////////////////////
49// FLOPPY functions //
50//////////////////////
51
52void set_diskette_ret_status(uint8_t value)
53{
54 write_byte(0x0040, 0x0041, value);
55}
56
57void set_diskette_current_cyl(uint8_t drive, uint8_t cyl)
58{
59 if (drive > 1)
60 BX_PANIC("set_diskette_current_cyl: drive > 1\n");
61 write_byte(0x0040, 0x0094+drive, cyl);
62}
63
64#if 1 //BX_SUPPORT_FLOPPY
65
66#if DEBUG_INT13_FL
67# define BX_DEBUG_INT13_FL(...) BX_DEBUG(__VA_ARGS__)
68#else
69# define BX_DEBUG_INT13_FL(...)
70#endif
71
72#define BX_FLOPPY_ON_CNT 37 /* 2 seconds */
73
74extern int diskette_param_table; /* At a fixed location. */
75
76#ifndef VBOX_WITH_FLOPPY_IRQ_POLLING
77
78/**
79 * Wait for the 7th bit of 0040:003e to be set by int0e_handler.
80 * @returns first 7 bits of byte 0040:003e, interrupts disabled.
81 */
82uint8_t floppy_wait_for_interrupt(void)
83{
84 int_disable();
85 for (;;)
86 {
87 uint8_t val8 = read_byte(0x0040, 0x003e);
88 if (val8 & 0x80)
89 return val8 & ~0x7f;
90 int_enable_hlt_disable();
91 }
92}
93
94/**
95 * Wait for the 7th bit of 0040:003e to be set by int0e_handler or 0040:0040 to
96 * be cleared by the timer, clearing the interrupt flag on success.
97 *
98 * @returns 0 on timeout with interrupts enabled.
99 * All 8 bits at 0040:003e on interrupt with interrupts disabled (i.e.
100 * non-zero), after first clearing the 7th bit at 0040:003e.
101 */
102uint8_t floppy_wait_for_interrupt_or_timeout(void)
103{
104 int_disable();
105 for (;;)
106 {
107 uint8_t val8 = read_byte(0x0040, 0x0040);
108 if (val8 == 0) {
109 int_enable();
110 return 0;
111 }
112
113 val8 = read_byte(0x0040, 0x003e);
114 if (val8 & 0x80) {
115 write_byte(0x0040, 0x003e, val8 & 0x7f);
116 return val8;
117 }
118 int_enable_hlt_disable();
119 }
120}
121
122#endif /* !VBOX_WITH_FLOPPY_IRQ_POLLING */
123
124void floppy_reset_controller(void)
125{
126 uint8_t val8;
127
128 // Reset controller
129 val8 = inb(0x03f2);
130 outb(0x03f2, val8 & ~0x04);
131 outb(0x03f2, val8 | 0x04);
132
133 // Wait for controller to come out of reset
134 do {
135 val8 = inb(0x3f4);
136 } while ( (val8 & 0xc0) != 0x80 );
137}
138
139void floppy_prepare_controller(uint16_t drive)
140{
141 uint8_t val8, dor, prev_reset;
142
143 // set 40:3e bit 7 to 0
144 val8 = read_byte(0x0040, 0x003e);
145 val8 &= 0x7f;
146 write_byte(0x0040, 0x003e, val8);
147
148 // turn on motor of selected drive, DMA & int enabled, normal operation
149 prev_reset = inb(0x03f2) & 0x04;
150 if (drive)
151 dor = 0x20;
152 else
153 dor = 0x10;
154 dor |= 0x0c;
155 dor |= drive;
156 outb(0x03f2, dor);
157
158 // reset the disk motor timeout value of INT 08
159 write_byte(0x0040,0x0040, BX_FLOPPY_ON_CNT);
160
161 // program data rate
162 val8 = read_byte(0x0040, 0x008b);
163 val8 >>= 6;
164 outb(0x03f7, val8);
165
166 // wait for drive readiness
167 do {
168 val8 = inb(0x3f4);
169 } while ( (val8 & 0xc0) != 0x80 );
170
171 if (prev_reset == 0) {
172#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
173 // turn on interrupts
174 int_enable();
175 // wait on 40:3e bit 7 to become 1
176 do {
177 val8 = read_byte(0x0040, 0x003e);
178 } while ( (val8 & 0x80) == 0 );
179 val8 &= 0x7f;
180 int_disable();
181#else
182 val8 = floppy_wait_for_interrupt(); /* (7th bit cleared in ret val) */
183#endif
184 write_byte(0x0040, 0x003e, val8);
185 }
186}
187
188bx_bool floppy_media_known(uint16_t drive)
189{
190 uint8_t val8;
191 uint16_t media_state_offset;
192
193 val8 = read_byte(0x0040, 0x003e); // diskette recal status
194 if (drive)
195 val8 >>= 1;
196 val8 &= 0x01;
197 if (val8 == 0)
198 return 0;
199
200 media_state_offset = 0x0090;
201 if (drive)
202 media_state_offset += 1;
203
204 val8 = read_byte(0x0040, media_state_offset);
205 val8 = (val8 >> 4) & 0x01;
206 if (val8 == 0)
207 return 0;
208
209 // checks passed, return KNOWN
210 return 1;
211}
212
213bx_bool floppy_read_id(uint16_t drive)
214{
215#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
216 uint8_t val8;
217#endif
218 uint8_t return_status[7];
219 int i;
220
221 floppy_prepare_controller(drive);
222
223 // send Read ID command (2 bytes) to controller
224 outb(0x03f5, 0x4a); // 4a: Read ID (MFM)
225 outb(0x03f5, drive); // 0=drive0, 1=drive1, head always 0
226
227#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
228 // turn on interrupts
229 int_enable();
230
231 // wait on 40:3e bit 7 to become 1
232 do {
233 val8 = (read_byte(0x0040, 0x003e) & 0x80);
234 } while ( val8 == 0 );
235
236 val8 = 0; // separate asm from while() loop
237 // turn off interrupts
238 int_disable();
239#else
240 floppy_wait_for_interrupt();
241#endif
242
243 // read 7 return status bytes from controller
244 for (i = 0; i < 7; ++i) {
245 return_status[i] = inb(0x3f5);
246 }
247
248 if ( (return_status[0] & 0xc0) != 0 )
249 return 0;
250 else
251 return 1;
252}
253
254bx_bool floppy_drive_recal(uint16_t drive)
255{
256 uint8_t val8;
257 uint16_t curr_cyl_offset;
258
259 floppy_prepare_controller(drive);
260
261 // send Recalibrate command (2 bytes) to controller
262 outb(0x03f5, 0x07); // 07: Recalibrate
263 outb(0x03f5, drive); // 0=drive0, 1=drive1
264
265#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
266 // turn on interrupts
267 int_enable();
268
269 // wait on 40:3e bit 7 to become 1
270 do {
271 val8 = (read_byte(0x0040, 0x003e) & 0x80);
272 } while ( val8 == 0 );
273
274 val8 = 0; // separate asm from while() loop
275 // turn off interrupts
276 int_disable();
277
278 // set 40:3e bit 7 to 0, and calibrated bit
279 val8 = read_byte(0x0040, 0x003e);
280 val8 &= 0x7f;
281#else
282 val8 = floppy_wait_for_interrupt(); /* (7th bit cleared in ret val) */
283
284 // set 40:3e bit 7 to 0, and calibrated bit
285#endif
286 if (drive) {
287 val8 |= 0x02; // Drive 1 calibrated
288 curr_cyl_offset = 0x0095;
289 } else {
290 val8 |= 0x01; // Drive 0 calibrated
291 curr_cyl_offset = 0x0094;
292 }
293 write_byte(0x0040, 0x003e, val8);
294 write_byte(0x0040, curr_cyl_offset, 0); // current cylinder is 0
295
296 return 1;
297}
298
299
300bx_bool floppy_media_sense(uint16_t drive)
301{
302 bx_bool retval;
303 uint16_t media_state_offset;
304 uint8_t drive_type, config_data, media_state;
305
306 if (floppy_drive_recal(drive) == 0)
307 return 0;
308
309 // Try the diskette data rates in the following order:
310 // 1 Mbps -> 500 Kbps -> 300 Kbps -> 250 Kbps
311 // The 1 Mbps rate is only tried for 2.88M drives.
312
313 // ** config_data **
314 // Bitfields for diskette media control:
315 // Bit(s) Description (Table M0028)
316 // 7-6 last data rate set by controller
317 // 00=500kbps, 01=300kbps, 10=250kbps, 11=1Mbps
318 // 5-4 last diskette drive step rate selected
319 // 00=0Ch, 01=0Dh, 10=0Eh, 11=0Ah
320 // 3-2 {data rate at start of operation}
321 // 1-0 reserved
322
323 // ** media_state **
324 // Bitfields for diskette drive media state:
325 // Bit(s) Description (Table M0030)
326 // 7-6 data rate
327 // 00=500kbps, 01=300kbps, 10=250kbps, 11=1Mbps
328 // 5 double stepping required (e.g. 360kB in 1.2MB)
329 // 4 media type established
330 // 3 drive capable of supporting 4MB media
331 // 2-0 on exit from BIOS, contains
332 // 000 trying 360kB in 360kB
333 // 001 trying 360kB in 1.2MB
334 // 010 trying 1.2MB in 1.2MB
335 // 011 360kB in 360kB established
336 // 100 360kB in 1.2MB established
337 // 101 1.2MB in 1.2MB established
338 // 110 reserved
339 // 111 all other formats/drives
340
341 // @todo: break out drive type determination
342 drive_type = inb_cmos(0x10);
343 if (drive == 0)
344 drive_type >>= 4;
345 else
346 drive_type &= 0x0f;
347 if ( drive_type == 1 ) {
348 // 360K 5.25" drive
349 config_data = 0x00; // 0000 0000
350 media_state = 0x15; // 0001 0101
351 retval = 1;
352 }
353 else if ( drive_type == 2 ) {
354 // 1.2 MB 5.25" drive
355 config_data = 0x00; // 0000 0000
356 media_state = 0x35; // 0011 0101 // need double stepping??? (bit 5)
357 retval = 1;
358 }
359 else if ( drive_type == 3 ) {
360 // 720K 3.5" drive
361 config_data = 0x00; // 0000 0000 ???
362 media_state = 0x17; // 0001 0111
363 retval = 1;
364 }
365 else if ( drive_type == 4 ) {
366 // 1.44 MB 3.5" drive
367 config_data = 0x00; // 0000 0000
368 media_state = 0x17; // 0001 0111
369 retval = 1;
370 }
371 else if ( drive_type == 5 ) {
372 // 2.88 MB 3.5" drive
373 config_data = 0xCC; // 1100 1100
374 media_state = 0xD7; // 1101 0111
375 retval = 1;
376 }
377 // Extended floppy size uses special cmos setting
378 else if ( drive_type == 14 || drive_type == 15 ) {
379 // 15.6 MB 3.5" (fake) || 63.5 MB 3.5" (fake) - report same as 2.88 MB.
380 config_data = 0xCC; // 1100 1100
381 media_state = 0xD7; // 1101 0111
382 retval = 1;
383 }
384 else {
385 // not recognized
386 config_data = 0x00; // 0000 0000
387 media_state = 0x00; // 0000 0000
388 retval = 0;
389 }
390
391 write_byte(0x0040, 0x008B, config_data);
392 while (!floppy_read_id(drive)) {
393 if ((config_data & 0xC0) == 0x80) {
394 // If even 250 Kbps failed, we can't do much
395 break;
396 }
397 switch (config_data & 0xC0) {
398 case 0xC0: // 1 Mbps
399 config_data = config_data & 0x3F | 0x00;
400 break;
401 case 0x00: // 500 Kbps
402 config_data = config_data & 0x3F | 0x40;
403 break;
404 case 0x40: // 300 Kbps
405 config_data = config_data & 0x3F | 0x80;
406 break;
407 }
408 write_byte(0x0040, 0x008B, config_data);
409 }
410
411 if (drive == 0)
412 media_state_offset = 0x0090;
413 else
414 media_state_offset = 0x0091;
415 write_byte(0x0040, 0x008B, config_data);
416 write_byte(0x0040, media_state_offset, media_state);
417
418 return retval;
419}
420
421
422bx_bool floppy_drive_exists(uint16_t drive)
423{
424 uint8_t drive_type;
425
426 // check CMOS to see if drive exists
427 // @todo: break out drive type determination
428 drive_type = inb_cmos(0x10);
429 if (drive == 0)
430 drive_type >>= 4;
431 else
432 drive_type &= 0x0f;
433 return drive_type != 0;
434}
435
436//@todo: put in a header
437#define AX r.gr.u.r16.ax
438#define BX r.gr.u.r16.bx
439#define CX r.gr.u.r16.cx
440#define DX r.gr.u.r16.dx
441#define SI r.gr.u.r16.si
442#define DI r.gr.u.r16.di
443#define BP r.gr.u.r16.bp
444#define ELDX r.gr.u.r16.sp
445#define DS r.ds
446#define ES r.es
447#define FLAGS r.ra.flags.u.r16.flags
448
449void BIOSCALL int13_diskette_function(disk_regs_t r)
450{
451 uint8_t drive, num_sectors, track, sector, head;
452 uint16_t base_address, base_count, base_es;
453 uint8_t page, mode_register, val8, media_state;
454 uint8_t return_status[7];
455 uint8_t drive_type, num_floppies, ah;
456 uint16_t last_addr;
457 int i;
458
459 BX_DEBUG_INT13_FL("%s: AX=%04x BX=%04x CX=%04x DX=%04x ES=%04x\n", __func__, AX, BX, CX, DX, ES);
460
461 ah = GET_AH();
462
463 switch ( ah ) {
464 case 0x00: // diskette controller reset
465 BX_DEBUG_INT13_FL("floppy f00\n");
466 drive = GET_ELDL();
467 if (drive > 1) {
468 SET_AH(1); // invalid param
469 set_diskette_ret_status(1);
470 SET_CF();
471 return;
472 }
473 // @todo: break out drive type determination
474 drive_type = inb_cmos(0x10);
475 if (drive == 0)
476 drive_type >>= 4;
477 else
478 drive_type &= 0x0f;
479 if (drive_type == 0) {
480 SET_AH(0x80); // drive not responding
481 set_diskette_ret_status(0x80);
482 SET_CF();
483 return;
484 }
485
486 // force re-calibration etc.
487 write_byte(0x0040, 0x003e, 0);
488
489 SET_AH(0);
490 set_diskette_ret_status(0);
491 CLEAR_CF(); // successful
492 set_diskette_current_cyl(drive, 0); // current cylinder
493 return;
494
495 case 0x01: // Read Diskette Status
496 CLEAR_CF();
497 val8 = read_byte(0x0000, 0x0441);
498 SET_AH(val8);
499 if (val8) {
500 SET_CF();
501 }
502 return;
503
504 case 0x02: // Read Diskette Sectors
505 case 0x03: // Write Diskette Sectors
506 case 0x04: // Verify Diskette Sectors
507 num_sectors = GET_AL();
508 track = GET_CH();
509 sector = GET_CL();
510 head = GET_DH();
511 drive = GET_ELDL();
512
513 if ( (drive > 1) || (head > 1) ||
514 (num_sectors == 0) || (num_sectors > 72) ) {
515 BX_INFO("%s: drive>1 || head>1 ...\n", __func__);
516 SET_AH(1);
517 set_diskette_ret_status(1);
518 SET_AL(0); // no sectors read
519 SET_CF(); // error occurred
520 return;
521 }
522
523 // see if drive exists
524 if (floppy_drive_exists(drive) == 0) {
525 SET_AH(0x80); // not responding
526 set_diskette_ret_status(0x80);
527 SET_AL(0); // no sectors read
528 SET_CF(); // error occurred
529 return;
530 }
531
532 // see if media in drive, and type is known
533 if (floppy_media_known(drive) == 0) {
534 if (floppy_media_sense(drive) == 0) {
535 SET_AH(0x0C); // Media type not found
536 set_diskette_ret_status(0x0C);
537 SET_AL(0); // no sectors read
538 SET_CF(); // error occurred
539 return;
540 }
541 }
542
543 if (ah == 0x02) {
544 // Read Diskette Sectors
545
546 //-----------------------------------
547 // set up DMA controller for transfer
548 //-----------------------------------
549
550 // es:bx = pointer to where to place information from diskette
551 // port 04: DMA-1 base and current address, channel 2
552 // port 05: DMA-1 base and current count, channel 2
553 // @todo: merge/factor out pointer normalization
554 page = (ES >> 12); // upper 4 bits
555 base_es = (ES << 4); // lower 16bits contributed by ES
556 base_address = base_es + BX; // lower 16 bits of address
557 // contributed by ES:BX
558 if ( base_address < base_es ) {
559 // in case of carry, adjust page by 1
560 page++;
561 }
562 base_count = (num_sectors * 512) - 1;
563
564 // check for 64K boundary overrun
565 last_addr = base_address + base_count;
566 if (last_addr < base_address) {
567 SET_AH(0x09);
568 set_diskette_ret_status(0x09);
569 SET_AL(0); // no sectors read
570 SET_CF(); // error occurred
571 return;
572 }
573
574 BX_DEBUG_INT13_FL("masking DMA-1 c2\n");
575 outb(0x000a, 0x06);
576
577 BX_DEBUG_INT13_FL("clear flip-flop\n");
578 outb(0x000c, 0x00); // clear flip-flop
579 outb(0x0004, base_address);
580 outb(0x0004, base_address>>8);
581 BX_DEBUG_INT13_FL("clear flip-flop\n");
582 outb(0x000c, 0x00); // clear flip-flop
583 outb(0x0005, base_count);
584 outb(0x0005, base_count>>8);
585 BX_DEBUG_INT13_FL("xfer buf %x bytes at %x:%x\n",
586 base_count, page, base_address);
587
588 // port 0b: DMA-1 Mode Register
589 mode_register = 0x46; // single mode, increment, autoinit disable,
590 // transfer type=write, channel 2
591 BX_DEBUG_INT13_FL("setting mode register\n");
592 outb(0x000b, mode_register);
593
594 BX_DEBUG_INT13_FL("setting page register\n");
595 // port 81: DMA-1 Page Register, channel 2
596 outb(0x0081, page);
597
598 BX_DEBUG_INT13_FL("unmasking DMA-1 c2\n");
599 outb(0x000a, 0x02); // unmask channel 2
600
601 //--------------------------------------
602 // set up floppy controller for transfer
603 //--------------------------------------
604 floppy_prepare_controller(drive);
605
606 // send read-normal-data command (9 bytes) to controller
607 outb(0x03f5, 0xe6); // e6: read normal data
608 outb(0x03f5, (head << 2) | drive); // HD DR1 DR2
609 outb(0x03f5, track);
610 outb(0x03f5, head);
611 outb(0x03f5, sector);
612 outb(0x03f5, 2); // 512 byte sector size
613 outb(0x03f5, sector + num_sectors - 1); // last sector to read on track
614 outb(0x03f5, 0); // Gap length
615 outb(0x03f5, 0xff); // Gap length
616
617#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
618 // turn on interrupts
619 int_enable();
620
621 // wait on 40:3e bit 7 to become 1 or timeout (latter isn't armed so it won't happen)
622 do {
623 val8 = read_byte(0x0040, 0x0040);
624 if (val8 == 0) {
625 floppy_reset_controller();
626 SET_AH(0x80); // drive not ready (timeout)
627 set_diskette_ret_status(0x80);
628 SET_AL(0); // no sectors read
629 SET_CF(); // error occurred
630 return;
631 }
632 val8 = (read_byte(0x0040, 0x003e) & 0x80);
633 } while ( val8 == 0 );
634
635 val8 = 0; // separate asm from while() loop
636 // turn off interrupts
637 int_disable();
638
639 // set 40:3e bit 7 to 0
640 val8 = read_byte(0x0040, 0x003e);
641 val8 &= 0x7f;
642 write_byte(0x0040, 0x003e, val8);
643
644#else
645 val8 = floppy_wait_for_interrupt_or_timeout();
646 if (val8 == 0) { /* Note! Interrupts enabled in this branch. */
647 floppy_reset_controller();
648 SET_AH(0x80); // drive not ready (timeout)
649 set_diskette_ret_status(0x80);
650 SET_AL(0); // no sectors read
651 SET_CF(); // error occurred
652 return;
653 }
654#endif
655
656 // check port 3f4 for accessibility to status bytes
657 val8 = inb(0x3f4);
658 if ( (val8 & 0xc0) != 0xc0 )
659 BX_PANIC("%s: ctrl not ready\n", __func__);
660
661 // read 7 return status bytes from controller and store in BDA
662 for (i = 0; i < 7; ++i) {
663 return_status[i] = inb(0x3f5);
664 write_byte(0x0040, 0x0042 + i, return_status[i]);
665 }
666
667 if ( (return_status[0] & 0xc0) != 0 ) {
668 SET_AH(0x20);
669 set_diskette_ret_status(0x20);
670 SET_AL(0); // no sectors read
671 SET_CF(); // error occurred
672 return;
673 }
674
675#ifdef DMA_WORKAROUND
676 rep_movsw(ES :> BX, ES :> BX, num_sectors * 512 / 2);
677#endif
678 // ??? should track be new val from return_status[3] ?
679 set_diskette_current_cyl(drive, track);
680 // AL = number of sectors read (same value as passed)
681 SET_AH(0x00); // success
682 CLEAR_CF(); // success
683 return;
684 } else if (ah == 0x03) {
685 // Write Diskette Sectors
686
687 //-----------------------------------
688 // set up DMA controller for transfer
689 //-----------------------------------
690
691 // es:bx = pointer to where to place information from diskette
692 // port 04: DMA-1 base and current address, channel 2
693 // port 05: DMA-1 base and current count, channel 2
694 // @todo: merge/factor out pointer normalization
695 page = (ES >> 12); // upper 4 bits
696 base_es = (ES << 4); // lower 16bits contributed by ES
697 base_address = base_es + BX; // lower 16 bits of address
698 // contributed by ES:BX
699 if ( base_address < base_es ) {
700 // in case of carry, adjust page by 1
701 page++;
702 }
703 base_count = (num_sectors * 512) - 1;
704
705 // check for 64K boundary overrun
706 last_addr = base_address + base_count;
707 if (last_addr < base_address) {
708 SET_AH(0x09);
709 set_diskette_ret_status(0x09);
710 SET_AL(0); // no sectors read
711 SET_CF(); // error occurred
712 return;
713 }
714
715 BX_DEBUG_INT13_FL("masking DMA-1 c2\n");
716 outb(0x000a, 0x06);
717
718 outb(0x000c, 0x00); // clear flip-flop
719 outb(0x0004, base_address);
720 outb(0x0004, base_address>>8);
721 outb(0x000c, 0x00); // clear flip-flop
722 outb(0x0005, base_count);
723 outb(0x0005, base_count>>8);
724 BX_DEBUG_INT13_FL("xfer buf %x bytes at %x:%x\n",
725 base_count, page, base_address);
726
727 // port 0b: DMA-1 Mode Register
728 mode_register = 0x4a; // single mode, increment, autoinit disable,
729 // transfer type=read, channel 2
730 outb(0x000b, mode_register);
731
732 // port 81: DMA-1 Page Register, channel 2
733 outb(0x0081, page);
734
735 BX_DEBUG_INT13_FL("unmasking DMA-1 c2\n");
736 outb(0x000a, 0x02);
737
738 //--------------------------------------
739 // set up floppy controller for transfer
740 //--------------------------------------
741 floppy_prepare_controller(drive);
742
743 // send write-normal-data command (9 bytes) to controller
744 outb(0x03f5, 0xc5); // c5: write normal data
745 outb(0x03f5, (head << 2) | drive); // HD DR1 DR2
746 outb(0x03f5, track);
747 outb(0x03f5, head);
748 outb(0x03f5, sector);
749 outb(0x03f5, 2); // 512 byte sector size
750 outb(0x03f5, sector + num_sectors - 1); // last sector to write on track
751 outb(0x03f5, 0); // Gap length
752 outb(0x03f5, 0xff); // Gap length
753
754#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
755 // turn on interrupts
756 int_enable();
757
758 // wait on 40:3e bit 7 to become 1
759 do {
760 val8 = read_byte(0x0040, 0x0040);
761 if (val8 == 0) {
762 floppy_reset_controller();
763 SET_AH(0x80); // drive not ready (timeout)
764 set_diskette_ret_status(0x80);
765 SET_AL(0); // no sectors written
766 SET_CF(); // error occurred
767 return;
768 }
769 val8 = (read_byte(0x0040, 0x003e) & 0x80);
770 } while ( val8 == 0 );
771
772 val8 = 0; // separate asm from while() loop @todo: why??
773 // turn off interrupts
774 int_disable();
775
776 // set 40:3e bit 7 to 0
777 val8 = read_byte(0x0040, 0x003e);
778 val8 &= 0x7f;
779 write_byte(0x0040, 0x003e, val8);
780#else
781 val8 = floppy_wait_for_interrupt_or_timeout();
782 if (val8 == 0) { /* Note! Interrupts enabled in this branch. */
783 floppy_reset_controller();
784 SET_AH(0x80); // drive not ready (timeout)
785 set_diskette_ret_status(0x80);
786 SET_AL(0); // no sectors written
787 SET_CF(); // error occurred
788 return;
789 }
790#endif
791
792 // check port 3f4 for accessibility to status bytes
793 val8 = inb(0x3f4);
794 if ( (val8 & 0xc0) != 0xc0 )
795 BX_PANIC("%s: ctrl not ready\n", __func__);
796
797 // read 7 return status bytes from controller and store in BDA
798 for (i = 0; i < 7; ++i) {
799 return_status[i] = inb(0x3f5);
800 write_byte(0x0040, 0x0042 + i, return_status[i]);
801 }
802
803 if ( (return_status[0] & 0xc0) != 0 ) {
804 if ( (return_status[1] & 0x02) != 0 ) {
805 // diskette not writable.
806 // AH=status code=0x03 (tried to write on write-protected disk)
807 // AL=number of sectors written=0
808 AX = 0x0300;
809 } else {
810 // Some other problem occurred.
811 AX = 0x0100;
812 }
813 SET_CF();
814 return;
815 }
816
817 // ??? should track be new val from return_status[3] ?
818 set_diskette_current_cyl(drive, track);
819 // AL = number of sectors read (same value as passed)
820 SET_AH(0x00); // success
821 CLEAR_CF(); // success
822 return;
823 } else { // if (ah == 0x04)
824 // Verify Diskette Sectors
825
826 // ??? should track be new val from return_status[3] ?
827 set_diskette_current_cyl(drive, track);
828 // AL = number of sectors verified (same value as passed)
829 CLEAR_CF(); // success
830 SET_AH(0x00); // success
831 return;
832 }
833 break;
834
835 case 0x05: // format diskette track
836 BX_DEBUG_INT13_FL("floppy f05\n");
837
838 num_sectors = GET_AL();
839 track = GET_CH();
840 head = GET_DH();
841 drive = GET_ELDL();
842
843 if ((drive > 1) || (head > 1) || (track > 79) ||
844 (num_sectors == 0) || (num_sectors > 18)) {
845 SET_AH(1);
846 set_diskette_ret_status(1);
847 SET_CF(); // error occurred
848 }
849
850 // see if drive exists
851 if (floppy_drive_exists(drive) == 0) {
852 SET_AH(0x80); // drive not responding
853 set_diskette_ret_status(0x80);
854 SET_CF(); // error occurred
855 return;
856 }
857
858 // see if media in drive, and type is known
859 if (floppy_media_known(drive) == 0) {
860 if (floppy_media_sense(drive) == 0) {
861 SET_AH(0x0C); // Media type not found
862 set_diskette_ret_status(0x0C);
863 SET_AL(0); // no sectors read
864 SET_CF(); // error occurred
865 return;
866 }
867 }
868
869 // set up DMA controller for transfer
870 // @todo: merge/factor out pointer normalization
871 page = (ES >> 12); // upper 4 bits
872 base_es = (ES << 4); // lower 16bits contributed by ES
873 base_address = base_es + BX; // lower 16 bits of address
874 // contributed by ES:BX
875 if ( base_address < base_es ) {
876 // in case of carry, adjust page by 1
877 page++;
878 }
879 base_count = (num_sectors * 4) - 1;
880
881 // check for 64K boundary overrun
882 last_addr = base_address + base_count;
883 if (last_addr < base_address) {
884 SET_AH(0x09);
885 set_diskette_ret_status(0x09);
886 SET_AL(0); // no sectors read
887 SET_CF(); // error occurred
888 return;
889 }
890
891 outb(0x000a, 0x06);
892 outb(0x000c, 0x00); // clear flip-flop
893 outb(0x0004, base_address);
894 outb(0x0004, base_address>>8);
895 outb(0x000c, 0x00); // clear flip-flop
896 outb(0x0005, base_count);
897 outb(0x0005, base_count>>8);
898 mode_register = 0x4a; // single mode, increment, autoinit disable,
899 // transfer type=read, channel 2
900 outb(0x000b, mode_register);
901 // port 81: DMA-1 Page Register, channel 2
902 outb(0x0081, page);
903 outb(0x000a, 0x02);
904
905 // set up floppy controller for transfer
906 floppy_prepare_controller(drive);
907
908 // send seek command to controller
909 outb(0x03f5, 0x0f); // 0f: seek
910 outb(0x03f5, (head << 2) | drive); // HD DR1 DR2
911 outb(0x03f5, track);
912
913 // send format-track command (6 bytes) to controller
914 outb(0x03f5, 0x4d); // 4d: format track
915 outb(0x03f5, (head << 2) | drive); // HD DR1 DR2
916 outb(0x03f5, 2); // 512 byte sector size
917 outb(0x03f5, num_sectors); // number of sectors per track
918 outb(0x03f5, 0); // Gap length
919 outb(0x03f5, 0xf6); // Fill byte
920
921#ifdef VBOX_WITH_FLOPPY_IRQ_POLLING
922 // turn on interrupts
923 int_enable();
924
925 // wait on 40:3e bit 7 to become 1
926 do {
927 val8 = read_byte(0x0040, 0x0040);
928 if (val8 == 0) {
929 floppy_reset_controller();
930 SET_AH(0x80); // drive not ready (timeout)
931 set_diskette_ret_status(0x80);
932 SET_CF(); // error occurred
933 return;
934 }
935 val8 = (read_byte(0x0040, 0x003e) & 0x80);
936 } while ( val8 == 0 );
937
938 val8 = 0; // separate asm from while() loop
939 // turn off interrupts
940 int_disable();
941
942 // set 40:3e bit 7 to 0
943 val8 = read_byte(0x0040, 0x003e);
944 val8 &= 0x7f;
945 write_byte(0x0040, 0x003e, val8);
946#else
947 val8 = floppy_wait_for_interrupt_or_timeout();
948 if (val8 == 0) { /* Note! Interrupts enabled in this branch. */
949 floppy_reset_controller();
950 SET_AH(0x80); // drive not ready (timeout)
951 set_diskette_ret_status(0x80);
952 SET_CF(); // error occurred
953 return;
954 }
955#endif
956
957 // check port 3f4 for accessibility to status bytes
958 val8 = inb(0x3f4);
959 if ( (val8 & 0xc0) != 0xc0 )
960 BX_PANIC("%s: ctrl not ready\n", __func__);
961
962 // read 7 return status bytes from controller and store in BDA
963 for (i = 0; i < 7; ++i) {
964 return_status[i] = inb(0x3f5);
965 write_byte(0x0040, 0x0042 + i, return_status[i]);
966 }
967
968 if ( (return_status[0] & 0xc0) != 0 ) {
969 if ( (return_status[1] & 0x02) != 0 ) {
970 // diskette not writable.
971 // AH=status code=0x03 (tried to write on write-protected disk)
972 // AL=number of sectors written=0
973 AX = 0x0300;
974 SET_CF();
975 return;
976 } else {
977 BX_PANIC("%s: write error\n", __func__);
978 }
979 }
980
981 SET_AH(0);
982 set_diskette_ret_status(0);
983 set_diskette_current_cyl(drive, 0);
984 CLEAR_CF(); // successful
985 return;
986
987
988 case 0x08: // read diskette drive parameters
989 BX_DEBUG_INT13_FL("floppy f08\n");
990 drive = GET_ELDL();
991
992 if (drive > 1) {
993 AX = 0;
994 BX = 0;
995 CX = 0;
996 DX = 0;
997 ES = 0;
998 DI = 0;
999 SET_DL(num_floppies);
1000 SET_CF();
1001 return;
1002 }
1003
1004 // @todo: break out drive type determination
1005 drive_type = inb_cmos(0x10);
1006 num_floppies = 0;
1007 if (drive_type & 0xf0)
1008 num_floppies++;
1009 if (drive_type & 0x0f)
1010 num_floppies++;
1011
1012 if (drive == 0)
1013 drive_type >>= 4;
1014 else
1015 drive_type &= 0x0f;
1016
1017 SET_BH(0);
1018 SET_BL(drive_type);
1019 SET_AH(0);
1020 SET_AL(0);
1021 SET_DL(num_floppies);
1022 SET_DH(1); // max head #
1023
1024 switch (drive_type) {
1025 case 0: // none
1026 CX = 0;
1027 SET_DH(0); // max head #
1028 break;
1029
1030 case 1: // 360KB, 5.25"
1031 CX = 0x2709; // 40 tracks, 9 sectors
1032 break;
1033
1034 case 2: // 1.2MB, 5.25"
1035 CX = 0x4f0f; // 80 tracks, 15 sectors
1036 break;
1037
1038 case 3: // 720KB, 3.5"
1039 CX = 0x4f09; // 80 tracks, 9 sectors
1040 break;
1041
1042 case 4: // 1.44MB, 3.5"
1043 CX = 0x4f12; // 80 tracks, 18 sectors
1044 break;
1045
1046 case 5: // 2.88MB, 3.5"
1047 CX = 0x4f24; // 80 tracks, 36 sectors
1048 break;
1049
1050 case 14: // 15.6 MB 3.5" (fake)
1051 CX = 0xfe3f; // 255 tracks, 63 sectors
1052 break;
1053
1054 case 15: // 63.5 MB 3.5" (fake)
1055 CX = 0xfeff; // 255 tracks, 255 sectors - This works because the cylinder
1056 break; // and sectors limits/encoding aren't checked by the BIOS
1057 // due to copy protection schemes and such stuff.
1058
1059 default: // ?
1060 BX_PANIC("%s: bad floppy type\n", __func__);
1061 }
1062
1063 /* set es & di to point to 11 byte diskette param table in ROM */
1064 ES = 0xF000; // @todo: any way to make this relocatable?
1065 DI = get_floppy_dpt(drive_type);
1066 CLEAR_CF(); // success
1067 /* disk status not changed upon success */
1068 return;
1069
1070 case 0x15: // read diskette drive type
1071 BX_DEBUG_INT13_FL("floppy f15\n");
1072 drive = GET_ELDL();
1073 if (drive > 1) {
1074 SET_AH(0); // only 2 drives supported
1075 // set_diskette_ret_status here ???
1076 SET_CF();
1077 return;
1078 }
1079 // @todo: break out drive type determination
1080 drive_type = inb_cmos(0x10);
1081 if (drive == 0)
1082 drive_type >>= 4;
1083 else
1084 drive_type &= 0x0f;
1085 CLEAR_CF(); // successful, not present
1086 if (drive_type==0) {
1087 SET_AH(0); // drive not present
1088 } else if (drive_type > 1) {
1089 SET_AH(2); // drive present, supports change line
1090 } else {
1091 SET_AH(1); // drive present, does not support change line
1092 }
1093
1094 return;
1095
1096 case 0x16: // get diskette change line status
1097 BX_DEBUG_INT13_FL("floppy f16\n");
1098 drive = GET_ELDL();
1099 if (drive > 1) {
1100 SET_AH(0x01); // invalid drive
1101 set_diskette_ret_status(0x01);
1102 SET_CF();
1103 return;
1104 }
1105
1106 SET_AH(0x06); // change line not supported
1107 set_diskette_ret_status(0x06);
1108 SET_CF();
1109 return;
1110
1111 case 0x17: // set diskette type for format(old)
1112 BX_DEBUG_INT13_FL("floppy f17\n");
1113 // NOTE: 1.44M diskette not supported by this function, use INT14h/18h instead.
1114 // Drive number (0 or 1) values allowed
1115 drive = GET_ELDL();
1116
1117 // Format type (AL)
1118 // 00 - NOT USED
1119 // 01 - DISKETTE 360K IN 360K DRIVE
1120 // 02 - DISKETTE 360K IN 1.2M DRIVE
1121 // 03 - DISKETTE 1.2M IN 1.2M DRIVE
1122 // 04 - DISKETTE 720K IN 720K DRIVE
1123 val8 = GET_AL();
1124
1125 BX_DEBUG_INT13_FL("floppy f17 - drive: %d, format type: %d\n", drive, val8);
1126
1127 if (drive > 1) {
1128 SET_AH(0x01); // invalid drive
1129 set_diskette_ret_status(0x01); // bad parameter
1130 SET_CF();
1131 return;
1132 }
1133
1134 // see if drive exists
1135 if (floppy_drive_exists(drive) == 0) {
1136 SET_AH(0x80); // not responding/time out
1137 set_diskette_ret_status(0x80);
1138 SET_CF();
1139 return;
1140 }
1141
1142 // Get current drive state. Set 'base_address' to media status offset address
1143 base_address = (drive) ? 0x0091 : 0x0090;
1144 media_state = read_byte(0x0040, base_address);
1145
1146 // Mask out (clear) bits 4-7 (4:media type established, 5:double stepping, 6-7:data rate)
1147 media_state &= 0x0f;
1148
1149 switch (val8) {
1150 case 1:
1151 // 360K media in 360K drive
1152 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1153 break;
1154 case 2:
1155 // 360K media in 1.2M drive
1156 media_state |= 0x70; // 0111 0000 (media type established, double stepping, 300 kbps)
1157 break;
1158 case 3:
1159 // 1.2M media in 1.2M drive
1160 media_state |= 0x10; // 0001 0000 (media type established, 500 kbps)
1161 break;
1162 case 4:
1163 // 720K media in 720K drive
1164 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1165 break;
1166 default:
1167 // bad parameter
1168 SET_AH(0x01); // invalid format mode parameter
1169 set_diskette_ret_status(0x01);
1170 SET_CF();
1171 return;
1172 }
1173
1174 // Update media status
1175 write_byte(0x0040, base_address, media_state);
1176 BX_DEBUG_INT13_FL("floppy f17 - media status set to: %02x\n", media_state);
1177
1178 // return success!
1179 SET_AH(0);
1180 set_diskette_ret_status(0);
1181 CLEAR_CF();
1182 return;
1183
1184 case 0x18: // set diskette type for format(new)
1185 BX_DEBUG_INT13_FL("floppy f18\n");
1186 // Set Media Type for Format. Verifies that the device supports a specific geometry.
1187 // Unlike INT13h/17h, this service supports higher capacity drives (1.44M and 2.88M).
1188 // Drive number (0 or 1) values allowed
1189 drive = GET_ELDL();
1190
1191 val8 = GET_CL();
1192 num_sectors = val8 & 0x3f; // max sector number per cylinder
1193 track = ((val8 >> 6) << 8) + GET_CH(); // max cylinder number (max cylinders - 1)
1194
1195 BX_DEBUG_INT13_FL("floppy f18 - drive: %d, max cylinder/track number: %d, sectors-per-tracks: %d\n",
1196 drive, track, num_sectors);
1197
1198 if (drive > 1) {
1199 SET_AH(0x01); // invalid drive
1200 set_diskette_ret_status(0x01);
1201 SET_CF();
1202 return;
1203 }
1204
1205 // see if drive exists
1206 if (floppy_drive_exists(drive) == 0) {
1207 SET_AH(0x80); // not responding/time out
1208 set_diskette_ret_status(0x80);
1209 SET_CF();
1210 return;
1211 }
1212
1213 // see if media in drive, and media type is known
1214 if (floppy_media_known(drive) == 0) {
1215 if (floppy_media_sense(drive) == 0) {
1216 SET_AH(0x0C); // drive/media type unknown
1217 set_diskette_ret_status(0x0C);
1218 SET_CF();
1219 return;
1220 }
1221 }
1222
1223 // @todo: break out drive type determination
1224 drive_type = inb_cmos(0x10);
1225 if (drive == 0)
1226 drive_type >>= 4;
1227 else
1228 drive_type &= 0x0f;
1229
1230 // Get current drive state. Set 'base_address' to media status offset address
1231 base_address = (drive) ? 0x0091 : 0x0090;
1232 media_state = read_byte(0x0040, base_address);
1233
1234 // Mask out (clear) bits 4-7 (4:media type established, 5:double stepping, 6-7:data rate)
1235 media_state &= 0x0f;
1236
1237 switch (drive_type) {
1238 case 1: // 360KB, 5.25"
1239 if (track == 39 && num_sectors == 9)
1240 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1241
1242 break;
1243 case 2: // 1.2MB, 5.25"
1244 if (track == 39 && num_sectors == 9) { // 360K disk in 1.2M drive
1245 media_state |= 0x70; // 0111 0000 (media type established, double stepping, 300 kbps)
1246 } else if (track == 79 && num_sectors == 15) { // 1.2M disk in 1.2M drive
1247 media_state |= 0x10; // 0001 0000 (media type established, 500 kbps)
1248 }
1249 break;
1250 case 3: // 720KB, 3.5"
1251 if (track == 79 && num_sectors == 9)
1252 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1253
1254 break;
1255 case 4: // 1.44MB, 3.5"
1256 if (track == 79) {
1257 if (num_sectors == 9) { // 720K disk in 1.44M drive
1258 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1259 } else if (num_sectors == 18) { // 1.44M disk in 1.44M drive
1260 media_state |= 0x10; // 0001 0000 (media type established, 500 kbps)
1261 }
1262 }
1263 break;
1264 case 5: // 2.88MB, 3.5"
1265 if (track == 79) {
1266 if (num_sectors == 9) { // 720K disk in 2.88M drive
1267 media_state |= 0x90; // 1001 0000 (media type established, 250 kbps)
1268 } else if (num_sectors == 18) { // 1.44M disk in 2.88M drive
1269 media_state |= 0x10; // 0001 0000 (media type established, 500 kbps)
1270 } else if (num_sectors == 36) { // 2.88M disk in 2.88M drive
1271 media_state |= 0xD0; // 1101 0000 (media type established, 1 Mbps)
1272 }
1273 }
1274 break;
1275 default:
1276 break;
1277 }
1278
1279 // Error if bit 4 (media type established) has not just been set above.
1280 if (((media_state >> 4) & 0x01) == 0) {
1281 // Error - assume requested tracks/sectors-per-track not supported
1282 // for current drive type - or drive type is unknown!
1283 SET_AH(0x0C);
1284 set_diskette_ret_status(0x0C);
1285 SET_CF();
1286 return;
1287 }
1288
1289 // Update media status
1290 write_byte(0x0040, base_address, media_state);
1291
1292 // set es & di to point to 11 byte diskette param table in ROM
1293 ES = 0xF000; // @todo: any way to make this relocatable?
1294 DI = get_floppy_dpt(drive_type);
1295
1296 // return success!
1297 SET_AH(0);
1298 set_diskette_ret_status(0);
1299 CLEAR_CF();
1300 return;
1301
1302 default:
1303 BX_INFO("%s: unsupported AH=%02x\n", __func__, GET_AH());
1304
1305 // if ( (ah==0x20) || ((ah>=0x41) && (ah<=0x49)) || (ah==0x4e) ) {
1306 SET_AH(0x01); // ???
1307 set_diskette_ret_status(1);
1308 SET_CF();
1309 return;
1310 // }
1311 }
1312}
1313
1314#else // #if BX_SUPPORT_FLOPPY
1315
1316void BIOSCALL int13_diskette_function(disk_regs_t r)
1317{
1318 uint8_t val8;
1319
1320 switch ( GET_AH() ) {
1321
1322 case 0x01: // Read Diskette Status
1323 CLEAR_CF();
1324 val8 = read_byte(0x0000, 0x0441);
1325 SET_AH(val8);
1326 if (val8) {
1327 SET_CF();
1328 }
1329 return;
1330
1331 default:
1332 SET_CF();
1333 write_byte(0x0000, 0x0441, 0x01);
1334 SET_AH(0x01);
1335 }
1336}
1337
1338#endif // #if BX_SUPPORT_FLOPPY
1339
1340#if 0
1341void determine_floppy_media(uint16_t drive)
1342{
1343 uint8_t val8, DOR, ctrl_info;
1344
1345 ctrl_info = read_byte(0x0040, 0x008F);
1346 if (drive==1)
1347 ctrl_info >>= 4;
1348 else
1349 ctrl_info &= 0x0f;
1350
1351#if 0
1352 if (drive == 0) {
1353 DOR = 0x1c; // DOR: drive0 motor on, DMA&int enabled, normal op, drive select 0
1354 }
1355 else {
1356 DOR = 0x2d; // DOR: drive1 motor on, DMA&int enabled, normal op, drive select 1
1357 }
1358#endif
1359
1360 if ( (ctrl_info & 0x04) != 0x04 ) {
1361 // Drive not determined means no drive exists, done.
1362 return;
1363 }
1364
1365#if 0
1366 // check Main Status Register for readiness
1367 val8 = inb(0x03f4) & 0x80; // Main Status Register
1368 if (val8 != 0x80)
1369 BX_PANIC("d_f_m: MRQ bit not set\n");
1370
1371 // change line
1372
1373 // existing BDA values
1374
1375 // turn on drive motor
1376 outb(0x03f2, DOR); // Digital Output Register
1377 //
1378#endif
1379 BX_PANIC("d_f_m: OK so far\n");
1380}
1381#endif
1382
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