// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "sandbox/win/src/restricted_token.h" #include "sandbox/win/src/restricted_token_utils.h" #include "sandbox/win/tools/finder/finder.h" #include "sandbox/win/tools/finder/ntundoc.h" #define BUFFER_SIZE 0x800 #define CHECKPTR(x) if (!x) return ::GetLastError() // NT API NTQUERYDIRECTORYOBJECT NtQueryDirectoryObject; NTOPENDIRECTORYOBJECT NtOpenDirectoryObject; NTOPENEVENT NtOpenEvent; NTOPENJOBOBJECT NtOpenJobObject; NTOPENKEYEDEVENT NtOpenKeyedEvent; NTOPENMUTANT NtOpenMutant; NTOPENSECTION NtOpenSection; NTOPENSEMAPHORE NtOpenSemaphore; NTOPENSYMBOLICLINKOBJECT NtOpenSymbolicLinkObject; NTOPENTIMER NtOpenTimer; NTOPENFILE NtOpenFile; NTCLOSE NtClose; DWORD Finder::InitNT() { HMODULE ntdll_handle = ::LoadLibrary(L"ntdll.dll"); CHECKPTR(ntdll_handle); NtOpenSymbolicLinkObject = (NTOPENSYMBOLICLINKOBJECT) ::GetProcAddress( ntdll_handle, "NtOpenSymbolicLinkObject"); CHECKPTR(NtOpenSymbolicLinkObject); NtQueryDirectoryObject = (NTQUERYDIRECTORYOBJECT) ::GetProcAddress( ntdll_handle, "NtQueryDirectoryObject"); CHECKPTR(NtQueryDirectoryObject); NtOpenDirectoryObject = (NTOPENDIRECTORYOBJECT) ::GetProcAddress( ntdll_handle, "NtOpenDirectoryObject"); CHECKPTR(NtOpenDirectoryObject); NtOpenKeyedEvent = (NTOPENKEYEDEVENT) ::GetProcAddress( ntdll_handle, "NtOpenKeyedEvent"); CHECKPTR(NtOpenKeyedEvent); NtOpenJobObject = (NTOPENJOBOBJECT) ::GetProcAddress( ntdll_handle, "NtOpenJobObject"); CHECKPTR(NtOpenJobObject); NtOpenSemaphore = (NTOPENSEMAPHORE) ::GetProcAddress( ntdll_handle, "NtOpenSemaphore"); CHECKPTR(NtOpenSemaphore); NtOpenSection = (NTOPENSECTION) ::GetProcAddress( ntdll_handle, "NtOpenSection"); CHECKPTR(NtOpenSection); NtOpenMutant= (NTOPENMUTANT) ::GetProcAddress(ntdll_handle, "NtOpenMutant"); CHECKPTR(NtOpenMutant); NtOpenEvent = (NTOPENEVENT) ::GetProcAddress(ntdll_handle, "NtOpenEvent"); CHECKPTR(NtOpenEvent); NtOpenTimer = (NTOPENTIMER) ::GetProcAddress(ntdll_handle, "NtOpenTimer"); CHECKPTR(NtOpenTimer); NtOpenFile = (NTOPENFILE) ::GetProcAddress(ntdll_handle, "NtOpenFile"); CHECKPTR(NtOpenFile); NtClose = (NTCLOSE) ::GetProcAddress(ntdll_handle, "NtClose"); CHECKPTR(NtClose); return ERROR_SUCCESS; } DWORD Finder::ParseKernelObjects(ATL::CString path) { UNICODE_STRING unicode_str; unicode_str.Length = (USHORT)path.GetLength()*2; unicode_str.MaximumLength = (USHORT)path.GetLength()*2+2; unicode_str.Buffer = path.GetBuffer(); OBJECT_ATTRIBUTES path_attributes; InitializeObjectAttributes(&path_attributes, &unicode_str, 0, // No Attributes NULL, // No Root Directory NULL); // No Security Descriptor DWORD object_index = 0; DWORD data_written = 0; // TODO(nsylvain): Do not use BUFFER_SIZE. Try to get the size // dynamically. OBJDIR_INFORMATION *object_directory_info = (OBJDIR_INFORMATION*) ::HeapAlloc(GetProcessHeap(), 0, BUFFER_SIZE); HANDLE file_handle; NTSTATUS status_code = NtOpenDirectoryObject(&file_handle, DIRECTORY_QUERY, &path_attributes); if (status_code != 0) return ERROR_UNIDENTIFIED_ERROR; status_code = NtQueryDirectoryObject(file_handle, object_directory_info, BUFFER_SIZE, TRUE, // Get Next Index TRUE, // Ignore Input Index &object_index, &data_written); if (status_code != 0) return ERROR_UNIDENTIFIED_ERROR; while (NtQueryDirectoryObject(file_handle, object_directory_info, BUFFER_SIZE, TRUE, FALSE, &object_index, &data_written) == 0 ) { ATL::CString cur_path(object_directory_info->ObjectName.Buffer, object_directory_info->ObjectName.Length / sizeof(WCHAR)); ATL::CString cur_type(object_directory_info->ObjectTypeName.Buffer, object_directory_info->ObjectTypeName.Length / sizeof(WCHAR)); ATL::CString new_path; if (path == L"\\") { new_path = path + cur_path; } else { new_path = path + L"\\" + cur_path; } TestKernelObjectAccess(new_path, cur_type); // Call the function recursively for all subdirectories if (cur_type == L"Directory") { ParseKernelObjects(new_path); } } NtClose(file_handle); return ERROR_SUCCESS; } DWORD Finder::TestKernelObjectAccess(ATL::CString path, ATL::CString type) { Impersonater impersonate(token_handle_); kernel_object_stats_[PARSE]++; NTGENERICOPEN func = NULL; GetFunctionForType(type, &func); if (!func) { kernel_object_stats_[BROKEN]++; Output(OBJ_ERR, type + L" Unsupported", path); return ERROR_UNSUPPORTED_TYPE; } UNICODE_STRING unicode_str; unicode_str.Length = (USHORT)path.GetLength()*2; unicode_str.MaximumLength = (USHORT)path.GetLength()*2+2; unicode_str.Buffer = path.GetBuffer(); OBJECT_ATTRIBUTES path_attributes; InitializeObjectAttributes(&path_attributes, &unicode_str, 0, // No Attributes NULL, // No Root Directory NULL); // No Security Descriptor HANDLE handle; NTSTATUS status_code = 0; if (access_type_ & kTestForAll) { status_code = NtGenericOpen(GENERIC_ALL, &path_attributes, func, &handle); if (STATUS_SUCCESS == status_code) { kernel_object_stats_[ALL]++; Output(OBJ, L"R/W", path); NtClose(handle); return GENERIC_ALL; } else if (status_code != EXCEPTION_ACCESS_VIOLATION && status_code != STATUS_ACCESS_DENIED) { Output(OBJ_ERR, status_code, path); kernel_object_stats_[BROKEN]++; } } if (access_type_ & kTestForWrite) { status_code = NtGenericOpen(GENERIC_WRITE, &path_attributes, func, &handle); if (STATUS_SUCCESS == status_code) { kernel_object_stats_[WRITE]++; Output(OBJ, L"W", path); NtClose(handle); return GENERIC_WRITE; } else if (status_code != EXCEPTION_ACCESS_VIOLATION && status_code != STATUS_ACCESS_DENIED) { Output(OBJ_ERR, status_code, path); kernel_object_stats_[BROKEN]++; } } if (access_type_ & kTestForRead) { status_code = NtGenericOpen(GENERIC_READ, &path_attributes, func, &handle); if (STATUS_SUCCESS == status_code) { kernel_object_stats_[READ]++; Output(OBJ, L"R", path); NtClose(handle); return GENERIC_READ; } else if (status_code != EXCEPTION_ACCESS_VIOLATION && status_code != STATUS_ACCESS_DENIED) { Output(OBJ_ERR, status_code, path); kernel_object_stats_[BROKEN]++; } } return 0; } NTSTATUS Finder::NtGenericOpen(ACCESS_MASK desired_access, OBJECT_ATTRIBUTES *object_attributes, NTGENERICOPEN func_to_call, HANDLE *handle) { return func_to_call(handle, desired_access, object_attributes); } bool Finder::GetFunctionForType(ATL::CString type, NTGENERICOPEN * func_to_call) { NTGENERICOPEN func = NULL; if (type == L"Event") func = NtOpenEvent; else if (type == L"Job") func = NtOpenJobObject; else if (type == L"KeyedEvent") func = NtOpenKeyedEvent; else if (type == L"Mutant") func = NtOpenMutant; else if (type == L"Section") func = NtOpenSection; else if (type == L"Semaphore") func = NtOpenSemaphore; else if (type == L"Timer") func = NtOpenTimer; else if (type == L"SymbolicLink") func = NtOpenSymbolicLinkObject; else if (type == L"Directory") func = NtOpenDirectoryObject; if (func) { *func_to_call = func; return true; } return false; }