1
mirror of https://github.com/rapid7/metasploit-payloads synced 2025-01-14 17:37:27 +01:00

Add child window enumeration and flag

Enumeration of child windows is now possible if the appropriate TLV is
included in the request message.

Inclusion of "unknown" windows is also possible now, again if the
appropriate TLV is included.
This commit is contained in:
OJ 2013-10-15 17:52:51 +10:00
parent 984880d8b2
commit a61bbc5a7d
2 changed files with 29 additions and 14 deletions

View File

@ -13,6 +13,7 @@
#define TLV_TYPE_EXT_WINDOW_ENUM_PID MAKE_CUSTOM_TLV(TLV_META_TYPE_UINT, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 2)
#define TLV_TYPE_EXT_WINDOW_ENUM_HANDLE MAKE_CUSTOM_TLV(TLV_META_TYPE_QWORD, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 3)
#define TLV_TYPE_EXT_WINDOW_ENUM_TITLE MAKE_CUSTOM_TLV(TLV_META_TYPE_STRING, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 4)
#define TLV_TYPE_EXT_WINDOW_ENUM_INCLUDEUNKNOWN MAKE_CUSTOM_TLV(TLV_META_TYPE_BOOL, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 5)
#define TLV_TYPE_EXT_SERVICE_ENUM_GROUP MAKE_CUSTOM_TLV(TLV_META_TYPE_GROUP, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 10)
#define TLV_TYPE_EXT_SERVICE_ENUM_NAME MAKE_CUSTOM_TLV(TLV_META_TYPE_STRING, TLV_TYPE_EXTENSION_EXTAPI, TLV_EXTENSIONS + 11)

View File

@ -11,13 +11,14 @@ DWORD enumerate_windows( Packet *response );
#ifdef _WIN32
#define MAX_WINDOW_TITLE 256
typedef BOOL (WINAPI * PENUMDESKTOPWINDOWS)( HDESK hDesktop, WNDENUMPROC enumProc, LPARAM lparam );
typedef BOOL (WINAPI * PENUMCHILDWINDOWS)( HWND hWndParent, WNDENUMPROC enumProc, LPARAM lparam );
typedef int (WINAPI * PGETWINDOWTEXA)( HWND hWnd, LPSTR lpString, int nMaxCount );
typedef DWORD (WINAPI * PGETWINDOWTHREADPROCESSID)( HWND hWnd, LPDWORD lpdwProcessId );
typedef struct _EnumWindowsState
{
Packet* pResponse;
BOOL bIncludeUnknown;
PGETWINDOWTEXA pGetWindowTextA;
PGETWINDOWTHREADPROCESSID pGetWindowThreadProcessId;
} EnumWindowsState;
@ -35,9 +36,12 @@ BOOL CALLBACK enumerate_windows_callback( HWND hWnd, LPARAM lParam )
{
dprintf( "Getting window title %p", pState->pGetWindowTextA );
if( pState->pGetWindowTextA( hWnd, windowTitle, MAX_WINDOW_TITLE ) == 0 ) {
strncpy_s( windowTitle, MAX_WINDOW_TITLE, "<unknown>", MAX_WINDOW_TITLE - 1 );
dprintf( "Unable to get window title. Skipping." );
break;
dprintf( "Unable to get window title. Setting to <unknown>." );
if( pState->bIncludeUnknown ) {
strncpy_s( windowTitle, MAX_WINDOW_TITLE, "<unknown>", MAX_WINDOW_TITLE - 1 );
} else {
break;
}
}
dprintf( "Getting process ID %p", pState->pGetWindowThreadProcessId );
@ -51,14 +55,14 @@ BOOL CALLBACK enumerate_windows_callback( HWND hWnd, LPARAM lParam )
}
#endif
DWORD enumerate_windows( Packet *response )
DWORD enumerate_windows( Packet *response, BOOL bIncludeUnknown, QWORD parentWindow )
{
#ifdef _WIN32
// currently we only support Windoze
DWORD dwResult;
HMODULE hUser32 = NULL;
PENUMDESKTOPWINDOWS pEnumDesktopWindows;
PENUMCHILDWINDOWS pEnumChildWindows;
EnumWindowsState state;
do
@ -67,10 +71,6 @@ DWORD enumerate_windows( Packet *response )
if( (hUser32 = LoadLibraryA( "user32.dll" )) == NULL)
BREAK_ON_ERROR( "Unable to load user32.dll" );
dprintf( "Searching for EnumDesktopWindows" );
if( (pEnumDesktopWindows = (PENUMDESKTOPWINDOWS)GetProcAddress( hUser32, "EnumDesktopWindows" )) == NULL )
BREAK_ON_ERROR( "Unable to locate EnumDesktopWindows in user32.dll" );
dprintf( "Searching for GetWindowTextA" );
if( (state.pGetWindowTextA = (PGETWINDOWTEXA)GetProcAddress( hUser32, "GetWindowTextA" )) == NULL )
BREAK_ON_ERROR( "Unable to locate GetWindowTextA in user32.dll" );
@ -82,10 +82,15 @@ DWORD enumerate_windows( Packet *response )
dprintf( "Found GetWindowThreadProcessId %p", state.pGetWindowThreadProcessId );
state.pResponse = response;
state.bIncludeUnknown = bIncludeUnknown;
dprintf( "Beginning enumeration of desktop windows" );
if( !pEnumDesktopWindows( NULL, (WNDENUMPROC)enumerate_windows_callback, (LPARAM)&state ) )
BREAK_ON_ERROR( "Failed to enumerate windows" );
dprintf( "Searching for EnumChildWindows" );
if( (pEnumChildWindows = (PENUMCHILDWINDOWS)GetProcAddress( hUser32, "EnumChildWindows" )) == NULL )
BREAK_ON_ERROR( "Unable to locate EnumChildWindows in user32.dll" );
dprintf( "Beginning enumeration of child windows with parent %u", parentWindow );
if( !pEnumChildWindows( parentWindow != 0 ? (HWND)parentWindow : NULL, (WNDENUMPROC)enumerate_windows_callback, (LPARAM)&state ) )
BREAK_ON_ERROR( "Failed to enumerate child windows" );
dwResult = ERROR_SUCCESS;
} while(0);
@ -101,7 +106,9 @@ DWORD enumerate_windows( Packet *response )
DWORD request_window_enum( Remote *remote, Packet *packet )
{
QWORD parentWindow = NULL;
DWORD dwResult = ERROR_SUCCESS;
BOOL bIncludeUnknown = FALSE;
Packet * response = packet_create_response( packet );
do
@ -112,8 +119,15 @@ DWORD request_window_enum( Remote *remote, Packet *packet )
break;
}
// Extract the specified parent window. If this is NULL, that's ok, as we'll
// just enumerate top-level windows.
parentWindow = packet_get_tlv_value_qword( packet, TLV_TYPE_EXT_WINDOW_ENUM_HANDLE );
// Extract the flag that indicates of unknown windows should be included in the output
bIncludeUnknown = packet_get_tlv_value_bool( packet, TLV_TYPE_EXT_WINDOW_ENUM_INCLUDEUNKNOWN );
dprintf( "Beginning window enumeration" );
dwResult = enumerate_windows( response );
dwResult = enumerate_windows( response, bIncludeUnknown, parentWindow );
} while(0);