Home | History | Annotate | Line # | Download | only in lib
bios_disk.S revision 1.18.78.1
      1 /*	$NetBSD: bios_disk.S,v 1.18.78.1 2010/03/11 15:02:30 yamt Exp $	*/
      2 
      3 /*
      4  * Ported to boot 386BSD by Julian Elischer (julian (at) tfs.com) Sept 1992
      5  *
      6  * Mach Operating System
      7  * Copyright (c) 1992, 1991 Carnegie Mellon University
      8  * All Rights Reserved.
      9  *
     10  * Permission to use, copy, modify and distribute this software and its
     11  * documentation is hereby granted, provided that both the copyright
     12  * notice and this permission notice appear in all copies of the
     13  * software, derivative works or modified versions, and any portions
     14  * thereof, and that both notices appear in supporting documentation.
     15  *
     16  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
     17  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
     18  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
     19  *
     20  * Carnegie Mellon requests users of this software to return to
     21  *
     22  *  Software Distribution Coordinator  or  Software.Distribution (at) CS.CMU.EDU
     23  *  School of Computer Science
     24  *  Carnegie Mellon University
     25  *  Pittsburgh PA 15213-3890
     26  *
     27  * any improvements or extensions that they make and grant Carnegie Mellon
     28  * the rights to redistribute these changes.
     29  */
     30 
     31 /*
     32   Copyright 1988, 1989, 1990, 1991, 1992
     33    by Intel Corporation, Santa Clara, California.
     34 
     35                 All Rights Reserved
     36 
     37 Permission to use, copy, modify, and distribute this software and
     38 its documentation for any purpose and without fee is hereby
     39 granted, provided that the above copyright notice appears in all
     40 copies and that both the copyright notice and this permission notice
     41 appear in supporting documentation, and that the name of Intel
     42 not be used in advertising or publicity pertaining to distribution
     43 of the software without specific, written prior permission.
     44 
     45 INTEL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
     46 INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS,
     47 IN NO EVENT SHALL INTEL BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
     48 CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
     49 LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
     50 NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
     51 WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
     52 */
     53 
     54 /* extracted from netbsd:sys/arch/i386/boot/bios.S */
     55 
     56 #include <machine/asm.h>
     57 
     58 /*
     59  * BIOS call "INT 0x13 Function 0x0" to reset the disk subsystem
     60  *	Call with	%ah = 0x0
     61  *			%dl = drive (0x80 for hard disk, 0x0 for floppy disk)
     62  *	Return:
     63  *			%al = 0x0 on success; err code on failure
     64  */
     65 ENTRY(biosdisk_reset)
     66 	pushl	%ebp
     67 	movl	%esp, %ebp
     68 	pushl	%ebx
     69 	push	%edx
     70 	push	%edi
     71 
     72 	movb	8(%ebp), %dl	# device
     73 
     74 	call	_C_LABEL(prot_to_real)	# enter real mode
     75 	.code16
     76 
     77 	movb	$0x0, %ah	# subfunction
     78 	int	$0x13
     79 	setc	%bl
     80 	movb	%ah, %bh	# save error code
     81 
     82 	calll	_C_LABEL(real_to_prot) # back to protected mode
     83 	.code32
     84 
     85 	xorl	%eax, %eax
     86 	movw	%bx, %ax	# return value in %ax
     87 
     88 	pop	%edi
     89 	pop	%edx
     90 	popl	%ebx
     91 	popl	%ebp
     92 	ret
     93 
     94 /*
     95  * BIOS call "INT 0x13 Function 0x2" to read sectors from disk into memory
     96  *	Call with	%ah = 0x2
     97  *			%al = number of sectors
     98  *			%ch = cylinder
     99  *			%cl = sector
    100  *			%dh = head
    101  *			%dl = drive (0x80 for hard disk, 0x0 for floppy disk)
    102  *			%es:%bx = segment:offset of buffer
    103  *	Return:
    104  *			%al = 0x0 on success; err code on failure
    105  *
    106  * biosdisk_read(dev, cyl, head, sect, count, buff_addr);
    107  *
    108  *  Note: On failure, you must reset the disk with biosdisk_reset() before
    109  *        sending another command.
    110  */
    111 ENTRY(biosdisk_read)
    112 	pushl	%ebp
    113 	movl	%esp, %ebp
    114 	pushl	%ebx
    115 	push	%ecx
    116 	push	%edx
    117 	push	%esi
    118 	push	%edi
    119 
    120 	movb	16(%ebp), %dh
    121 	movw	12(%ebp), %cx
    122 	xchgb	%ch, %cl	# cylinder; the highest 2 bits of cyl is in %cl
    123 	rorb	$2, %cl
    124 	movb	20(%ebp), %al
    125 	orb	%al, %cl
    126 	incb	%cl		# sector; sec starts from 1, not 0
    127 	movb	8(%ebp), %dl	# device
    128 	movl	28(%ebp), %ebx	# buffer address (may be >64k)
    129 	movb	24(%ebp), %al	# number of sectors
    130 
    131 	call	_C_LABEL(prot_to_real)	# enter real mode
    132 	.code16
    133 
    134 	push	%bx
    135 	shrl	$4, %ebx	# max segment
    136 	mov	%ds, %si
    137 	add	%si, %bx
    138 	mov	%bx, %es	# %es:%bx now valid buffer address
    139 	pop	%bx
    140 	and	$0xf, %bx	# and min offset - to avoid overrun
    141 
    142 	movb	$0x2, %ah	# subfunction
    143 	int	$0x13
    144 	setc	%al		# error code is in %ah
    145 
    146 	calll	_C_LABEL(real_to_prot) # back to protected mode
    147 	.code32
    148 
    149 	andl	$0xffff, %eax
    150 
    151 	pop	%edi
    152 	pop	%esi
    153 	pop	%edx
    154 	pop	%ecx
    155 	popl	%ebx
    156 	popl	%ebp
    157 	ret
    158 
    159 /*
    160  * biosdisk_getinfo(int dev):  return a word that represents the
    161  *	max number of sectors, heads and cylinders for this device
    162  */
    163 ENTRY(biosdisk_getinfo)
    164 	pushl	%ebp
    165 	movl	%esp, %ebp
    166 	push	%es
    167 	pushl	%ebx
    168 	push	%ecx
    169 	push	%edx
    170 	push	%esi
    171 	push	%edi
    172 
    173 	movb	8(%ebp), %dl		# diskinfo(drive #)
    174 
    175 	call	_C_LABEL(prot_to_real)	# enter real mode
    176 	.code16
    177 
    178 	push	%dx			# save drive #
    179 	movb	$0x08, %ah		# ask for disk info
    180 	int	$0x13
    181 	pop	%bx			# restore drive #
    182 	jnc	ok
    183 
    184 	testb	$0x80, %bl		# is it a hard disk?
    185 	jnz	ok
    186 
    187 	/*
    188 	 * Urk.  Call failed.  It is not supported for floppies by old BIOS's.
    189 	 * Guess it's a 15-sector floppy.  Initialize all the registers for
    190 	 * documentation, although we only need head and sector counts.
    191 	 */
    192 	xorw	%ax, %ax		# set status to success
    193 #	movb	%ah, %bh		# %bh = 0
    194 #	movb	$2, %bl			# %bl bits 0-3 = drive type, 2 = 1.2M
    195 	movb	$79, %ch		# max track
    196 	movb	$15, %cl		# max sector
    197 	movb	$1, %dh			# max head
    198 #	movb	$1, %dl			# # floppy drives installed
    199 	# es:di = parameter table
    200 	# carry = 0
    201 
    202 ok:
    203 	calll	_C_LABEL(real_to_prot)	# back to protected mode
    204 	.code32
    205 
    206 	/* form a longword representing all this gunk */
    207 	shrl	$8, %eax		# clear unnecessary bits
    208 	shll	$24, %eax
    209 	shll	$16, %ecx		# do the same for %ecx
    210 	shrl	$8, %ecx
    211 	movb	%dh, %cl		# max head
    212 	orl	%ecx, %eax		# return value in %eax
    213 
    214 	pop	%edi
    215 	pop	%esi
    216 	pop	%edx
    217 	pop	%ecx
    218 	popl	%ebx
    219 	pop	%es
    220 	popl	%ebp
    221 	ret
    222 
    223 /*
    224  * int biosdisk_int13ext(int dev):
    225  *	check for availibility of int13 extensions.
    226  */
    227 ENTRY(biosdisk_int13ext)
    228 	pushl	%ebp
    229 	movl	%esp, %ebp
    230 	pushl	%ebx
    231 	pushl	%ecx
    232 	pushl	%edx
    233 	pushl	%esi
    234 	pushl	%edi
    235 
    236 	movb	8(%ebp), %dl		# drive #
    237 	movw	$0x55aa, %bx
    238 
    239 	call	_C_LABEL(prot_to_real)	# enter real mode
    240 	.code16
    241 
    242 	movb	$0x41, %ah		# ask for disk info
    243 	int	$0x13
    244 	setnc	%dl
    245 
    246 	calll	_C_LABEL(real_to_prot)	# switch back
    247 	.code32
    248 
    249 	xorl	%eax, %eax
    250 	movb	%dl, %al	# return value in %ax
    251 
    252 	cmpw	$0xaa55, %bx
    253 	sete	%dl
    254 	andb	%dl, %al
    255 
    256 	andb	%cl, %al
    257 
    258 	popl	%edi
    259 	popl	%esi
    260 	popl	%edx
    261 	popl	%ecx
    262 	popl	%ebx
    263 	popl	%ebp
    264 	ret
    265 
    266 /*
    267  * BIOS call "INT 0x13 Function 0x42" to read sectors from disk into memory
    268  *	Call with	%ah = 0x42
    269  *			%ds:%si = parameter block (data buffer address
    270  *				must be a real mode physical address).
    271  *			%dl = drive (0x80 for hard disk, 0x0 for floppy disk)
    272  *	Return:
    273  *			%al = 0x0 on success; err code on failure
    274  */
    275 ENTRY(biosdisk_extread)
    276 	pushl	%ebp
    277 	movl	%esp, %ebp
    278 	pushl	%ebx
    279 	push	%ecx
    280 	push	%edx
    281 	push	%esi
    282 	push	%edi
    283 
    284 	movb	8(%ebp), %dl	# device
    285 	movl	12(%ebp), %esi	# parameter block
    286 
    287 	call	_C_LABEL(prot_to_real)	# enter real mode
    288 	.code16
    289 
    290 	push	%ds
    291 	movl	%esi, %eax
    292 	shrl	$4, %eax
    293 	movw	%ds, %bx
    294 	addw	%bx, %ax
    295 	movw	%ax, %ds
    296 	andw	$0xf, %si
    297 
    298 	movb	$0x42, %ah	# subfunction
    299 	int	$0x13
    300 	setc	%bl
    301 	movb	%ah, %bh	# save error code
    302 	pop	%ds
    303 
    304 	calll	_C_LABEL(real_to_prot) # back to protected mode
    305 	.code32
    306 
    307 	xorl	%eax, %eax
    308 	movw	%bx, %ax	# return value in %ax
    309 
    310 	pop	%edi
    311 	pop	%esi
    312 	pop	%edx
    313 	pop	%ecx
    314 	popl	%ebx
    315 	popl	%ebp
    316 	ret
    317 
    318 ENTRY(biosdisk_getextinfo)
    319 	pushl	%ebp
    320 	movl	%esp, %ebp
    321 	pushl	%ebx
    322 	push	%ecx
    323 	push	%edx
    324 	push	%esi
    325 	push	%edi
    326 
    327 	movb	8(%ebp), %dl	# device
    328 	movl	12(%ebp), %esi	# parameter block
    329 
    330 	call	_C_LABEL(prot_to_real)	# enter real mode
    331 	.code16
    332 
    333 	push	%ds
    334 	movl	%esi, %eax
    335 	shrl	$4, %eax
    336 	andw	$0xf, %si
    337 	movw	%ds, %bx
    338 	addw	%bx, %ax
    339 	movw	%ax, %ds
    340 
    341 	movb	$0x48, %ah	# subfunction
    342 	int	$0x13
    343 	setc	%bl
    344 	pop	%ds
    345 
    346 	calll	_C_LABEL(real_to_prot) # back to protected mode
    347 	.code32
    348 
    349 	xorl	%eax, %eax
    350 	movb	%bl, %al	# return value in %ax
    351 
    352 	pop	%edi
    353 	pop	%esi
    354 	pop	%edx
    355 	pop	%ecx
    356 	popl	%ebx
    357 	popl	%ebp
    358 	ret
    359