• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1/*
2 *  GRUB  --  GRand Unified Bootloader
3 *  Copyright (C) 1999,2000,2001,2002,2004 Free Software Foundation, Inc.
4 *
5 *  This program is free software; you can redistribute it and/or modify
6 *  it under the terms of the GNU General Public License as published by
7 *  the Free Software Foundation; either version 2 of the License, or
8 *  (at your option) any later version.
9 *
10 *  This program is distributed in the hope that it will be useful,
11 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 *  GNU General Public License for more details.
14 *
15 *  You should have received a copy of the GNU General Public License
16 *  along with this program; if not, write to the Free Software
17 *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20
21/*
22 * Note: These functions defined in this file may be called from C.
23 *       Be careful of that you must not modify some registers. Quote
24 *       from gcc-2.95.2/gcc/config/i386/i386.h:
25
26   1 for registers not available across function calls.
27   These must include the FIXED_REGISTERS and also any
28   registers that can be used without being saved.
29   The latter must include the registers where values are returned
30   and the register where structure-value addresses are passed.
31   Aside from that, you can include as many other registers as you like.
32
33  ax,dx,cx,bx,si,di,bp,sp,st,st1,st2,st3,st4,st5,st6,st7,arg
34{  1, 1, 1, 0, 0, 0, 0, 1, 1,  1,  1,  1,  1,  1,  1,  1,  1 }
35 */
36
37#define ASM_FILE
38
39#include "shared.h"
40
41#ifdef STAGE1_5
42# define	ABS(x)	((x) - EXT_C(main) + 0x2200)
43#else
44# define	ABS(x)	((x) - EXT_C(main) + 0x8200)
45#endif
46
47	.file	"asm.S"
48
49	.text
50
51	/* Tell GAS to generate 16-bit instructions so that this code works
52	   in real mode. */
53	.code16
54
55#ifndef STAGE1_5
56	/*
57	 * In stage2, do not link start.S with the rest of the source
58	 * files directly, so define the start symbols here just to
59	 * force ld quiet. These are not referred anyway.
60	 */
61	.globl	start, _start
62start:
63_start:
64#endif /* ! STAGE1_5 */
65
66ENTRY(main)
67	/*
68	 *  Guarantee that "main" is loaded at 0x0:0x8200 in stage2 and
69	 *  at 0x0:0x2200 in stage1.5.
70	 */
71	ljmp $0, $ABS(codestart)
72
73	/*
74	 *  Compatibility version number
75	 *
76	 *  These MUST be at byte offset 6 and 7 of the executable
77	 *  DO NOT MOVE !!!
78	 */
79	. = EXT_C(main) + 0x6
80	.byte	COMPAT_VERSION_MAJOR, COMPAT_VERSION_MINOR
81
82	/*
83	 *  This is a special data area 8 bytes from the beginning.
84	 */
85
86	. = EXT_C(main) + 0x8
87
88VARIABLE(install_partition)
89	.long	0xFFFFFF
90/* This variable is here only because of a historical reason.  */
91VARIABLE(saved_entryno)
92	.long	0
93VARIABLE(stage2_id)
94	.byte	STAGE2_ID
95VARIABLE(force_lba)
96	.byte	0
97VARIABLE(version_string)
98	.string VERSION
99VARIABLE(config_file)
100#ifndef STAGE1_5
101	.string "/boot/grub/menu.lst"
102#else   /* STAGE1_5 */
103	.long	0xffffffff
104	.string "/boot/grub/stage2"
105#endif  /* STAGE1_5 */
106
107	/*
108	 *  Leave some breathing room for the config file name.
109	 */
110
111	. = EXT_C(main) + 0x70
112
113/* the real mode code continues... */
114codestart:
115	cli		/* we're not safe here! */
116
117	/* set up %ds, %ss, and %es */
118	xorw	%ax, %ax
119	movw	%ax, %ds
120	movw	%ax, %ss
121	movw	%ax, %es
122
123#ifndef SUPPORT_DISKLESS
124	/*
125	 * Save the sector number of the second sector (i.e. this sector)
126	 * in INSTALL_SECOND_SECTOR. See also "stage2/start.S".
127	 */
128	ADDR32	movl	%ebp, EXT_C(install_second_sector)
129#endif
130
131	/* set up the real mode/BIOS stack */
132	movl	$STACKOFF, %ebp
133	movl	%ebp, %esp
134
135	sti		/* we're safe again */
136
137#ifndef SUPPORT_DISKLESS
138	/* save boot drive reference */
139	ADDR32	movb	%dl, EXT_C(boot_drive)
140
141	/* reset disk system (%ah = 0) */
142	int	$0x13
143#endif
144
145	/* transition to protected mode */
146	DATA32	call EXT_C(real_to_prot)
147
148	/* The ".code32" directive takes GAS out of 16-bit mode. */
149	.code32
150
151	/* clean out the bss */
152
153	/* set %edi to the bss starting address */
154#if defined(HAVE_USCORE_USCORE_BSS_START_SYMBOL)
155	movl	$__bss_start, %edi
156#elif defined(HAVE_USCORE_EDATA_SYMBOL)
157	movl	$_edata, %edi
158#elif defined(HAVE_EDATA_SYMBOL)
159	movl	$edata, %edi
160#endif
161
162	/* set %ecx to the bss end */
163#if defined(HAVE_END_SYMBOL)
164	movl	$end, %ecx
165#elif defined(HAVE_USCORE_END_SYMBOL)
166	movl	$_end, %ecx
167#endif
168
169	/* compute the bss length */
170	subl	%edi, %ecx
171
172	/* zero %al */
173	xorb	%al, %al
174
175	/* set the direction */
176	cld
177
178	/* clean out */
179	rep
180	stosb
181
182	/*
183	 *  Call the start of main body of C code, which does some
184	 *  of it's own initialization before transferring to "cmain".
185	 */
186	call EXT_C(init_bios_info)
187
188
189/*
190 *  This call is special...  it never returns...  in fact it should simply
191 *  hang at this point!
192 */
193
194ENTRY(stop)
195	call	EXT_C(prot_to_real)
196
197	/*
198	 * This next part is sort of evil.  It takes advantage of the
199	 * byte ordering on the x86 to work in either 16-bit or 32-bit
200	 * mode, so think about it before changing it.
201	 */
202
203ENTRY(hard_stop)
204	hlt
205	jmp EXT_C(hard_stop)
206
207#ifndef STAGE1_5
208/*
209 * stop_floppy()
210 *
211 * Stops the floppy drive from spinning, so that other software is
212 * jumped to with a known state.
213 */
214ENTRY(stop_floppy)
215	pusha
216	call	EXT_C(prot_to_real)
217	.code16
218	xorb	%dl, %dl
219	int	$0x13
220	DATA32  call EXT_C(real_to_prot)
221	.code32
222	popa
223	ret
224
225/*
226 * grub_reboot()
227 *
228 * Reboot the system. At the moment, rely on BIOS.
229 */
230ENTRY(grub_reboot)
231	call	EXT_C(prot_to_real)
232	.code16
233	/* cold boot */
234	movw	$0x0472, %di
235	movw	%ax, (%di)
236	ljmp	$0xFFFF, $0x0000
237	.code32
238
239/*
240 * grub_halt(int no_apm)
241 *
242 * Halt the system, using APM if possible. If NO_APM is true, don't use
243 * APM even if it is available.
244 */
245ENTRY(grub_halt)
246	/* get the argument */
247	movl	4(%esp), %eax
248
249	/* see if zero */
250	testl	%eax, %eax
251	jnz	EXT_C(stop)
252
253	call	EXT_C(prot_to_real)
254	.code16
255
256	/* detect APM */
257	movw	$0x5300, %ax
258	xorw	%bx, %bx
259	int	$0x15
260	jc	EXT_C(hard_stop)
261	/* don't check %bx for buggy BIOSes... */
262
263	/* disconnect APM first */
264	movw	$0x5304, %ax
265	xorw	%bx, %bx
266	int	$0x15
267
268	/* connect APM */
269	movw	$0x5301, %ax
270	xorw	%bx, %bx
271	int	$0x15
272	jc	EXT_C(hard_stop)
273
274	/* set APM protocol level - 1.1 or bust. (this covers APM 1.2 also) */
275	movw	$0x530E, %ax
276	xorw	%bx, %bx
277	movw	$0x0101, %cx
278	int	$0x15
279	jc	EXT_C(hard_stop)
280
281	/* set the power state to off */
282	movw	$0x5307, %ax
283	movw	$1, %bx
284	movw	$3, %cx
285	int	$0x15
286
287	/* shouldn't reach here */
288	jmp	EXT_C(hard_stop)
289	.code32
290
291/*
292 * track_int13(int drive)
293 *
294 * Track the int13 handler to probe I/O address space.
295 */
296ENTRY(track_int13)
297	pushl	%ebp
298	movl	%esp, %ebp
299
300	pushl	%ebx
301	pushl	%edi
302
303	/* copy the original int13 handler segment:offset */
304	movl	$0x4c, %edi
305	movl	(%edi), %eax
306	movl	%eax, track_int13_addr
307
308	/* replace the int1 handler */
309	movl	$0x4, %edi
310	pushl	(%edi)
311	movl	$ABS(int1_handler), %eax
312	movl	%eax, (%edi)
313
314	/* read the MBR to call int13 successfully */
315	movb	8(%ebp), %dl
316
317	call	EXT_C(prot_to_real)
318	.code16
319
320	movw	$SCRATCHSEG, %ax
321	movw	%ax, %es
322	xorw	%bx, %bx
323	movw	$1, %cx
324	xorb	%dh, %dh
325
326	/* save FLAGS on the stack to emulate int13 */
327	pushfw
328
329	/* set the TF flag */
330	/* FIXME: this can be simplified not to use AX */
331	pushfw
332	popw	%ax
333	orw	$0x100, %ax
334	pushw	%ax
335	popfw
336
337	movw	$0x0201, %ax
338
339	.byte	0x9a		/* lcall */
340track_int13_addr:
341	.word	0		/* offset */
342	.word	0		/* segment */
343
344	/* TF is cleared here automatically */
345
346	DATA32	call	EXT_C(real_to_prot)
347	.code32
348
349	/* restore the int1 handler */
350	movl	$0x4, %edi
351	popl	(%edi)
352
353	popl	%edi
354	popl	%ebx
355	popl	%ebp
356
357	ret
358
359
360/*
361 * Check if the next instruction is I/O, and if this is true, add the
362 * port into the io map.
363 *
364 * Note: Probably this will make the execution of int13 very slow.
365 *
366 * Note2: In this implementation, all we can know is I/O-mapped I/O. It
367 * is impossible to detect memory-mapped I/O.
368 */
369int1_handler:
370	.code16
371
372	pushw	%bp
373	movw	%sp, %bp
374	pushw	%ds
375	pushw	%ax
376	pushw	%si
377	pushw	%dx
378
379	/* IP */
380	movw	2(%bp), %si
381	/* CS */
382	movw	4(%bp), %ax
383	movw	%ax, %ds
384
385	/* examine the next instruction */
3861:	lodsb	(%si), %al
387	/* skip this code if it is a prefix */
388	cmpb	$0x2E, %al
389	je	1b
390	cmpb	$0x36, %al
391	je	1b
392	cmpb	$0x3E, %al
393	je	1b
394	cmpb	$0x26, %al
395	je	1b
396	cmpb	$0x64, %al
397	jl	2f
398	cmpb	$0x67, %al
399	jle	1b
4002:	cmpb	$0xF0, %al
401	jl	3f
402	cmpb	$0xF3, %al
403	jle	1b
404
4053:	/* check if this code is out* or in* */
406
407	/* ins? or outs? */
408	cmpb	$0x6C, %al
409	jl	4f
410	cmpb	$0x6F, %al
411	jle	5f
412
4134:	/* in? or out? (register operand version) */
414	cmpb	$0xEC, %al
415	jl	6f
416	cmpb	$0xEF, %al
417	jle	5f
418
4196:	/* in? or out? (immediate operand version) */
420	cmpb	$0xE4, %al
421	jl	8f
422	cmpb	$0xE7, %al
423	jg	8f
424
4257:	/* immediate has a port */
426	lodsb	(%si), %al
427	movzbw	%al, %dx
428
4295:	/* %dx has a port */
430
431	/* set %ds to zero */
432	xorw	%ax, %ax
433	movw	%ax, %ds
434
435	/* set %si to the io map */
436	movw	$ABS(EXT_C(io_map)), %si
437
438
4399:	/* check if the io map already has the port */
440	lodsw	(%si), %ax
441	/* check if this is the end */
442	testw	%ax, %ax
443	jz	1f
444	/* check if this matches the port */
445	cmpw	%ax, %dx
446	jne	9b
447	/* if so, leave from this handler */
448	jmp	8f
449
4501:	/* check for the buffer overrun */
451	cmpw	$(ABS(EXT_C(io_map)) + (IO_MAP_SIZE + 1) * 2), %si
452	je	8f
453	/* add the port into the io map */
454	movw	%dx, -2(%si)
455
4568:	/* restore registers */
457	popw	%dx
458	popw	%si
459	popw	%ax
460	popw	%ds
461	popw	%bp
462
463	iret
464
465	.code32
466
467ENTRY(io_map)
468	.space	(IO_MAP_SIZE + 1) * 2
469
470
471/*
472 * set_int15_handler(void)
473 *
474 * Set up int15_handler.
475 */
476ENTRY(set_int15_handler)
477	pushl	%edi
478
479	/* save the original int15 handler */
480	movl	$0x54, %edi
481	movw	(%edi), %ax
482	movw	%ax, ABS(int15_offset)
483	movw	2(%edi), %ax
484	movw	%ax, ABS(int15_segment)
485
486	/* save the new int15 handler */
487	movw	$ABS(int15_handler), %ax
488	movw	%ax, (%edi)
489	xorw	%ax, %ax
490	movw	%ax, 2(%edi)
491
492	popl	%edi
493	ret
494
495
496/*
497 * unset_int15_handler(void)
498 *
499 * Restore the original int15 handler
500 */
501ENTRY(unset_int15_handler)
502	pushl	%edi
503
504	/* check if int15_handler is set */
505	movl	$0x54, %edi
506	movw	$ABS(int15_handler), %ax
507	cmpw	%ax, (%edi)
508	jne	1f
509	xorw	%ax, %ax
510	cmpw	%ax, 2(%edi)
511	jne	1f
512
513	/* restore the original */
514	movw	ABS(int15_offset), %ax
515	movw	%ax, (%edi)
516	movw	ABS(int15_segment), %ax
517	movw	%ax, 2(%edi)
518
5191:
520	popl	%edi
521	ret
522
523
524/*
525 * Translate a key code to another.
526 *
527 * Note: This implementation cannot handle more than one length
528 * scancodes (such as Right Ctrl).
529 */
530	.code16
531int15_handler:
532	/* if non-carrier, ignore it */
533	jnc	1f
534	/* check if AH=4F */
535	cmpb	$0x4F, %ah
536	jne	1f
537
538	/* E0 and E1 are special */
539	cmpb	$0xE1, %al
540	je	4f
541	cmpb	$0xE0, %al
542	/* this flag is actually the machine code (je or jmp) */
543int15_skip_flag:
544	je	4f
545
546	pushw	%bp
547	movw	%sp, %bp
548
549	pushw	%bx
550	pushw	%dx
551	pushw	%ds
552	pushw	%si
553
554	/* save bits 0-6 of %al in %dl */
555	movw	%ax, %dx
556	andb	$0x7f, %dl
557	/* save the highest bit in %bl */
558	movb	%al, %bl
559	xorb	%dl, %bl
560	/* set %ds to 0 */
561	xorw	%ax, %ax
562	movw	%ax, %ds
563	/* set %si to the key map */
564	movw	$ABS(EXT_C(bios_key_map)), %si
565
566	/* find the key code from the key map */
5672:
568	lodsw
569	/* check if this is the end */
570	testw	%ax, %ax
571	jz	3f
572	/* check if this matches the key code */
573	cmpb	%al, %dl
574	jne	2b
575	/* if so, perform the mapping */
576	movb	%ah, %dl
5773:
578	/* restore %ax */
579	movw	%dx, %ax
580	orb	%bl, %al
581	/* make sure that CF is set */
582	orw	$1, 6(%bp)
583	/* restore other registers */
584	popw	%si
585	popw	%ds
586	popw	%dx
587	popw	%bx
588	popw	%bp
589	iret
590
5914:
592	/* tricky: jmp (0x74) <-> je (0xeb) */
593	xorb	$(0x74 ^ 0xeb), ABS(int15_skip_flag)
5941:
595	/* just cascade to the original */
596	/* ljmp */
597	.byte	0xea
598int15_offset:	.word	0
599int15_segment:	.word	0
600
601	.code32
602
603	.align	4
604ENTRY(bios_key_map)
605	.space	(KEY_MAP_SIZE + 1) * 2
606
607
608/*
609 * set_int13_handler(map)
610 *
611 * Copy MAP to the drive map and set up int13_handler.
612 */
613ENTRY(set_int13_handler)
614	pushl	%ebp
615	movl	%esp, %ebp
616
617	pushl	%edi
618	pushl	%esi
619
620	/* copy MAP to the drive map */
621	movl	$(DRIVE_MAP_SIZE * 2), %ecx
622	movl	$ABS(drive_map), %edi
623	movl	8(%ebp), %esi
624	cld
625	rep
626	movsb
627
628	/* save the original int13 handler */
629	movl	$0x4c, %edi
630	movw	(%edi), %ax
631	movw	%ax, ABS(int13_offset)
632	movw	2(%edi), %ax
633	movw	%ax, ABS(int13_segment)
634
635	/* decrease the lower memory size and set it to the BIOS memory */
636	movl	$0x413, %edi
637	decw	(%edi)
638	xorl	%eax, %eax
639	movw	(%edi), %ax
640
641	/* compute the segment */
642	shll	$6, %eax
643
644	/* save the new int13 handler */
645	movl	$0x4c, %edi
646	movw	%ax, 2(%edi)
647	xorw	%cx, %cx
648	movw	%cx, (%edi)
649
650	/* copy int13_handler to the reserved area */
651	shll	$4, %eax
652	movl	%eax, %edi
653	movl	$ABS(int13_handler), %esi
654	movl	$(int13_handler_end - int13_handler), %ecx
655	rep
656	movsb
657
658	popl	%esi
659	popl	%edi
660	popl	%ebp
661	ret
662
663
664/*
665 * Map a drive to another drive.
666 */
667
668	.code16
669
670int13_handler:
671	pushw	%ax
672	pushw	%bp
673	movw	%sp, %bp
674
675	pushw	%si
676
677	/* set %si to the drive map */
678	movw	$(drive_map - int13_handler), %si
679	/* find the drive number from the drive map */
680	cld
6811:
682	lodsw	%cs:(%si), %ax
683	/* check if this is the end */
684	testw	%ax, %ax
685	jz	2f
686	/* check if this matches the drive number */
687	cmpb	%al, %dl
688	jne	1b
689	/* if so, perform the mapping */
690	movb	%ah, %dl
6912:
692	/* restore %si */
693	popw	%si
694	/* save %ax in the stack */
695	pushw	%ax
696	/* simulate the interrupt call */
697	pushw	8(%bp)
698	/* set %ax and %bp to the original values */
699	movw	2(%bp), %ax
700	movw	(%bp), %bp
701	/* lcall */
702	.byte	0x9a
703int13_offset:	.word	0
704int13_segment:	.word	0
705	/* save flags */
706	pushf
707	/* restore %bp */
708	movw	%sp, %bp
709	/* save %ax */
710	pushw	%ax
711	/* set the flags in the stack to the value returned by int13 */
712	movw	(%bp), %ax
713	movw	%ax, 0xc(%bp)
714	/* check if should map the drive number */
715	movw	6(%bp), %ax
716	cmpw	$0x8, %ax
717	jne	3f
718	cmpw	$0x15, %ax
719	jne	3f
720	/* check if the mapping was performed */
721	movw	2(%bp), %ax
722	testw	%ax, %ax
723	jz	3f
724	/* perform the mapping */
725	movb	%al, %dl
7263:
727	popw	%ax
728	movw	4(%bp), %bp
729	addw	$8, %sp
730	iret
731
732	.align	4
733drive_map:	.space	(DRIVE_MAP_SIZE + 1) * 2
734int13_handler_end:
735
736	.code32
737
738
739/*
740 * chain_stage1(segment, offset, part_table_addr)
741 *
742 *  This starts another stage1 loader, at segment:offset.
743 */
744
745ENTRY(chain_stage1)
746	/* no need to save anything, just use %esp */
747
748	/* store %ESI, presuming %ES is 0 */
749	movl	0xc(%esp), %esi
750
751	/* store new offset */
752	movl	0x8(%esp), %eax
753	movl	%eax, offset
754
755	/* store new segment */
756	movw	0x4(%esp), %ax
757	movw	%ax, segment
758
759	/* set up to pass boot drive */
760	movb	EXT_C(boot_drive), %dl
761
762	call	EXT_C(prot_to_real)
763	.code16
764
765#ifdef ABSOLUTE_WITHOUT_ASTERISK
766	DATA32	ADDR32	ljmp	(offset)
767#else
768	DATA32	ADDR32	ljmp	*(offset)
769#endif
770	.code32
771#endif /* STAGE1_5 */
772
773
774#ifdef STAGE1_5
775/*
776 * chain_stage2(segment, offset, second_sector)
777 *
778 *  This starts another stage2 loader, at segment:offset.  It presumes
779 *  that the other one starts with this same "asm.S" file, and passes
780 *  parameters by writing the embedded install variables.
781 */
782
783ENTRY(chain_stage2)
784	/* no need to save anything, just use %esp */
785
786	/* store new offset */
787	movl	0x8(%esp), %eax
788	movl	%eax, offset
789	movl	%eax, %ebx
790
791	/* store new segment */
792	movw	0x4(%esp), %ax
793	movw	%ax, segment
794	shll	$4, %eax
795
796	/* generate linear address */
797	addl	%eax, %ebx
798
799	/* set up to pass the partition where stage2 is located in */
800	movl	EXT_C(current_partition), %eax
801	movl	%eax, (EXT_C(install_partition)-EXT_C(main))(%ebx)
802
803	/* set up to pass the drive where stage2 is located in */
804	movb	EXT_C(current_drive), %dl
805
806	/* set up to pass the second sector of stage2 */
807	movl	0xc(%esp), %ecx
808
809	call	EXT_C(prot_to_real)
810	.code16
811
812	movl	%ecx, %ebp
813
814#ifdef ABSOLUTE_WITHOUT_ASTERISK
815	DATA32	ADDR32	ljmp	(offset)
816#else
817	DATA32	ADDR32	ljmp	*(offset)
818#endif
819
820	.code32
821#endif /* STAGE1_5 */
822
823/*
824 *  These next two routines, "real_to_prot" and "prot_to_real" are structured
825 *  in a very specific way.  Be very careful when changing them.
826 *
827 *  NOTE:  Use of either one messes up %eax and %ebp.
828 */
829
830ENTRY(real_to_prot)
831	.code16
832	cli
833
834	/* load the GDT register */
835	DATA32	ADDR32	lgdt	gdtdesc
836
837	/* turn on protected mode */
838	movl	%cr0, %eax
839	orl	$CR0_PE_ON, %eax
840	movl	%eax, %cr0
841
842	/* jump to relocation, flush prefetch queue, and reload %cs */
843	DATA32	ljmp	$PROT_MODE_CSEG, $protcseg
844
845	/*
846	 *  The ".code32" directive only works in GAS, the GNU assembler!
847	 *  This gets out of "16-bit" mode.
848	 */
849	.code32
850
851protcseg:
852	/* reload other segment registers */
853	movw	$PROT_MODE_DSEG, %ax
854	movw	%ax, %ds
855	movw	%ax, %es
856	movw	%ax, %fs
857	movw	%ax, %gs
858	movw	%ax, %ss
859
860	/* put the return address in a known safe location */
861	movl	(%esp), %eax
862	movl	%eax, STACKOFF
863
864	/* get protected mode stack */
865	movl	protstack, %eax
866	movl	%eax, %esp
867	movl	%eax, %ebp
868
869	/* get return address onto the right stack */
870	movl	STACKOFF, %eax
871	movl	%eax, (%esp)
872
873	/* zero %eax */
874	xorl	%eax, %eax
875
876	/* return on the old (or initialized) stack! */
877	ret
878
879
880ENTRY(prot_to_real)
881	/* just in case, set GDT */
882	lgdt	gdtdesc
883
884	/* save the protected mode stack */
885	movl	%esp, %eax
886	movl	%eax, protstack
887
888	/* get the return address */
889	movl	(%esp), %eax
890	movl	%eax, STACKOFF
891
892	/* set up new stack */
893	movl	$STACKOFF, %eax
894	movl	%eax, %esp
895	movl	%eax, %ebp
896
897	/* set up segment limits */
898	movw	$PSEUDO_RM_DSEG, %ax
899	movw	%ax, %ds
900	movw	%ax, %es
901	movw	%ax, %fs
902	movw	%ax, %gs
903	movw	%ax, %ss
904
905	/* this might be an extra step */
906	ljmp	$PSEUDO_RM_CSEG, $tmpcseg	/* jump to a 16 bit segment */
907
908tmpcseg:
909	.code16
910
911	/* clear the PE bit of CR0 */
912	movl	%cr0, %eax
913	andl 	$CR0_PE_OFF, %eax
914	movl	%eax, %cr0
915
916	/* flush prefetch queue, reload %cs */
917	DATA32	ljmp	$0, $realcseg
918
919realcseg:
920	/* we are in real mode now
921	 * set up the real mode segment registers : DS, SS, ES
922	 */
923	/* zero %eax */
924	xorl	%eax, %eax
925
926	movw	%ax, %ds
927	movw	%ax, %es
928	movw	%ax, %fs
929	movw	%ax, %gs
930	movw	%ax, %ss
931
932	/* restore interrupts */
933	sti
934
935	/* return on new stack! */
936	DATA32	ret
937
938	.code32
939
940
941/*
942 *   int biosdisk_int13_extensions (int ax, int drive, void *dap)
943 *
944 *   Call IBM/MS INT13 Extensions (int 13 %ax=AX) for DRIVE. DAP
945 *   is passed for disk address packet. If an error occurs, return
946 *   non-zero, otherwise zero.
947 */
948
949ENTRY(biosdisk_int13_extensions)
950	pushl	%ebp
951	movl	%esp, %ebp
952
953	pushl	%esi
954	pushl	%ebx
955
956	/* compute the address of disk_address_packet */
957	movl	0x10(%ebp), %eax
958	movw	%ax, %si
959	xorw	%ax, %ax
960	shrl	$4, %eax
961	movw	%ax, %cx	/* save the segment to cx */
962
963	/* drive */
964	movb	0xc(%ebp), %dl
965	/* ax */
966	movw	0x8(%ebp), %bx
967	/* enter real mode */
968	call	EXT_C(prot_to_real)
969
970	.code16
971	movw	%bx, %ax
972	movw	%cx, %ds
973	int	$0x13		/* do the operation */
974	movb	%ah, %dl	/* save return value */
975	/* clear the data segment */
976	xorw	%ax, %ax
977	movw	%ax, %ds
978	/* back to protected mode */
979	DATA32	call	EXT_C(real_to_prot)
980	.code32
981
982	movb	%dl, %al	/* return value in %eax */
983
984	popl	%ebx
985	popl	%esi
986	popl	%ebp
987
988	ret
989
990/*
991 *   int biosdisk_standard (int ah, int drive, int coff, int hoff, int soff,
992 *                          int nsec, int segment)
993 *
994 *   Call standard and old INT13 (int 13 %ah=AH) for DRIVE. Read/write
995 *   NSEC sectors from COFF/HOFF/SOFF into SEGMENT. If an error occurs,
996 *   return non-zero, otherwise zero.
997 */
998
999ENTRY(biosdisk_standard)
1000	pushl	%ebp
1001	movl	%esp, %ebp
1002
1003	pushl	%ebx
1004	pushl	%edi
1005	pushl	%esi
1006
1007	/* set up CHS information */
1008	movl	0x10(%ebp), %eax
1009	movb	%al, %ch
1010	movb	0x18(%ebp), %al
1011	shlb	$2, %al
1012	shrw	$2, %ax
1013	movb	%al, %cl
1014	movb	0x14(%ebp), %dh
1015	/* drive */
1016	movb	0xc(%ebp), %dl
1017	/* segment */
1018	movw	0x20(%ebp), %bx
1019	/* save nsec and ah to %di */
1020	movb	0x8(%ebp), %ah
1021	movb	0x1c(%ebp), %al
1022	movw	%ax, %di
1023	/* enter real mode */
1024	call	EXT_C(prot_to_real)
1025
1026	.code16
1027	movw	%bx, %es
1028	xorw	%bx, %bx
1029	movw	$3, %si		/* attempt at least three times */
1030
10311:
1032	movw	%di, %ax
1033	int	$0x13		/* do the operation */
1034	jnc	2f		/* check if successful */
1035
1036	movb	%ah, %bl	/* save return value */
1037	/* if fail, reset the disk system */
1038	xorw	%ax, %ax
1039	int	$0x13
1040
1041	decw	%si
1042	cmpw	$0, %si
1043	je	2f
1044	xorb	%bl, %bl
1045	jmp	1b		/* retry */
10462:
1047	/* back to protected mode */
1048	DATA32	call	EXT_C(real_to_prot)
1049	.code32
1050
1051	movb	%bl, %al	/* return value in %eax */
1052
1053	popl	%esi
1054	popl	%edi
1055	popl	%ebx
1056	popl	%ebp
1057
1058	ret
1059
1060
1061/*
1062 *   int check_int13_extensions (int drive)
1063 *
1064 *   Check if LBA is supported for DRIVE. If it is supported, then return
1065 *   the major version of extensions, otherwise zero.
1066 */
1067
1068ENTRY(check_int13_extensions)
1069	pushl	%ebp
1070	movl	%esp, %ebp
1071
1072	pushl	%ebx
1073
1074	/* drive */
1075	movb	0x8(%ebp), %dl
1076	/* enter real mode */
1077	call	EXT_C(prot_to_real)
1078
1079	.code16
1080	movb	$0x41, %ah
1081	movw	$0x55aa, %bx
1082	int	$0x13		/* do the operation */
1083
1084	/* check the result */
1085	jc	1f
1086	cmpw	$0xaa55, %bx
1087	jne	1f
1088
1089	movb	%ah, %bl	/* save the major version into %bl */
1090
1091	/* check if AH=0x42 is supported if FORCE_LBA is zero */
1092	movb	EXT_C(force_lba), %al
1093	testb	%al, %al
1094	jnz	2f
1095	andw	$1, %cx
1096	jnz	2f
1097
10981:
1099	xorb	%bl, %bl
11002:
1101	/* back to protected mode */
1102	DATA32	call	EXT_C(real_to_prot)
1103	.code32
1104
1105	movb	%bl, %al	/* return value in %eax */
1106
1107	popl	%ebx
1108	popl	%ebp
1109
1110	ret
1111
1112
1113/*
1114 *   int get_diskinfo_standard (int drive, unsigned long *cylinders,
1115 *                              unsigned long *heads, unsigned long *sectors)
1116 *
1117 *   Return the geometry of DRIVE in CYLINDERS, HEADS and SECTORS. If an
1118 *   error occurs, then return non-zero, otherwise zero.
1119 */
1120
1121ENTRY(get_diskinfo_standard)
1122	pushl	%ebp
1123	movl	%esp, %ebp
1124
1125	pushl	%ebx
1126	pushl	%edi
1127
1128	/* drive */
1129	movb	0x8(%ebp), %dl
1130	/* enter real mode */
1131	call	EXT_C(prot_to_real)
1132
1133	.code16
1134	movb	$0x8, %ah
1135	int	$0x13		/* do the operation */
1136	/* check if successful */
1137	testb	%ah, %ah
1138	jnz	1f
1139	/* bogus BIOSes may not return an error number */
1140	testb	$0x3f, %cl	/* 0 sectors means no disk */
1141	jnz	1f		/* if non-zero, then succeed */
1142	/* XXX 0x60 is one of the unused error numbers */
1143	movb	$0x60, %ah
11441:
1145	movb	%ah, %bl	/* save return value in %bl */
1146	/* back to protected mode */
1147	DATA32	call	EXT_C(real_to_prot)
1148	.code32
1149
1150	/* restore %ebp */
1151	leal	0x8(%esp), %ebp
1152
1153	/* heads */
1154	movb	%dh, %al
1155	incl	%eax		/* the number of heads is counted from zero */
1156	movl	0x10(%ebp), %edi
1157	movl	%eax, (%edi)
1158
1159	/* sectors */
1160	xorl	%eax, %eax
1161	movb	%cl, %al
1162	andb	$0x3f, %al
1163	movl	0x14(%ebp), %edi
1164	movl	%eax, (%edi)
1165
1166	/* cylinders */
1167	shrb	$6, %cl
1168	movb	%cl, %ah
1169	movb	%ch, %al
1170	incl	%eax		/* the number of cylinders is
1171				   counted from zero */
1172	movl	0xc(%ebp), %edi
1173	movl	%eax, (%edi)
1174
1175	xorl	%eax, %eax
1176	movb	%bl, %al	/* return value in %eax */
1177
1178	popl	%edi
1179	popl	%ebx
1180	popl	%ebp
1181
1182	ret
1183
1184
1185#if 0
1186/*
1187 *   int get_diskinfo_floppy (int drive, unsigned long *cylinders,
1188 *                            unsigned long *heads, unsigned long *sectors)
1189 *
1190 *   Return the geometry of DRIVE in CYLINDERS, HEADS and SECTORS. If an
1191 *   error occurs, then return non-zero, otherwise zero.
1192 */
1193
1194ENTRY(get_diskinfo_floppy)
1195	pushl	%ebp
1196	movl	%esp, %ebp
1197
1198	pushl	%ebx
1199	pushl	%esi
1200
1201	/* drive */
1202	movb	0x8(%ebp), %dl
1203	/* enter real mode */
1204	call	EXT_C(prot_to_real)
1205
1206	.code16
1207	/* init probe value */
1208	movl	$probe_values-1, %esi
12091:
1210	xorw	%ax, %ax
1211	int	$0x13		/* reset floppy controller */
1212
1213	incw	%si
1214	movb	(%si), %cl
1215	cmpb	$0, %cl		/* probe failed if zero */
1216	je	2f
1217
1218	/* perform read */
1219	movw	$SCRATCHSEG, %ax
1220	movw	%ax, %es
1221	xorw	%bx, %bx
1222	movw	$0x0201, %ax
1223	movb	$0, %ch
1224	movb	$0, %dh
1225	int	$0x13
1226
1227	/* FIXME: Read from floppy may fail even if the geometry is correct.
1228	   So should retry at least three times.  */
1229	jc	1b		/* next value */
1230
1231	/* succeed */
1232	jmp	2f
1233
1234probe_values:
1235	.byte	36, 18, 15, 9, 0
1236
12372:
1238	/* back to protected mode */
1239	DATA32	call	EXT_C(real_to_prot)
1240	.code32
1241
1242	/* restore %ebp */
1243	leal	0x8(%esp), %ebp
1244
1245	/* cylinders */
1246	movl	0xc(%ebp), %eax
1247	movl	$80, %ebx
1248	movl	%ebx, (%eax)
1249	/* heads */
1250	movl	0x10(%ebp), %eax
1251	movl	$2, %ebx
1252	movl	%ebx, (%eax)
1253	/* sectors */
1254	movl	0x14(%ebp), %eax
1255	movzbl	%cl, %ebx
1256	movl	%ebx, (%eax)
1257
1258	/* return value in %eax */
1259	xorl	%eax, %eax
1260	cmpb	$0, %cl
1261	jne	3f
1262	incl	%eax		/* %eax = 1 (non-zero) */
12633:
1264	popl	%esi
1265	popl	%ebx
1266	popl	%ebp
1267
1268	ret
1269#endif
1270
1271
1272/* Source files are splitted, as they have different copyrights.  */
1273#ifndef STAGE1_5
1274# include "setjmp.S"
1275# include "apm.S"
1276#endif /* ! STAGE1_5 */
1277
1278
1279
1280#ifndef STAGE1_5
1281/* get_code_end() :  return the address of the end of the code
1282 * This is here so that it can be replaced by asmstub.c.
1283 */
1284ENTRY(get_code_end)
1285	/* will be the end of the bss */
1286# if defined(HAVE_END_SYMBOL)
1287	movl	$end, %eax
1288# elif defined(HAVE_USCORE_END_SYMBOL)
1289	movl	$_end, %eax
1290# endif
1291	shrl	$2, %eax		/* Round up to the next word. */
1292	incl	%eax
1293	shll	$2, %eax
1294	ret
1295#endif /* ! STAGE1_5 */
1296
1297/*
1298 *
1299 * get_memsize(i) :  return the memory size in KB. i == 0 for conventional
1300 *		memory, i == 1 for extended memory
1301 *	BIOS call "INT 12H" to get conventional memory size
1302 *	BIOS call "INT 15H, AH=88H" to get extended memory size
1303 *		Both have the return value in AX.
1304 *
1305 */
1306
1307ENTRY(get_memsize)
1308	push	%ebp
1309	push	%ebx
1310
1311	mov	0xc(%esp), %ebx
1312
1313	call	EXT_C(prot_to_real)	/* enter real mode */
1314	.code16
1315
1316	cmpb	$0x1, %bl
1317	DATA32	je	xext
1318
1319	int	$0x12
1320	DATA32	jmp	xdone
1321
1322xext:
1323	movb	$0x88, %ah
1324	int	$0x15
1325
1326xdone:
1327	movw	%ax, %bx
1328
1329	DATA32	call	EXT_C(real_to_prot)
1330	.code32
1331
1332	movw	%bx, %ax
1333	pop	%ebx
1334	pop	%ebp
1335	ret
1336
1337
1338#ifndef STAGE1_5
1339
1340/*
1341 *
1342 * get_eisamemsize() :  return packed EISA memory map, lower 16 bits is
1343 *		memory between 1M and 16M in 1K parts, upper 16 bits is
1344 *		memory above 16M in 64K parts.  If error, return -1.
1345 *	BIOS call "INT 15H, AH=E801H" to get EISA memory map,
1346 *		AX = memory between 1M and 16M in 1K parts.
1347 *		BX = memory above 16M in 64K parts.
1348 *
1349 */
1350
1351ENTRY(get_eisamemsize)
1352	push	%ebp
1353	push	%ebx
1354
1355	call	EXT_C(prot_to_real)	/* enter real mode */
1356	.code16
1357
1358	movw	$0xe801, %ax
1359	int	$0x15
1360
1361	shll	$16, %ebx
1362	movw	%ax, %bx
1363
1364	DATA32	call	EXT_C(real_to_prot)
1365	.code32
1366
1367	movl	$0xFFFFFFFF, %eax
1368	cmpb	$0x86, %bh
1369	je	xnoteisa
1370
1371	movl	%ebx, %eax
1372
1373xnoteisa:
1374	pop	%ebx
1375	pop	%ebp
1376	ret
1377
1378/*
1379 *
1380 * get_mmap_entry(addr, cont) :  address and old continuation value (zero to
1381 *		start), for the Query System Address Map BIOS call.
1382 *
1383 *  Sets the first 4-byte int value of "addr" to the size returned by
1384 *  the call.  If the call fails, sets it to zero.
1385 *
1386 *	Returns:  new (non-zero) continuation value, 0 if done.
1387 *
1388 * NOTE: Currently hard-coded for a maximum buffer length of 1024.
1389 */
1390
1391ENTRY(get_mmap_entry)
1392	push	%ebp
1393	push	%ebx
1394	push	%edi
1395	push	%esi
1396
1397	/* place address (+4) in ES:DI */
1398	movl	0x14(%esp), %eax
1399	addl	$4, %eax
1400	movl	%eax, %edi
1401	andl	$0xf, %edi
1402	shrl	$4, %eax
1403	movl	%eax, %esi
1404
1405	/* set continuation value */
1406	movl	0x18(%esp), %ebx
1407
1408	/* set default maximum buffer size */
1409	movl	$0x14, %ecx
1410
1411	/* set EDX to 'SMAP' */
1412	movl	$0x534d4150, %edx
1413
1414	call	EXT_C(prot_to_real)	/* enter real mode */
1415	.code16
1416
1417	movw	%si, %es
1418	movl	$0xe820, %eax
1419	int	$0x15
1420
1421	DATA32	jc	xnosmap
1422
1423	cmpl	$0x534d4150, %eax
1424	DATA32	jne	xnosmap
1425
1426	cmpl	$0x14, %ecx
1427	DATA32	jl	xnosmap
1428
1429	cmpl	$0x400, %ecx
1430	DATA32	jg	xnosmap
1431
1432	DATA32	jmp	xsmap
1433
1434xnosmap:
1435	movl	$0, %ecx
1436
1437xsmap:
1438	DATA32	call	EXT_C(real_to_prot)
1439	.code32
1440
1441	/* write length of buffer (zero if error) into "addr" */
1442	movl	0x14(%esp), %eax
1443	movl	%ecx, (%eax)
1444
1445	/* set return value to continuation */
1446	movl	%ebx, %eax
1447
1448	pop	%esi
1449	pop	%edi
1450	pop	%ebx
1451	pop	%ebp
1452	ret
1453
1454/*
1455 * get_rom_config_table()
1456 *
1457 * Get the linear address of a ROM configuration table. Return zero,
1458 * if fails.
1459 */
1460
1461ENTRY(get_rom_config_table)
1462	pushl	%ebp
1463	pushl	%ebx
1464
1465	/* zero %ebx for simplicity */
1466	xorl	%ebx, %ebx
1467
1468	call	EXT_C(prot_to_real)
1469	.code16
1470
1471	movw	$0xc0, %ax
1472	int	$0x15
1473
1474	jc	no_rom_table
1475	testb	%ah, %ah
1476	jnz	no_rom_table
1477
1478	movw	%es, %dx
1479	jmp	found_rom_table
1480
1481no_rom_table:
1482	xorw	%dx, %dx
1483	xorw	%bx, %bx
1484
1485found_rom_table:
1486	DATA32	call	EXT_C(real_to_prot)
1487	.code32
1488
1489	/* compute the linear address */
1490	movw	%dx, %ax
1491	shll	$4, %eax
1492	addl	%ebx, %eax
1493
1494	popl	%ebx
1495	popl	%ebp
1496	ret
1497
1498
1499/*
1500 * int get_vbe_controller_info (struct vbe_controller *controller_ptr)
1501 *
1502 * Get VBE controller information.
1503 */
1504
1505ENTRY(get_vbe_controller_info)
1506	pushl	%ebp
1507	movl	%esp, %ebp
1508
1509	pushl	%edi
1510	pushl	%ebx
1511
1512	/* Convert the linear address to segment:offset */
1513	movl	8(%ebp), %eax
1514	movl	%eax, %edi
1515	andl	$0x0000000f, %edi
1516	shrl	$4, %eax
1517	movl	%eax, %ebx
1518
1519	call	EXT_C(prot_to_real)
1520	.code16
1521
1522	movw	%bx, %es
1523	movw	$0x4F00, %ax
1524	int	$0x10
1525
1526	movw	%ax, %bx
1527	DATA32	call	EXT_C(real_to_prot)
1528	.code32
1529
1530	movzwl	%bx, %eax
1531
1532	popl	%ebx
1533	popl	%edi
1534	popl	%ebp
1535	ret
1536
1537
1538/*
1539 * int get_vbe_mode_info (int mode_number, struct vbe_mode *mode_ptr)
1540 *
1541 * Get VBE mode information.
1542 */
1543
1544ENTRY(get_vbe_mode_info)
1545	pushl	%ebp
1546	movl	%esp, %ebp
1547
1548	pushl	%edi
1549	pushl	%ebx
1550
1551	/* Convert the linear address to segment:offset */
1552	movl	0xc(%ebp), %eax
1553	movl	%eax, %edi
1554	andl	$0x0000000f, %edi
1555	shrl	$4, %eax
1556	movl	%eax, %ebx
1557
1558	/* Save the mode number in %cx */
1559	movl	0x8(%ebp), %ecx
1560
1561	call	EXT_C(prot_to_real)
1562	.code16
1563
1564	movw	%bx, %es
1565	movw	$0x4F01, %ax
1566	int	$0x10
1567
1568	movw	%ax, %bx
1569	DATA32	call	EXT_C(real_to_prot)
1570	.code32
1571
1572	movzwl	%bx, %eax
1573
1574	popl	%ebx
1575	popl	%edi
1576	popl	%ebp
1577	ret
1578
1579
1580/*
1581 * int set_vbe_mode (int mode_number)
1582 *
1583 * Set VBE mode. Don't support user-specified CRTC information.
1584 */
1585
1586ENTRY(set_vbe_mode)
1587	pushl	%ebp
1588	movl	%esp, %ebp
1589
1590	pushl	%ebx
1591
1592	/* Save the mode number in %bx */
1593	movl	0x8(%ebp), %ebx
1594	/* Clear bit D11 */
1595	andl	$0xF7FF, %ebx
1596
1597	call	EXT_C(prot_to_real)
1598	.code16
1599
1600	movw	$0x4F02, %ax
1601	int	$0x10
1602
1603	movw	%ax, %bx
1604	DATA32	call	EXT_C(real_to_prot)
1605	.code32
1606
1607	movzwl	%bx, %eax
1608
1609	popl	%ebx
1610	popl	%ebp
1611	ret
1612
1613
1614/*
1615 * gateA20(int linear)
1616 *
1617 * Gate address-line 20 for high memory.
1618 *
1619 * This routine is probably overconservative in what it does, but so what?
1620 *
1621 * It also eats any keystrokes in the keyboard buffer.  :-(
1622 */
1623
1624ENTRY(gateA20)
1625	/* first, try a BIOS call */
1626	pushl	%ebp
1627	movl	8(%esp), %edx
1628
1629	call	EXT_C(prot_to_real)
1630
1631	.code16
1632	movw	$0x2400, %ax
1633	testw	%dx, %dx
1634	jz	1f
1635	incw	%ax
16361:	stc
1637	int	$0x15
1638	jnc	2f
1639
1640	/* set non-zero if failed */
1641	movb	$1, %ah
1642
1643	/* save the status */
16442:	movb	%ah, %dl
1645
1646	DATA32	call	EXT_C(real_to_prot)
1647	.code32
1648
1649	popl	%ebp
1650	testb	%dl, %dl
1651	jnz	3f
1652	ret
1653
16543:	/* use keyboard controller */
1655	pushl	%eax
1656
1657	call    gloop1
1658
1659	movb	$KC_CMD_WOUT, %al
1660	outb	$K_CMD
1661
1662gloopint1:
1663	inb	$K_STATUS
1664	andb	$K_IBUF_FUL, %al
1665	jnz	gloopint1
1666
1667	movb	$KB_OUTPUT_MASK, %al
1668	cmpb	$0, 0x8(%esp)
1669	jz	gdoit
1670
1671	orb	$KB_A20_ENABLE, %al
1672gdoit:
1673	outb	$K_RDWR
1674
1675	call	gloop1
1676
1677	/* output a dummy command (USB keyboard hack) */
1678	movb	$0xff, %al
1679	outb	$K_CMD
1680	call	gloop1
1681
1682	popl	%eax
1683	ret
1684
1685gloop1:
1686	inb	$K_STATUS
1687	andb	$K_IBUF_FUL, %al
1688	jnz	gloop1
1689
1690gloop2:
1691	inb	$K_STATUS
1692	andb	$K_OBUF_FUL, %al
1693	jz	gloop2ret
1694	inb	$K_RDWR
1695	jmp	gloop2
1696
1697gloop2ret:
1698	ret
1699
1700
1701ENTRY(patch_code)	/* labels start with "pc_" */
1702	.code16
1703
1704	mov	%cs, %ax
1705	mov	%ax, %ds
1706	mov	%ax, %es
1707	mov	%ax, %fs
1708	mov	%ax, %gs
1709	ADDR32	movl	$0, 0
1710pc_stop:
1711	hlt
1712	DATA32	jmp	pc_stop
1713ENTRY(patch_code_end)
1714
1715	.code32
1716
1717
1718/*
1719 * linux_boot()
1720 *
1721 * Does some funky things (including on the stack!), then jumps to the
1722 * entry point of the Linux setup code.
1723 */
1724
1725VARIABLE(linux_text_len)
1726	.long	0
1727
1728VARIABLE(linux_data_tmp_addr)
1729	.long	0
1730
1731VARIABLE(linux_data_real_addr)
1732	.long	0
1733
1734ENTRY(linux_boot)
1735	/* don't worry about saving anything, we're committed at this point */
1736	cld	/* forward copying */
1737
1738	/* copy kernel */
1739	movl	EXT_C(linux_text_len), %ecx
1740	addl	$3, %ecx
1741	shrl	$2, %ecx
1742	movl	$LINUX_BZIMAGE_ADDR, %esi
1743	movl	$LINUX_ZIMAGE_ADDR, %edi
1744
1745	rep
1746	movsl
1747
1748ENTRY(big_linux_boot)
1749	movl	EXT_C(linux_data_real_addr), %ebx
1750
1751	/* copy the real mode part */
1752	movl	EXT_C(linux_data_tmp_addr), %esi
1753	movl	%ebx, %edi
1754	movl	$LINUX_SETUP_MOVE_SIZE, %ecx
1755	cld
1756	rep
1757	movsb
1758
1759	/* change %ebx to the segment address */
1760	shrl	$4, %ebx
1761	movl	%ebx, %eax
1762	addl	$0x20, %eax
1763	movl	%eax, linux_setup_seg
1764
1765	/* XXX new stack pointer in safe area for calling functions */
1766	movl	$0x4000, %esp
1767	call	EXT_C(stop_floppy)
1768
1769	/* final setup for linux boot */
1770
1771	call	EXT_C(prot_to_real)
1772	.code16
1773
1774	/* final setup for linux boot */
1775	cli
1776	movw	%bx, %ss
1777	movw	$LINUX_SETUP_STACK, %sp
1778
1779	movw	%bx, %ds
1780	movw	%bx, %es
1781	movw	%bx, %fs
1782	movw	%bx, %gs
1783
1784	/* jump to start */
1785	/* ljmp */
1786	.byte	0xea
1787	.word	0
1788linux_setup_seg:
1789	.word	0
1790	.code32
1791
1792
1793/*
1794 * multi_boot(int start, int mb_info)
1795 *
1796 *  This starts a kernel in the manner expected of the multiboot standard.
1797 */
1798
1799ENTRY(multi_boot)
1800	/* no need to save anything */
1801	call	EXT_C(stop_floppy)
1802
1803	movl	$0x2BADB002, %eax
1804	movl	0x8(%esp), %ebx
1805
1806	/* boot kernel here (absolute address call) */
1807	call	*0x4(%esp)
1808
1809	/* error */
1810	call	EXT_C(stop)
1811
1812#endif /* ! STAGE1_5 */
1813
1814/*
1815 * void console_putchar (int c)
1816 *
1817 * Put the character C on the console. Because GRUB wants to write a
1818 * character with an attribute, this implementation is a bit tricky.
1819 * If C is a control character (CR, LF, BEL, BS), use INT 10, AH = 0Eh
1820 * (TELETYPE OUTPUT). Otherwise, save the original position, put a space,
1821 * save the current position, restore the original position, write the
1822 * character and the attribute, and restore the current position.
1823 *
1824 * The reason why this is so complicated is that there is no easy way to
1825 * get the height of the screen, and the TELETYPE OUPUT BIOS call doesn't
1826 * support setting a background attribute.
1827 */
1828ENTRY(console_putchar)
1829	movl	0x4(%esp), %edx
1830	pusha
1831#ifdef STAGE1_5
1832	movb	$0x07, %bl
1833#else
1834	movl	EXT_C(console_current_color), %ebx
1835#endif
1836
1837	call	EXT_C(prot_to_real)
1838	.code16
1839	movb	%dl, %al
1840	xorb	%bh, %bh
1841
1842#ifndef STAGE1_5
1843	/* use teletype output if control character */
1844	cmpb	$0x7, %al
1845	je	1f
1846	cmpb	$0x8, %al
1847	je	1f
1848	cmpb	$0xa, %al
1849	je	1f
1850	cmpb	$0xd, %al
1851	je	1f
1852
1853	/* save the character and the attribute on the stack */
1854	pushw	%ax
1855	pushw	%bx
1856
1857	/* get the current position */
1858	movb	$0x3, %ah
1859	int	$0x10
1860
1861	/* check the column with the width */
1862	cmpb	$79, %dl
1863	jl	2f
1864
1865	/* print CR and LF, if next write will exceed the width */
1866	movw	$0x0e0d, %ax
1867	int	$0x10
1868	movb	$0x0a, %al
1869	int	$0x10
1870
1871	/* get the current position */
1872	movb	$0x3, %ah
1873	int	$0x10
1874
18752:
1876	/* restore the character and the attribute */
1877	popw	%bx
1878	popw	%ax
1879
1880	/* write the character with the attribute */
1881	movb	$0x9, %ah
1882	movw	$1, %cx
1883	int	$0x10
1884
1885	/* move the cursor forward */
1886	incb	%dl
1887	movb	$0x2, %ah
1888	int	$0x10
1889
1890	jmp	3f
1891#endif /* ! STAGE1_5 */
1892
18931:	movb	$0xe, %ah
1894	int	$0x10
1895
18963:	DATA32	call	EXT_C(real_to_prot)
1897	.code32
1898
1899	popa
1900	ret
1901
1902
1903#ifndef STAGE1_5
1904
1905/* this table is used in translate_keycode below */
1906translation_table:
1907	.word	KEY_LEFT, 2
1908	.word	KEY_RIGHT, 6
1909	.word	KEY_UP, 16
1910	.word	KEY_DOWN, 14
1911	.word	KEY_HOME, 1
1912	.word	KEY_END, 5
1913	.word	KEY_DC, 4
1914	.word	KEY_BACKSPACE, 8
1915	.word	KEY_PPAGE, 7
1916	.word	KEY_NPAGE, 3
1917	.word	0
1918
1919/*
1920 * translate_keycode translates the key code %dx to an ascii code.
1921 */
1922	.code16
1923
1924translate_keycode:
1925	pushw	%bx
1926	pushw	%si
1927
1928	movw	$ABS(translation_table), %si
1929
19301:	lodsw
1931	/* check if this is the end */
1932	testw	%ax, %ax
1933	jz	2f
1934	/* load the ascii code into %ax */
1935	movw	%ax, %bx
1936	lodsw
1937	/* check if this matches the key code */
1938	cmpw	%bx, %dx
1939	jne	1b
1940	/* translate %dx, if successful */
1941	movw	%ax, %dx
1942
19432:	popw	%si
1944	popw	%bx
1945	ret
1946
1947	.code32
1948
1949
1950/*
1951 * remap_ascii_char remaps the ascii code %dl to another if the code is
1952 * contained in ASCII_KEY_MAP.
1953 */
1954	.code16
1955
1956remap_ascii_char:
1957	pushw	%si
1958
1959	movw	$ABS(EXT_C(ascii_key_map)), %si
19601:
1961	lodsw
1962	/* check if this is the end */
1963	testw	%ax, %ax
1964	jz	2f
1965	/* check if this matches the ascii code */
1966	cmpb	%al, %dl
1967	jne	1b
1968	/* if so, perform the mapping */
1969	movb	%ah, %dl
19702:
1971	/* restore %si */
1972	popw	%si
1973
1974	ret
1975
1976	.code32
1977
1978	.align	4
1979ENTRY(ascii_key_map)
1980	.space	(KEY_MAP_SIZE + 1) * 2
1981
1982
1983/*
1984 * int console_getkey (void)
1985 * BIOS call "INT 16H Function 00H" to read character from keyboard
1986 *	Call with	%ah = 0x0
1987 *	Return:		%ah = keyboard scan code
1988 *			%al = ASCII character
1989 */
1990
1991ENTRY(console_getkey)
1992	push	%ebp
1993
1994	call	EXT_C(prot_to_real)
1995	.code16
1996
1997	int	$0x16
1998
1999	movw	%ax, %dx		/* real_to_prot uses %eax */
2000	call	translate_keycode
2001	call	remap_ascii_char
2002
2003	DATA32	call	EXT_C(real_to_prot)
2004	.code32
2005
2006	movw	%dx, %ax
2007
2008	pop	%ebp
2009	ret
2010
2011
2012/*
2013 * int console_checkkey (void)
2014 *	if there is a character pending, return it; otherwise return -1
2015 * BIOS call "INT 16H Function 01H" to check whether a character is pending
2016 *	Call with	%ah = 0x1
2017 *	Return:
2018 *		If key waiting to be input:
2019 *			%ah = keyboard scan code
2020 *			%al = ASCII character
2021 *			Zero flag = clear
2022 *		else
2023 *			Zero flag = set
2024 */
2025ENTRY(console_checkkey)
2026	push	%ebp
2027	xorl	%edx, %edx
2028
2029	call	EXT_C(prot_to_real)	/* enter real mode */
2030	.code16
2031
2032	movb	$0x1, %ah
2033	int	$0x16
2034
2035	DATA32	jz	notpending
2036
2037	movw	%ax, %dx
2038	call	translate_keycode
2039	call	remap_ascii_char
2040	DATA32	jmp	pending
2041
2042notpending:
2043	movl	$0xFFFFFFFF, %edx
2044
2045pending:
2046	DATA32	call	EXT_C(real_to_prot)
2047	.code32
2048
2049	mov	%edx, %eax
2050
2051	pop	%ebp
2052	ret
2053
2054
2055/*
2056 * int console_getxy (void)
2057 * BIOS call "INT 10H Function 03h" to get cursor position
2058 *	Call with	%ah = 0x03
2059 *			%bh = page
2060 *      Returns         %ch = starting scan line
2061 *                      %cl = ending scan line
2062 *                      %dh = row (0 is top)
2063 *                      %dl = column (0 is left)
2064 */
2065
2066
2067ENTRY(console_getxy)
2068	push	%ebp
2069	push	%ebx                    /* save EBX */
2070
2071	call	EXT_C(prot_to_real)
2072	.code16
2073
2074        xorb	%bh, %bh                /* set page to 0 */
2075	movb	$0x3, %ah
2076	int	$0x10			/* get cursor position */
2077
2078	DATA32	call	EXT_C(real_to_prot)
2079	.code32
2080
2081	movb	%dl, %ah
2082	movb	%dh, %al
2083
2084	pop	%ebx
2085	pop	%ebp
2086	ret
2087
2088
2089/*
2090 * void console_gotoxy(int x, int y)
2091 * BIOS call "INT 10H Function 02h" to set cursor position
2092 *	Call with	%ah = 0x02
2093 *			%bh = page
2094 *                      %dh = row (0 is top)
2095 *                      %dl = column (0 is left)
2096 */
2097
2098
2099ENTRY(console_gotoxy)
2100	push	%ebp
2101	push	%ebx                    /* save EBX */
2102
2103	movb	0xc(%esp), %dl           /* %dl = x */
2104	movb	0x10(%esp), %dh          /* %dh = y */
2105
2106	call	EXT_C(prot_to_real)
2107	.code16
2108
2109        xorb	%bh, %bh                /* set page to 0 */
2110	movb	$0x2, %ah
2111	int	$0x10			/* set cursor position */
2112
2113	DATA32	call	EXT_C(real_to_prot)
2114	.code32
2115
2116	pop	%ebx
2117	pop	%ebp
2118	ret
2119
2120
2121/*
2122 * void console_cls (void)
2123 * BIOS call "INT 10H Function 09h" to write character and attribute
2124 *	Call with	%ah = 0x09
2125 *                      %al = (character)
2126 *                      %bh = (page number)
2127 *                      %bl = (attribute)
2128 *                      %cx = (number of times)
2129 */
2130
2131
2132ENTRY(console_cls)
2133	push	%ebp
2134	push	%ebx                    /* save EBX */
2135
2136	call	EXT_C(prot_to_real)
2137	.code16
2138
2139	/* move the cursor to the beginning */
2140	movb	$0x02, %ah
2141	xorb	%bh, %bh
2142	xorw	%dx, %dx
2143	int	$0x10
2144
2145	/* write spaces to the entire screen */
2146	movw	$0x0920, %ax
2147	movw	$0x07, %bx
2148	movw	$(80 * 25), %cx
2149        int	$0x10
2150
2151	/* move back the cursor */
2152	movb	$0x02, %ah
2153	int	$0x10
2154
2155	DATA32	call	EXT_C(real_to_prot)
2156	.code32
2157
2158	pop	%ebx
2159	pop	%ebp
2160	ret
2161
2162
2163/*
2164 * int console_setcursor (int on)
2165 * BIOS call "INT 10H Function 01h" to set cursor type
2166 *      Call with       %ah = 0x01
2167 *                      %ch = cursor starting scanline
2168 *                      %cl = cursor ending scanline
2169 */
2170
2171console_cursor_state:
2172	.byte	1
2173console_cursor_shape:
2174	.word	0
2175
2176ENTRY(console_setcursor)
2177	push	%ebp
2178	push	%ebx
2179
2180	/* check if the standard cursor shape has already been saved */
2181	movw	console_cursor_shape, %ax
2182	testw	%ax, %ax
2183	jne	1f
2184
2185	call	EXT_C(prot_to_real)
2186	.code16
2187
2188	movb	$0x03, %ah
2189	xorb	%bh, %bh
2190	int	$0x10
2191
2192	DATA32	call	EXT_C(real_to_prot)
2193	.code32
2194
2195	movw	%cx, console_cursor_shape
21961:
2197	/* set %cx to the designated cursor shape */
2198	movw	$0x2000, %cx
2199	movl	0xc(%esp), %ebx
2200	testl	%ebx, %ebx
2201	jz	2f
2202	movw	console_cursor_shape, %cx
22032:
2204	call	EXT_C(prot_to_real)
2205	.code16
2206
2207	movb    $0x1, %ah
2208	int     $0x10
2209
2210	DATA32	call	EXT_C(real_to_prot)
2211	.code32
2212
2213	movzbl	console_cursor_state, %eax
2214	movb	%bl, console_cursor_state
2215
2216	pop	%ebx
2217	pop	%ebp
2218	ret
2219
2220/*
2221 * getrtsecs()
2222 *	if a seconds value can be read, read it and return it (BCD),
2223 *      otherwise return 0xFF
2224 * BIOS call "INT 1AH Function 02H" to check whether a character is pending
2225 *	Call with	%ah = 0x2
2226 *	Return:
2227 *		If RT Clock can give correct values
2228 *			%ch = hour (BCD)
2229 *			%cl = minutes (BCD)
2230 *                      %dh = seconds (BCD)
2231 *                      %dl = daylight savings time (00h std, 01h daylight)
2232 *			Carry flag = clear
2233 *		else
2234 *			Carry flag = set
2235 *                         (this indicates that the clock is updating, or
2236 *                          that it isn't running)
2237 */
2238ENTRY(getrtsecs)
2239	push	%ebp
2240
2241	call	EXT_C(prot_to_real)	/* enter real mode */
2242	.code16
2243
2244	movb	$0x2, %ah
2245	int	$0x1a
2246
2247	DATA32	jnc	gottime
2248	movb	$0xff, %dh
2249
2250gottime:
2251	DATA32	call	EXT_C(real_to_prot)
2252	.code32
2253
2254	movb	%dh, %al
2255
2256	pop	%ebp
2257	ret
2258
2259
2260/*
2261 * currticks()
2262 *	return the real time in ticks, of which there are about
2263 *	18-20 per second
2264 */
2265ENTRY(currticks)
2266	pushl	%ebp
2267
2268	call	EXT_C(prot_to_real)	/* enter real mode */
2269	.code16
2270
2271	/* %ax is already zero */
2272        int	$0x1a
2273
2274	DATA32	call	EXT_C(real_to_prot)
2275	.code32
2276
2277	movl	%ecx, %eax
2278	shll	$16, %eax
2279	movw	%dx, %ax
2280
2281	popl	%ebp
2282	ret
2283
2284#endif /* STAGE1_5 */
2285
2286/*
2287 *  This is the area for all of the special variables.
2288 */
2289
2290	.p2align	2	/* force 4-byte alignment */
2291
2292protstack:
2293	.long	PROTSTACKINIT
2294
2295VARIABLE(boot_drive)
2296#ifdef SUPPORT_DISKLESS
2297	.long	NETWORK_DRIVE
2298#else
2299	.long	0
2300#endif
2301
2302VARIABLE(install_second_sector)
2303	.long	0
2304
2305	/* an address can only be long-jumped to if it is in memory, this
2306	   is used by multiple routines */
2307offset:
2308	.long	0x8000
2309segment:
2310	.word	0
2311
2312VARIABLE(apm_bios_info)
2313	.word	0	/* version */
2314	.word	0	/* cseg */
2315	.long	0	/* offset */
2316	.word	0	/* cseg_16 */
2317	.word	0	/* dseg_16 */
2318	.word	0	/* cseg_len */
2319	.word	0	/* cseg_16_len */
2320	.word	0	/* dseg_16_len */
2321
2322/*
2323 * This is the Global Descriptor Table
2324 *
2325 *  An entry, a "Segment Descriptor", looks like this:
2326 *
2327 * 31          24         19   16                 7           0
2328 * ------------------------------------------------------------
2329 * |             | |B| |A|       | |   |1|0|E|W|A|            |
2330 * | BASE 31..24 |G|/|0|V| LIMIT |P|DPL|  TYPE   | BASE 23:16 |
2331 * |             | |D| |L| 19..16| |   |1|1|C|R|A|            |
2332 * ------------------------------------------------------------
2333 * |                             |                            |
2334 * |        BASE 15..0           |       LIMIT 15..0          |
2335 * |                             |                            |
2336 * ------------------------------------------------------------
2337 *
2338 *  Note the ordering of the data items is reversed from the above
2339 *  description.
2340 */
2341
2342	.p2align	2	/* force 4-byte alignment */
2343gdt:
2344	.word	0, 0
2345	.byte	0, 0, 0, 0
2346
2347	/* code segment */
2348	.word	0xFFFF, 0
2349	.byte	0, 0x9A, 0xCF, 0
2350
2351	/* data segment */
2352	.word	0xFFFF, 0
2353	.byte	0, 0x92, 0xCF, 0
2354
2355	/* 16 bit real mode CS */
2356	.word	0xFFFF, 0
2357	.byte	0, 0x9E, 0, 0
2358
2359	/* 16 bit real mode DS */
2360	.word	0xFFFF, 0
2361	.byte	0, 0x92, 0, 0
2362
2363
2364/* this is the GDT descriptor */
2365gdtdesc:
2366	.word	0x27			/* limit */
2367	.long	gdt			/* addr */
2368