fixed
[bootcard] / bootcard.asm
index 9f70b59..384dd2d 100644 (file)
 ; ---- boot me! ----
 ; nasm -f bin -o bootcard.img bootcard.asm
 ; cat bootcard.img >/dev/<usbstick>
-; reboot
 
-       org 7c00h
-       bits 16
-
-data_start     equ 7e00h
-nticks         equ data_start
-tmoffs         equ nticks + 4
-muscur         equ tmoffs + 4
-data_end       equ muscur + 4
-
-backbuf_seg    equ 1000h
-
-OSC_FREQ       equ 1193182
-PIT_DATA0      equ 40h
-PIT_CMD                equ 43h
-PIT_CMD_CHAN0  equ 00h
-PIT_CMD_HILO   equ 30h
-PIT_CMD_SQWAVE equ 06h
-KB_CTRL                equ 61h
+%define MIDI
 
-%define DIV_ROUND(a, b)        ((a) / (b) + ((a) % (b)) / ((b) / 2))
-
-%macro setcursor 2
-       mov dl, %1
-       mov dh, %2
+       bits 16
+%ifdef DOS
+       org 100h
+       jmp start
+%else
+       org 7c00h
+%endif
+
+barh   equ 4
+nbars  equ 11
+barstart equ 200 - (nbars+1) * barh
+
+%ifdef DOS
+nticks         dd 0
+tmoffs         dd 0
+musptr         dd 0
+frame  dd 0
+fval   dd 0
+cmap   dd 0
+
+saved_tintr_offs dw 0
+saved_tintr_seg dw 0
+%else
+nticks equ 7e00h
+tmoffs equ 7e04h
+musptr equ 7e08h
+frame  equ 7e0ch
+fval   equ 7e10h
+cmap   equ 7e14h
+%endif
+
+%macro setcur 2
+       mov dx, %1 | (%2 << 8)
        xor bx, bx
        mov ah, 2
        int 10h
 %endmacro
 %macro spkon 0
-       in al, KB_CTRL
+       in al, 61h
        or al, 3
-       out KB_CTRL, al
+       out 61h, al
 %endmacro
 %macro spkoff 0
-       in al, KB_CTRL
+       in al, 61h
        and al, 0fch
-       out KB_CTRL, al
+       out 61h, al
 %endmacro
-%macro settimer 2
-       mov al, (PIT_CMD_CHAN0 + (%1 << 6)) | PIT_CMD_HILO | PIT_CMD_SQWAVE
-       out PIT_CMD, al
+%macro stimer 2
+       mov al, (%1 << 6) | 36h
+       out 43h, al
        mov ax, %2
-       out PIT_DATA0 + %1, al
+       out 40h + %1, al
        mov al, ah
-       out PIT_DATA0 + %1, al
+       out 40h + %1, al
 %endmacro
 
-start: xor ax, ax
+start:
+%ifndef DOS
+       xor ax, ax
        mov ds, ax
        mov es, ax
        mov ss, ax
        mov sp, 7c00h
 
-       mov di, data_start
-       mov cx, (data_end - data_start) / 2
-       rep stosw
-
-       mov word [32], timer_intr
+       cli
+       mov word [32], tintr
        mov [34], ax
+%else
+       ; for DOS save the previous interrupt handler to restore at the end
+       xor ax, ax
+       mov es, ax
+       mov ax, [es:32]
+       mov [saved_tintr_offs], ax
+       mov ax, [es:34]
+       mov [saved_tintr_seg], ax
+
+       cli
+       mov ax, ds
+       mov word [es:32], tintr
+       mov [es:34], ax
+%endif
+
+       mov di, nticks
+       mov cx, 16
+       rep stosw
 
-       settimer 0, DIV_ROUND(OSC_FREQ, 200)
-
+       stimer 0, 5966
+%ifdef MIDI
+       call resetmidi
+       mov ax, 0c0h    ; change program on chan 0
+       call sendmidi
+       mov ax, 19      ; church organ
+       call sendmidi
+%endif
        mov ax, 13h
        int 10h
-       push backbuf_seg
+       push 0a000h
        pop es
 
+       
+       mov al, 16
+       mov di, barstart * 320
+       mov bx, nbars
+.drawbars:
+       mov cx, barh * 320
+       rep stosb
+       inc al
+       dec bx
+       jnz .drawbars
+
+       setcur 12, 16
+       mov si, str1
+       call textout
+
        sti
 
 mainloop:
-       call drawbg
-
        mov dx, 3dah
 .invb: in al, dx
        and al, 8
-       jnz mainloop
+       jnz .invb
 .novb: in al, dx
        and al, 8
        jz .novb
 
-       push ds
-       push es
-       push es
-       pop ds
-       push 0a000h
-       pop es
-       xor di, di
-       xor si, si
-       mov cx, 32000
-       rep movsw
-       pop es
-       pop ds
-
-       setcursor 10, 0
-       mov si, str1
-       call textout
-       setcursor 12, 1
-       mov si, str2
-       call textout
-
-       jmp mainloop
-
 drawbg:
        mov bx, 200
-       mov di, 5120
+       xor di, di
 .fillgrad:
        mov ax, bx
        mov ah, al
-       mov cx, 2400    ; 15 lines
+       mov cx, 2400 ; 15 lines
        rep stosw
        inc bx
        cmp bx, 208
@@ -127,126 +151,259 @@ drawbg:
        fiadd word [w5]
        fimul word [w5]
        fistp word [bp - 2]
-       ;fistp word [bp - 4]
+       fstp st0
        mov bx, [bp - 2]
-       add bx, 100
+       add bx, 84
        imul bx, bx, 320
        add bx, cx
 .mntcol:
        mov byte [es:bx], 0
        add bx, 320
-       cmp bx, 64000
+       cmp bx, 128 * 320
        jb .mntcol
 
        dec cx
        jnz .mnt
-       
-       ret
 
+       ; upd colormap
+       mov dx, 3c8h
+       mov al, 16
+       out dx, al
+       inc dx
+       mov si, cmap
+       mov cx, 16 * 3
+       rep outsb
+
+%ifdef DOS
+       in al, 60h
+       dec al
+       jnz mainloop
+
+       mov ax, 3
+       int 10h
+
+       cli
+       xor ax, ax
+       mov es, ax
+       mov ax, [saved_tintr_offs]
+       mov word [es:32], ax
+       mov ax, [saved_tintr_seg]
+       mov [es:34], ax
+       stimer 0, 0xffff
+       sti
+       ret
+%else
+       jmp mainloop
+%endif
 
 textout:
        mov al, [si]
        and al, al
-       jz .done
+       jz textout_done
        mov ah, 0eh
        mov bx, 82
        int 10h
        inc si
        jmp textout
-.done: ret
+textout_done:
+       ret
+
+%ifdef MIDI
+note_on:
+       push ax
+       mov ax, 90h     ; note-on command for channel 0
+       call sendmidi
+       pop ax
+       call sendmidi
+       mov ax, 127
+       jmp sendmidi
+
+all_notes_off:
+       mov ax, 0b0h    ; channel mode message for channel 0...
+       call sendmidi
+       mov ax, 7bh     ; all notes off
+       call sendmidi
+       xor ax, ax
+       jmp sendmidi
+
+waitmidi:
+       mov dx, 331h
+.wait: in al, dx       ; read status port
+       test al, 40h    ; test output-ready bit (0: ready)
+       jnz .wait
+       ret
 
-timer_intr:
+sendmidi:
+       push dx
+       push ax
+       call waitmidi
+       pop ax
+       dec dx
+       out dx, al
+       pop dx
+       ret
+
+resetmidi:
+       call waitmidi
+       mov ax, 0ffh    ; reset command
+       out dx, al
+       call waitmidi
+       mov ax, 3fh     ; enter UART mode
+       out dx, al
+       ret
+%endif ; MIDI
+
+tintr:
        pusha
        mov ax, [nticks]
        inc ax
        mov [nticks], ax
 
+       mov bx, [musptr]
+%ifdef MIDI
+       cmp bx, 22*2
+%else
+       cmp bx, 22*3
+%endif
+       jnz .skiploop
+       xor bx, bx
+       mov [tmoffs], ax
+.skiploop:
+       xor cx, cx
+       mov cl, [music + bx]
+       shl cx, 4
        sub ax, [tmoffs]
-.pmus: mov bx, [muscur]
-       shl bx, 2
-       mov cx, [music + bx]    ; event time
-       cmp cx, 0ffffh
-       jz .loop
        cmp ax, cx
-       jb .eoi
-
-       inc word [muscur]
-       mov ax, [music + 2 + bx] ; event counter reload
+       jb .end
+
+%ifdef MIDI
+       mov al, [music + 1 + bx]
+       xor ah, ah
+       add bx, 2
+%else
+       mov ax, [music + 1 + bx]
+       add bx, 3
+%endif
+       mov [musptr], bx
        test ax, ax
        jz .off
+
        mov bx, ax
-       settimer 2, bx
+%ifdef MIDI
+       sub bx, 43
+%else
+       shr bx, 9
+       sub bx, 13
+%endif
+       imul bx, bx, 3
+       mov byte [cmap + bx], 3fh
+       mov word [cmap + bx + 1], 2f2fh
+
+%ifdef MIDI
+       push ax
+       call all_notes_off
+       pop ax
+       call note_on
+%else
+       mov bx, ax
+       stimer 2, bx
        spkon
-       jmp .eoi
-
-.off:  spkoff
-
+%endif
+       jmp .end
+
+.off:
+%ifdef MIDI
+       call all_notes_off
+%else
+       spkoff
+%endif
+
+.end:  test word [nticks], 1
+       jnz .eoi
+       mov cx, 16 * 3
+       mov si, cmap
+.fadecol:
+       lodsb
+       test al, al
+       jz .skipdec
+       dec al
+       mov [si-1], al
+.skipdec:
+       dec cx
+       jnz .fadecol
+       
 .eoi:  mov al, 20h
-       out 20h, al     ; EOI
+       out 20h, al
        popa
        iret
 
-.loop: neg cx
-       mov [muscur], cx
-       mov ax, [nticks]
-       mov [tmoffs], ax
-       jmp .pmus
-       
-
-str1:  db 'message message blah',0
-str2:  db 'Michael & Athena',0
-
-G2     equ 12175
-C3     equ 9121
-D3     equ 8126
-B2     equ 9664
-F3     equ 6833
-E3     equ 7239
-
-%define TM(x)  (40 + (x) * 4)
-
-music: dw 0, 0
-       dw TM(0),       G2
-       dw TM(40),      C3
-       dw TM(70),      C3
-
-       dw TM(80),      C3
-       dw TM(140),     0
-
-       dw TM(160),     G2
-       dw TM(200),     D3
-       dw TM(230),     B2
-
-       dw TM(240),     C3
-       dw TM(300),     0
-
-       dw TM(320),     G2
-       dw TM(360),     C3
-       dw TM(390),     F3
-
-       dw TM(400),     F3
-       dw TM(440),     E3
-       dw TM(470),     D3
-
-       dw TM(480),     C3
-       dw TM(520),     B2
-       dw TM(550),     C3
-
-       dw TM(560),     D3
-       dw TM(640),     0
+str1:  db 'Michael ',3,' Athena',0
+
+%ifdef MIDI
+G2     equ 43
+C3     equ 48
+D3     equ 50
+B2     equ 47
+F3     equ 53
+E3     equ 52
+%else
+G2      equ 12175
+C3      equ 9121
+D3      equ 8126
+B2      equ 9664
+F3      equ 6833
+E3      equ 7239
+%endif
+
+
+%ifdef MIDI
+%macro EV 2
+       db %1 >> 4
+       db %2
+%endmacro
+%else
+%macro EV 2
+       db %1 >> 4
+       dw %2
+%endmacro
+%endif
+
+music: EV     0,  G2
+       EV   160,  C3
+       EV   272,  C3
+       EV   320,  C3
+       EV   560,  0
+       EV   640,  G2
+       EV   800,  D3
+       EV   912,  B2
+       EV   960,  C3
+       EV  1200,  0
+       EV  1280,  G2
+       EV  1440,  C3
+       EV  1552,  F3
+       EV  1600,  F3
+       EV  1760,  E3
+       EV  1872,  D3
+       EV  1920,  C3
+       EV  2080,  B2
+       EV  2192,  C3
+       EV  2240,  D3
+       EV  2560,  0
+       EV  2720,  0
 
-       dw TM(680),     0
-       dw 0ffffh, 0
+w5:    dw 5
+w30:   dw 30
 
+%ifndef DOS
+%ifndef MIDI
        times 446-($-$$) db 0
        dd 00212080h
        dd 0820280ch
        dd 00000800h
        dd 0001f800h
+%endif
 
-w5:    dw 5
-w30:   dw 30
        times 510-($-$$) db 0
        dw 0aa55h
-
+%endif
 ; vi:ft=nasm ts=8 sts=8 sw=8: