| ; Run in dos (not under windows) and it will take us to 32 bit protected mode
 [ORG 0x100]         ; Reserve 256 bytes for dos
 [BITS 16]           ; Dos is 16 bits
 ; assemble using 'nasm' assemblaer
 ; C:>nasm asm_1.asm -o test.exe
 jmp entry           ; Jump to the start of our code
 msg1 db 'Where good to go..$'; msg2 db '32 Bit Protected Mode!..$',0x0d,0x0;
 entry:
 ; Use the interrupt 0x10 to clear the screen! mov ah,7     ; scroll down function mov al,0     ; 0 = entire window mov cx,0     ; 0,0 as upper left corner. mov dx,184fh ; 24,79 as lower right corner. mov bh,7     ; normal attribute int 10h      ; call bios                    ; Display a message showing where alive!
 mov dx, msg1        ; register dx=msg1 mov ah, 9           ; register ah=9 -- the print string function int 21h             ; dos service interrupt .. looks at register ah to figure out what to do
 ; Thanks from Brendan, as we have to make sure our GDTR points to the actual ; memory address, add code location and dos 0x100 onto our loaded offset 
     mov eax,0     mov ax,cs     shl eax,4     mov   ebx,eax          mov   [codesel + 2],ax                                         mov   [datasel + 2],ax                                     	 	shr   eax,16 	mov   [codesel + 4],al 	mov   [datasel + 4],al 	mov   [codesel + 7],ah 	mov   [datasel + 7],ah 	 	mov   eax, ebx 	     add [gdtr+2],eax     add [idtr+2],eax         ; set idtr and gdtr so it points to the 'real' address in memory
 ;---------------------------------------------------------------------------
   cli		    ; Clear or disable interrupts      mov     al, 0x70   mov     dx, 0x80   out     dx, al      ; outb(0x80, 0x70) - disable NMI (Non Maskable Interupts)          
   lgdt[gdtr]	    ; Load GDT      lidt[idtr]       ; Load IDT
      mov eax,cr0	    ; The lsb of cr0 is the protected mode bit   or al,0x01	    ; Set protected mode bit   mov cr0,eax	    ; Mov modified word to the control register
 jmp   0x8:go_pm     ; The 0x8 is so we select our code segment from our gdtr
 nop                 ; ignore - no operation opcodes :) nop
 align 4 ;--------------------------------------------------------------------------- ;                                 32 BIT ;--------------------------------------------------------------------------- ; Once we reach here where in protected mode!  32 Bit!  Where not in ; the real world (mode) anymore :) [BITS 32] go_pm :
 xor   edi,edi xor   esi,esi
 mov ax, 0x10        ; use our datasel selector ( alternatively mov ax, datasel-gdt ) mov ds, ax mov ss, ax mov es, ax mov fs, ax
 mov ax, 0x18 mov gs, ax
 mov esp, 0afffh    ; we need a simple stack if where calling functions!                    ; Such as interupt functions etc, as the return address is                    ; put on the stack remember!
 mov word [gs: 0xb8000],0x740 ; put a char to the screen!...yeahh!                              ; '@' in the top left of the screen if we made it                              ; here okay.
 ;========================================================================= ; Okay, its worth poking onto the screen a more descriptive message ; that we've made it to the 32 bit world!  We use this method of char by ; char poking first, but we'll use our text out functions later...just ; easier to see how it works :) ;========================================================================= ; ; Write '[ 32 bits OK   ]' at [gs:0B80A0h]. ; ;-------------------------------------------------------------------------   mov     byte [gs:0B80A0h], '['   mov     byte [gs:0B80A1h], 02h              ; Assign a color code   mov     byte [gs:0B80A2h], ' '   mov     byte [gs:0B80A3h], 02h              ; Assign a color code   mov     byte [gs:0B80A4h], '3'   mov     byte [gs:0B80A5h], 02h              ; Assign a color code   mov     byte [gs:0B80A6h], '2'   mov     byte [gs:0B80A7h], 02h              ; Assign a color code   mov     byte [gs:0B80A8h], ' '   mov     byte [gs:0B80A9h], 02h              ; Assign a color code   mov     byte [gs:0B80AAh], 'b'   mov     byte [gs:0B80ABh], 02h              ; Assign a color code   mov     byte [gs:0B80ACh], 'i'   mov     byte [gs:0B80ADh], 02h              ; Assign a color code   mov     byte [gs:0B80AEh], 't'   mov     byte [gs:0B80AFh], 02h              ; Assign a color code   mov     byte [gs:0B80B0h], 's'   mov     byte [gs:0B80B1h], 02h              ; Assign a color code   mov     byte [gs:0B80B2h], ' '   mov     byte [gs:0B80B3h], 02h              ; Assign a color code   mov     byte [gs:0B80B4h], 'O'   mov     byte [gs:0B80B5h], 02h              ; Assign a color code   mov     byte [gs:0B80B6h], 'K'   mov     byte [gs:0B80B7h], 02h              ; Assign a color code   mov     byte [gs:0B80B8h], ' '   mov     byte [gs:0B80B9h], 02h              ; Assign a color code   mov     byte [gs:0B80BAh], ' '   mov     byte [gs:0B80BBh], 02h              ; Assign a color code   mov     byte [gs:0B80BCh], ' '   mov     byte [gs:0B80BDh], 02h              ; Assign a color code   mov     byte [gs:0B80BEh], ']'   mov     byte [gs:0B80BFh], 02h              ; Assign a color code
 ;=========================================================================
 ;------------------------------------------------------------------------- ; Simple functions included here ;------------------------------------------------------------------------- jmp over_includes
 %include 'include\text.inc'        ; 32-bit. Set default text functions. %include 'include\datetime.inc'    ; 32-bit. Get/Set CMOS date time functions
 msg_basic     db 0xd2,0x7,'Working in 32 BIT Mode [ ',0xd2,0x2, ' Loaded OK', 0xd2,0x7, ' ]',0x0d,0x0
 msg_irqs_off     db   "Disabling IRQ's.................", 0x00 msg_remap_pics   db   "ReMapping the PIC's.............", 0x00 msg_sti_on       db   "Turning Interrupts back on......", 0x00 msg_loop         db   "Going into infinite loop........", 0x00
 msg_ok        db 0xd2,0x2, "[ OK ]", 0xd2,0x7,0xd,0x0
 over_includes: ;-------------------------------------------------------------------------
 ; 0x2 - green ; 0x7 - light grey ; 0x4 - red
 mov al, 0x09 call TextColor
 mov eax, 0x0 call set_text_pos
 call carriage_return call carriage_return
 mov al, 'B' call print_char
 call inc_scr_pointer
 mov esi, msg2 call print_string
 mov esi, msg_basic call print_string
 ;-------------------------------------------------------------------------
 call time      ; get time values (datetime.inc) call date      ; get date values (datetime.inc)
 call print_time call print_date
 ; Force a call to interrupt 4! ; Int 0x4           ; We call our interrupt 4 subroutine
 ;------------------------------------------------------------------------- ; Divide by Zero Warning ;------------------------------------------------------------------------- ; Just remember, when we do a divide by zero, and the interupt is called, ; the return address passed to the interrupt, is in fact the address of the ; line that caused the interrupt!  So if we just return from the interrupt ; it would just keep causing the interrupt over and over again. ;------------------------------------------------------------------------- ; Do a divide by 0 error, so we force a call to our interrupt 0 ;  mov eax, 0 ;  mov ebx, 0 ;  div ebx            ; eax divided by ebx, and stored back in eax
 mov byte [es: 0xb8002], "A" ; poke a character onto our screen buffer
 nop nop
 ;------------------------------------------------------------------------- ; Where not in basics anymore! ;------------------------------------------------------------------------- ; This is the line where we go from simple settups, to using the computers ; interal hardward, fiddling around with the irqs and pic (Programmable ; Interupt Controller) etc. ;-------------------------------------------------------------------------
 mov esi, msg_irqs_off call print_string
 ; Disable all IRQs. ;------------------------------------------------------------------------- disable_irqs:           mov     al, 0xFF           mov     dx, 0x21           out     dx, al      ; outb(0x21, 0xFF)
           mov     al, 0xFF           mov     dx, 0xA1           out     dx, al      ; outb(0x21, 0xFF)            ;-------------------------------------------------------------------------          
 mov esi, msg_ok call print_string
 mov esi, msg_remap_pics call print_string
 ;------------------------------------------------------------------------- ; ReMap PICs ;------------------------------------------------------------------------- ;   PIC 1 & 2 (Master and Slave) ;   Lower 8 IRQs 0x20 onwards ;   Higher 8 IRQs 0x28 onwards ;------------------------------------------------------------------------- remap_pics:
      ; IWC1      ;------           mov     al, 0x11           out     0x20, al      ; outb(0x20, 0x11)
           mov     al, 0x11           out     0xA0, al      ; outb(0xA0, 0x11)        ; IWC2      ;------           mov     al, 0x20           out     21, al      ; outb(0x21, pic1)
           mov     al, 0x28           out     0xA1, al      ; outb(0xA1, pic2)
      ; IWC3      ;------           mov     al, 0x04           out     0x21, al      ; outb(0x21, 4)
           mov     al, 0x02           out     0xA1, al      ; outb(0xA1, 2)
      ; IWC4      ;------           mov     al, 0x01           out     0x21, al      ; outb(0x21, 0x01)
           mov     al, 0x01           out     0xA1, al      ; outb(0xA1, 0x01) ;-------------------------------------------------------------------------
 mov esi, msg_ok call print_string           ;-------------------------------------------------------------------------
 ; Every time an irq interupt occurs, we must clear it before another irq ; is sent.  Else another interupt wont' be sent till its been cleared. ; We usually call this at the end of our interupt routine.
 ;EOI for IRQ 0-7 mov     al, 0x20         mov dx, 0x20         out dx, al      ; outb(0x20, 0x20)          ;-------------------------------------------------------------------------
 mov esi, msg_sti_on call print_string
 ;-------------------------------------------------------------------------
 sti              ; Interrupts back..
 ;-------------------------------------------------------------------------
 mov esi, msg_ok call print_string
 ;-------------------------------------------------------------------------
 mov esi, msg_loop call print_string
 lp: jmp lp  ; loops here forever and ever...
 ; Stays in our loop forever, till something happens, such as an interrupt ; is called by pressing a key or the timer calls an interrupt etc
 ; We use 16 bits here - as you'll notice we use dw and dd only, ; and out data will be packed together nice and tight.
 [BITS 16] align 4
 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; Our GDTR register value ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
 gdtr :    dw gdt_end-gdt-1    ; Length of the gdt    dd gdt	       ; physical address of gdt
 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; This is the start of our gdt ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; align 4
 gdt:     gdt0 		        ; 0x0    dd 0		          dd 0                 codesel:            ; 0x8    dw 0x0ffff	          dw 0x0000	           db 0x00             	    db 0x09a	          db 0x0cf	           db 0x00	        datasel:            ; 0x10    dw 0x0ffff	           dw 0x0000	          db 0x00	           db 0x092    db 0x0cf    db 0x00 linearsel:          ; 0x18    dw 0x0ffff	           dw 0x0000	          db 0x00	           db 0x092    db 0x0cf    db 0x00 gdt_end:
 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
 align 4 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; Our IDTR register value ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
 idtr :    dw idt_end - idt_start - 1  ; Length of the idt    dd idt_start                ; physical address of idt
 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; This is the start of our idt - its actual value ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;------------------------------------------------------------------------- [BITS 16] align 4
 idt_start: %rep 256                         ; We have enough room for 256 ISRs         dw intfunc               ; offset 15:0         dw 0x0008                ; selector         dw 0x8E00                ; present,ring 0,386 interrupt gate         dw 0                     ; offset 31:16 %endrep idt_end:
 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;------------------------------------------------------------------------- ;                        Interrupt Routine ;------------------------------------------------------------------------- [bits 32] align 4
 intfunc: pushad push es
 nop mov ax,0x10 mov es,ax                             mov byte [es: 0xb8012], "I" ; poke a character into the graphics output screen
 pop es popad iret
 ;-------------------------------------------------------------------------
 TIMES 0x1500-($-$$) DB 0x90    ; And of course, this will make our file size                              ; equal to 0x1500 a nice round number -                              ; 0x1500 ... so if you assemble the file                              ; you should find its that size exactly.
 ;-------------------------------------------------------------------------
 |