mirror of
https://github.com/xcat2/xNBA.git
synced 2025-01-22 15:33:14 +00:00
498 lines
12 KiB
NASM
498 lines
12 KiB
NASM
; Copyright (C) 1997 Markus Gutschke <gutschk@uni-muenster.de>
|
||
;
|
||
; This program is free software; you can redistribute it and/or modify
|
||
; it under the terms of the GNU General Public License as published by
|
||
; the Free Software Foundation; either version 2 of the License, or
|
||
; any later version.
|
||
;
|
||
; This program is distributed in the hope that it will be useful,
|
||
; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
; GNU General Public License for more details.
|
||
;
|
||
; You should have received a copy of the GNU General Public License
|
||
; along with this program; if not, write to the Free Software
|
||
; Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||
|
||
; Prepend this image file to an arbitrary ROM image. The resulting binary
|
||
; can be loaded from any BOOT-Prom that supports the "nbi" file format.
|
||
; When started, the image will reprogram the flash EPROM on the FlashCard
|
||
; ISA card. The flash EPROM has to be an AMD 29F010, and the programming
|
||
; algorithm is the same as that suggested by AMD in the appropriate data
|
||
; sheets.
|
||
|
||
|
||
#define SEGLOW 0xC800 /* lower range for EPROM segment */
|
||
#define SEGHIGH 0xE800 /* upper range for EPROM segment */
|
||
#define AMD_ID 0x2001 /* flash EPROM ID, only support AMD */
|
||
#define ERASE1_CMD 0x80 /* first cmd for erasing full chip */
|
||
#define ERASE2_CMD 0x10 /* second cmd for erasing full chip */
|
||
#define READID_CMD 0x90 /* cmd to read chip ID */
|
||
#define PROG_CMD 0xA0 /* cmd to program a byte */
|
||
#define RESET_CMD 0xF0 /* cmd to reset chip state machine */
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
|
||
.text
|
||
.org 0
|
||
|
||
; .globl _main
|
||
_main: mov ax,#0x0FE0
|
||
mov ds,ax
|
||
mov ax,magic ; verify that we have been loaded by
|
||
cmp ax,#0xE4E4 ; boot prom
|
||
jnz lderr
|
||
jmpi 0x200,0x0FE0 ; adjust code segment
|
||
lderr: mov si,#loaderr
|
||
cld
|
||
lderrlp:seg cs
|
||
lodsb ; loop over all characters of
|
||
or al,al ; string
|
||
jnz lderrnx
|
||
xor ah,ah
|
||
int 0x16 ; wait for keypress
|
||
jmpi 0x0000,0xFFFF ; reboot!
|
||
lderrnx:mov ah,#0x0E ; print it
|
||
mov bl,#0x07
|
||
xor bh,bh
|
||
int 0x10
|
||
jmp lderrlp
|
||
|
||
loaderr:.ascii "The flash EPROM utility has to be loaded from a BOOT-Prom"
|
||
.byte 0xa,0xd
|
||
.ascii "that knows about the 'nbi' file format!"
|
||
.byte 0xa,0xd
|
||
.ascii "Reboot to proceed..."
|
||
.byte 0
|
||
|
||
.org 510
|
||
.byte 0x55,0xAA
|
||
|
||
!----------------------------------------------------------------------------
|
||
|
||
start: mov ax,cs
|
||
mov ds,ax
|
||
mov ax,romdata ; verify that there is an Prom image
|
||
cmp ax,#0xAA55 ; attached to the utility
|
||
jnz resmag
|
||
mov al,romdata+2
|
||
or al,al ; non-zero size is required
|
||
jnz magicok
|
||
resmag: mov si,#badmagic ; print error message
|
||
reset: call prnstr
|
||
xor ah,ah
|
||
int 0x16 ; wait for keypress
|
||
jmpi 0x0000,0xFFFF ; reboot!
|
||
magicok:mov di,#clrline1
|
||
mov si,#welcome ; print welcome message
|
||
inpnew: call prnstr
|
||
inprest:xor bx,bx
|
||
mov cl,#0xC ; expect 4 nibbles input data
|
||
inploop:xor ah,ah
|
||
int 0x16
|
||
cmp al,#0x8 ; <Backspace>
|
||
jnz inpnobs
|
||
or bx,bx ; there has to be at least one input ch
|
||
jz inperr
|
||
mov si,#delchar ; wipe out char from screen
|
||
call prnstr
|
||
add cl,#4 ; compute bitmask for removing input
|
||
mov ch,cl
|
||
mov cl,#0xC
|
||
sub cl,ch
|
||
mov ax,#0xFFFF
|
||
shr ax,cl
|
||
not ax
|
||
and bx,ax
|
||
mov cl,ch
|
||
inploop1:jmp inploop
|
||
inpnobs:cmp al,#0x0D ; <Return>
|
||
jnz inpnocr
|
||
or bx,bx ; zero input -> autoprobing
|
||
jz inpdone
|
||
cmp cl,#-4 ; otherwise there have to be 4 nibbles
|
||
jz inpdone
|
||
inperr: mov al,#7 ; ring the console bell
|
||
jmp inpecho
|
||
inpnocr:cmp al,#0x15 ; <CTRL-U>
|
||
jnz inpnokl
|
||
mov si,di
|
||
call prnstr ; clear entire input and restart
|
||
jmp inprest
|
||
inpnokl:cmp cl,#-4 ; cannot input more than 4 nibbles
|
||
jz inperr
|
||
cmp al,#0x30 ; '0'
|
||
jb inperr
|
||
ja inpdig
|
||
or bx,bx ; leading '0' is not allowed
|
||
jz inperr
|
||
inpdig: cmp al,#0x39 ; '9'
|
||
ja inpnodg
|
||
mov ch,al
|
||
sub al,#0x30
|
||
inpnum: xor ah,ah ; compute new input value
|
||
shl ax,cl
|
||
add ax,bx
|
||
test ax,#0x1FF ; test for 8kB boundary
|
||
jnz inperr
|
||
cmp ax,#SEGHIGH ; input has to be below E800
|
||
jae inperr
|
||
cmp ax,#SEGLOW ; and above/equal C800
|
||
jae inpok
|
||
cmp cl,#0xC ; if there is just one nibble, yet,
|
||
jnz inperr ; then the lower limit ix C000
|
||
cmp ax,#0xC000
|
||
jb inperr
|
||
inpok: mov bx,ax ; adjust bitmask
|
||
sub cl,#4
|
||
mov al,ch
|
||
inpecho:call prnchr ; output new character
|
||
jmp inploop1
|
||
inpnodg:and al,#0xDF ; lower case -> upper case
|
||
cmp al,#0x41 ; 'A'
|
||
jb inperr
|
||
cmp al,#0x46 ; 'F'
|
||
ja inperr
|
||
mov ch,al
|
||
sub al,#0x37
|
||
jmp inpnum
|
||
inpdone:or bx,bx ; zero -> autoprobing
|
||
jnz probe
|
||
mov si,#automsg
|
||
call prnstr
|
||
mov cx,#0x10
|
||
mov bx,#SEGHIGH ; scan from E800 to C800
|
||
autoprb:sub bx,#0x0200 ; stepping down in 8kB increments
|
||
mov di,bx
|
||
call readid
|
||
cmp ax,#AMD_ID
|
||
jz prbfnd
|
||
loop autoprb
|
||
mov si,#failmsg
|
||
nofnd: mov di,#clrline2
|
||
jmp near inpnew ; failure -> ask user for new input
|
||
probe: mov di,bx
|
||
test bx,#0x07FF ; EPROM might have to be aligned to
|
||
jz noalign ; 32kB boundary
|
||
call readid
|
||
cmp ax,#AMD_ID ; check for AMDs id
|
||
jz prbfnd
|
||
mov si,#alignmsg
|
||
call prnstr
|
||
and bx,#0xF800 ; enforce alignment of hardware addr
|
||
noalign:call readid ; check for AMDs id
|
||
cmp ax,#AMD_ID
|
||
jz prbfnd
|
||
mov si,#nofndmsg ; could not find any EPROM at speci-
|
||
call prnstr ; fied location --- even tried
|
||
mov si,#basemsg ; aligning to 32kB boundary
|
||
jmp nofnd ; failure -> ask user for new input
|
||
prbfnd: mov si,#fndmsg
|
||
call prnstr ; we found a flash EPROM
|
||
mov ax,bx
|
||
call prnwrd
|
||
mov si,#ersmsg
|
||
call prnstr
|
||
call erase ; erase old contents
|
||
jnc ersdone
|
||
mov si,#failresmsg ; failure -> reboot machine
|
||
jmp near reset
|
||
ersdone:mov si,#prg1msg ; tell user that we are about
|
||
call prnstr ; to program the new data into
|
||
mov ax,di ; the specified range
|
||
call prnwrd
|
||
mov si,#prg2msg
|
||
call prnstr
|
||
xor dh,dh
|
||
mov dl,romdata+2
|
||
shl dx,#1
|
||
mov ah,dh
|
||
mov cl,#4
|
||
shl ah,cl
|
||
xor al,al
|
||
add ax,di
|
||
call prnwrd
|
||
mov al,#0x3A ; ':'
|
||
call prnchr
|
||
mov ah,dl
|
||
xor al,al
|
||
dec ax
|
||
call prnwrd
|
||
mov al,#0x20
|
||
call prnchr
|
||
mov dh,romdata+2 ; number of 512 byte blocks
|
||
push ds
|
||
mov ax,ds
|
||
add ax,#romdata>>4 ; adjust segment descriptor, so that
|
||
mov ds,ax ; we can handle images which are
|
||
prgloop:mov cx,#0x200 ; larger than 64kB
|
||
xor si,si
|
||
xor bp,bp
|
||
call program ; program 512 data bytes
|
||
jc prgerr ; check error condition
|
||
mov ax,ds
|
||
add ax,#0x20 ; increment segment descriptors
|
||
mov ds,ax
|
||
add di,#0x20
|
||
dec dh ; decrement counter
|
||
jnz prgloop
|
||
pop ds
|
||
mov si,#donemsg ; success -> reboot
|
||
prgdone:call prnstr
|
||
mov si,#resetmsg
|
||
jmp near reset
|
||
prgerr: pop ds ; failure -> reboot
|
||
mov si,#failresmsg
|
||
jmp prgdone
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; READID -- read EPROM id number, base address is passed in BX
|
||
; ======
|
||
;
|
||
; changes: AX, DL, ES
|
||
|
||
readid: mov dl,#RESET_CMD ; reset chip
|
||
call sendop
|
||
mov dl,#READID_CMD
|
||
call sendop ; send READID command
|
||
mov es,bx
|
||
seg es
|
||
mov ax,0x00 ; read manufacturer ID
|
||
mov dl,#RESET_CMD
|
||
jmp sendop ; reset chip
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; ERASE -- erase entire EPROM, base address is passed in BX
|
||
; =====
|
||
;
|
||
; changes: AL, CX, DL, ES, CF
|
||
|
||
erase: mov dl,#ERASE1_CMD
|
||
call sendop ; send ERASE1 command
|
||
mov dl,#ERASE2_CMD
|
||
call sendop ; send ERASE2 command
|
||
xor bp,bp
|
||
mov al,#0xFF
|
||
push di
|
||
mov di,bx
|
||
call waitop ; wait until operation finished
|
||
pop di
|
||
jnc erfail
|
||
mov dl,#RESET_CMD
|
||
call sendop ; reset chip
|
||
stc
|
||
erfail: ret
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; PROGRAM -- write data block at DS:SI of length CX into EPROM at DI:BP
|
||
; =======
|
||
;
|
||
; changes: AX, CX, DL, BP, ES, CF
|
||
|
||
program:mov dl,#PROG_CMD
|
||
call sendop ; send programming command
|
||
lodsb ; get next byte from buffer
|
||
mov es,di
|
||
seg es
|
||
mov byte ptr [bp],al ; write next byte into flash EPROM
|
||
call waitop ; wait until programming operation is
|
||
jc progdn ; completed
|
||
inc bp
|
||
loop program ; continue with next byte
|
||
clc ; return without error
|
||
progdn: ret
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; SENDOP -- send command in DL to EPROM, base address is passed in BX
|
||
; ======
|
||
;
|
||
; changes: ES
|
||
|
||
sendop: mov es,bx
|
||
seg es
|
||
mov byte ptr 0x5555,#0xAA ; write magic data bytes into
|
||
jcxz so1 ; magic locations. This unlocks
|
||
so1: jcxz so2 ; the flash EPROM. N.B. that the
|
||
so2: seg es ; magic locations are mirrored
|
||
mov byte ptr 0x2AAA,#0x55 ; every 32kB; the hardware address
|
||
jcxz so3 ; might have to be adjusted to a
|
||
so3: jcxz so4 ; 32kB boundary
|
||
so4: seg es
|
||
mov byte ptr 0x5555,dl
|
||
ret
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; WAITOP -- wait for command to complete, address is passed in DI:BP
|
||
; ======
|
||
;
|
||
; for details on the programming algorithm, c.f. http://www.amd.com
|
||
;
|
||
; changes: AX, DL, ES, CF
|
||
|
||
waitop: and al,#0x80 ; monitor bit 7
|
||
mov es,di
|
||
wait1: seg es ; read contents of EPROM cell that is
|
||
mov ah,byte ptr [bp] ; being programmed
|
||
mov dl,ah
|
||
and ah,#0x80
|
||
cmp al,ah ; bit 7 indicates sucess
|
||
je waitok
|
||
test dl,#0x20 ; bit 5 indicates timeout/error
|
||
jz wait1 ; otherwise wait for cmd to complete
|
||
seg es
|
||
mov ah,byte ptr [bp] ; check error condition once again,
|
||
and ah,#0x80 ; because bits 7 and 5 can change
|
||
cmp al,ah ; simultaneously
|
||
je waitok
|
||
stc
|
||
ret
|
||
waitok: clc
|
||
ret
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; PRNSTR -- prints a string in DS:SI onto the console
|
||
; ======
|
||
;
|
||
; changes: AL
|
||
|
||
prnstr: push si
|
||
cld
|
||
prns1: lodsb ; loop over all characters of
|
||
or al,al ; string
|
||
jz prns2
|
||
call prnchr ; print character
|
||
jmp prns1
|
||
prns2: pop si
|
||
ret
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
; PRNWRD, PRNBYT, PRNNIB, PRNCHR -- prints hexadezimal values, or ASCII chars
|
||
; ====== ====== ====== ======
|
||
;
|
||
; changes: AX
|
||
|
||
prnwrd: push ax
|
||
mov al,ah
|
||
call prnbyt ; print the upper byte
|
||
pop ax
|
||
prnbyt: push ax
|
||
shr al,1 ; prepare upper nibble
|
||
shr al,1
|
||
shr al,1
|
||
shr al,1
|
||
call prnnib ; print it
|
||
pop ax
|
||
prnnib: and al,#0x0F ; prepare lower nibble
|
||
add al,#0x30
|
||
cmp al,#0x39 ; convert it into hex
|
||
jle prnchr
|
||
add al,#7
|
||
prnchr: push bx
|
||
mov ah,#0x0E ; print it
|
||
mov bl,#0x07
|
||
xor bh,bh
|
||
int 0x10
|
||
pop bx
|
||
ret
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
magic: .byte 0xE4,0xE4
|
||
|
||
badmagic:.byte 0xa,0xd
|
||
.ascii "There does not appear to be a ROM image attached to the"
|
||
.ascii "flash EPROM utility;"
|
||
.byte 0xa,0xd
|
||
resetmsg:.ascii "Reboot to proceed..."
|
||
.byte 0
|
||
|
||
welcome:.byte 0xa,0xd
|
||
.ascii "Flash EPROM programming utility V1.0"
|
||
.byte 0xa,0xd
|
||
.ascii "Copyright (c) 1997 by M. Gutschke <gutschk@uni-muenster.de>"
|
||
.byte 0xa,0xd
|
||
.ascii "==========================================================="
|
||
.byte 0xa,0xd
|
||
prompt: .byte 0xa,0xd
|
||
.ascii "Enter base address for AMD29F010 flash EPROM on FlashCard or"
|
||
.byte 0xa,0xd
|
||
.ascii "press <RETURN> to start autoprobing; the base address has"
|
||
.byte 0xa
|
||
clrline1:.byte 0xd
|
||
.ascii "to be in the range C800..E600: "
|
||
.ascii " "
|
||
.byte 0x8,0x8,0x8,0x8
|
||
.byte 0
|
||
|
||
delchar:.byte 0x8,0x20,0x8
|
||
.byte 0
|
||
|
||
automsg:.ascii "autoprobing... "
|
||
.byte 0
|
||
|
||
failmsg:.ascii "failed!"
|
||
basemsg:.byte 0xa
|
||
clrline2:.byte 0xd
|
||
.ascii "Enter base address: "
|
||
.ascii " "
|
||
.byte 0x8,0x8,0x8,0x8
|
||
.byte 0
|
||
|
||
fndmsg: .byte 0xa,0xd
|
||
.ascii "Found flash EPROM at: "
|
||
.byte 0
|
||
|
||
alignmsg:.byte 0xa,0xd
|
||
.ascii "FlashCard requires the hardware address to be aligned to a"
|
||
.byte 0xa,0xd
|
||
.ascii "32kB boundary; automatically adjusting..."
|
||
.byte 0
|
||
|
||
nofndmsg:.byte 0xa,0xd
|
||
.ascii "No AMD29F010 flash EPROM found"
|
||
.byte 0
|
||
|
||
ersmsg: .byte 0xa,0xd
|
||
.ascii "Erasing old contents... "
|
||
.byte 0
|
||
|
||
prg1msg:.ascii "done"
|
||
.byte 0xa,0xd
|
||
.ascii "Programming from "
|
||
.byte 0
|
||
|
||
prg2msg:.ascii ":0000 to "
|
||
.byte 0
|
||
|
||
donemsg:.ascii "done!"
|
||
.byte 0xa,0xd
|
||
.byte 0
|
||
|
||
failresmsg:
|
||
.ascii "failed!"
|
||
.byte 0xa,0xd
|
||
.byte 0
|
||
|
||
|
||
;----------------------------------------------------------------------------
|
||
|
||
.align 16
|
||
.org *-1
|
||
.byte 0x00
|
||
romdata:
|