forked from mirrors/gecko-dev
		
	 916fbe8459
			
		
	
	
		916fbe8459
		
	
	
	
	
		
			
			Since socket process can be launched before `GeckoDependentInitialize`, we cache the install directory in `InitSignedPolicyRulesToBypassCig` instead of using `sBinDir`. This patch moves `GetInstallDirectory` to WinHeaderOnlyUtils.h to reuse and removes `WinUtils::GetModuleFullPath` that is unused. Differential Revision: https://phabricator.services.mozilla.com/D133596
		
			
				
	
	
		
			837 lines
		
	
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			837 lines
		
	
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| #ifndef mozilla_WinHeaderOnlyUtils_h
 | |
| #define mozilla_WinHeaderOnlyUtils_h
 | |
| 
 | |
| #include <windows.h>
 | |
| #include <winerror.h>
 | |
| #include <winnt.h>
 | |
| #include <winternl.h>
 | |
| #include <objbase.h>
 | |
| #include <shlwapi.h>
 | |
| #undef ParseURL
 | |
| #include <stdlib.h>
 | |
| #include <tuple>
 | |
| 
 | |
| #include "mozilla/Assertions.h"
 | |
| #include "mozilla/Attributes.h"
 | |
| #include "mozilla/DynamicallyLinkedFunctionPtr.h"
 | |
| #include "mozilla/Maybe.h"
 | |
| #include "mozilla/ResultVariant.h"
 | |
| #include "mozilla/UniquePtr.h"
 | |
| #include "mozilla/WindowsVersion.h"
 | |
| #include "nsWindowsHelpers.h"
 | |
| 
 | |
| #if defined(MOZILLA_INTERNAL_API)
 | |
| #  include "nsIFile.h"
 | |
| #  include "nsString.h"
 | |
| #endif  // defined(MOZILLA_INTERNAL_API)
 | |
| 
 | |
| /**
 | |
|  * This header is intended for self-contained, header-only, utility code for
 | |
|  * Win32. It may be used outside of xul.dll, in places such as firefox.exe or
 | |
|  * mozglue.dll. If your code creates dependencies on Mozilla libraries, you
 | |
|  * should put it elsewhere.
 | |
|  */
 | |
| 
 | |
| #if _WIN32_WINNT < _WIN32_WINNT_WIN8
 | |
| typedef struct _FILE_ID_INFO {
 | |
|   ULONGLONG VolumeSerialNumber;
 | |
|   FILE_ID_128 FileId;
 | |
| } FILE_ID_INFO;
 | |
| 
 | |
| #  define FileIdInfo ((FILE_INFO_BY_HANDLE_CLASS)18)
 | |
| 
 | |
| #endif  // _WIN32_WINNT < _WIN32_WINNT_WIN8
 | |
| 
 | |
| #if !defined(STATUS_SUCCESS)
 | |
| #  define STATUS_SUCCESS ((NTSTATUS)0x00000000L)
 | |
| #endif  // !defined(STATUS_SUCCESS)
 | |
| 
 | |
| // Our data indicates a few users of Win7 x86 hit failure to load urlmon.dll
 | |
| // for unknown reasons.  Since we don't always require urlmon.dll on Win7,
 | |
| // we delay-load it, which causes a crash if loading urlmon.dll fails.  This
 | |
| // macro is to safely load and call urlmon's API graciously without crash.
 | |
| #if defined(_X86_)
 | |
| #  define SAFECALL_URLMON_FUNC(FuncName, ...)                                  \
 | |
|     do {                                                                       \
 | |
|       static const mozilla::StaticDynamicallyLinkedFunctionPtr<                \
 | |
|           decltype(&::FuncName)>                                               \
 | |
|           func(L"urlmon.dll", #FuncName);                                      \
 | |
|       hr =                                                                     \
 | |
|           func ? func(__VA_ARGS__) : HRESULT_FROM_WIN32(ERROR_PROC_NOT_FOUND); \
 | |
|     } while (0)
 | |
| #else
 | |
| #  define SAFECALL_URLMON_FUNC(FuncName, ...) hr = ::FuncName(__VA_ARGS__)
 | |
| #endif
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| class WindowsError final {
 | |
|  private:
 | |
|   // HRESULT and NTSTATUS are both typedefs of LONG, so we cannot use
 | |
|   // overloading to properly differentiate between the two. Instead we'll use
 | |
|   // static functions to convert the various error types to HRESULTs before
 | |
|   // instantiating.
 | |
|   explicit constexpr WindowsError(HRESULT aHResult) : mHResult(aHResult) {}
 | |
| 
 | |
|  public:
 | |
|   using UniqueString = UniquePtr<WCHAR[], LocalFreeDeleter>;
 | |
| 
 | |
|   static constexpr WindowsError FromNtStatus(NTSTATUS aNtStatus) {
 | |
|     if (aNtStatus == STATUS_SUCCESS) {
 | |
|       // Special case: we don't want to set FACILITY_NT_BIT
 | |
|       // (HRESULT_FROM_NT does not handle this case, unlike HRESULT_FROM_WIN32)
 | |
|       return WindowsError(S_OK);
 | |
|     }
 | |
| 
 | |
|     return WindowsError(HRESULT_FROM_NT(aNtStatus));
 | |
|   }
 | |
| 
 | |
|   static constexpr WindowsError FromHResult(HRESULT aHResult) {
 | |
|     return WindowsError(aHResult);
 | |
|   }
 | |
| 
 | |
|   static constexpr WindowsError FromWin32Error(DWORD aWin32Err) {
 | |
|     return WindowsError(HRESULT_FROM_WIN32(aWin32Err));
 | |
|   }
 | |
| 
 | |
|   static WindowsError FromLastError() {
 | |
|     return FromWin32Error(::GetLastError());
 | |
|   }
 | |
| 
 | |
|   static WindowsError CreateSuccess() { return WindowsError(S_OK); }
 | |
| 
 | |
|   static WindowsError CreateGeneric() {
 | |
|     return FromWin32Error(ERROR_UNIDENTIFIED_ERROR);
 | |
|   }
 | |
| 
 | |
|   bool IsSuccess() const { return SUCCEEDED(mHResult); }
 | |
| 
 | |
|   bool IsFailure() const { return FAILED(mHResult); }
 | |
| 
 | |
|   bool IsAvailableAsWin32Error() const {
 | |
|     return IsAvailableAsNtStatus() ||
 | |
|            HRESULT_FACILITY(mHResult) == FACILITY_WIN32;
 | |
|   }
 | |
| 
 | |
|   bool IsAvailableAsNtStatus() const {
 | |
|     return mHResult == S_OK || (mHResult & FACILITY_NT_BIT);
 | |
|   }
 | |
| 
 | |
|   bool IsAvailableAsHResult() const { return true; }
 | |
| 
 | |
|   UniqueString AsString() const {
 | |
|     LPWSTR rawMsgBuf = nullptr;
 | |
|     constexpr DWORD flags = FORMAT_MESSAGE_ALLOCATE_BUFFER |
 | |
|                             FORMAT_MESSAGE_FROM_SYSTEM |
 | |
|                             FORMAT_MESSAGE_IGNORE_INSERTS;
 | |
|     DWORD result =
 | |
|         ::FormatMessageW(flags, nullptr, mHResult, 0,
 | |
|                          reinterpret_cast<LPWSTR>(&rawMsgBuf), 0, nullptr);
 | |
|     if (!result) {
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     return UniqueString(rawMsgBuf);
 | |
|   }
 | |
| 
 | |
|   HRESULT AsHResult() const { return mHResult; }
 | |
| 
 | |
|   // Not all HRESULTs are convertible to Win32 Errors, so we use Maybe
 | |
|   Maybe<DWORD> AsWin32Error() const {
 | |
|     if (mHResult == S_OK) {
 | |
|       return Some(static_cast<DWORD>(ERROR_SUCCESS));
 | |
|     }
 | |
| 
 | |
|     if (HRESULT_FACILITY(mHResult) == FACILITY_WIN32) {
 | |
|       // This is the inverse of HRESULT_FROM_WIN32
 | |
|       return Some(static_cast<DWORD>(HRESULT_CODE(mHResult)));
 | |
|     }
 | |
| 
 | |
|     // The NTSTATUS facility is a special case and thus does not utilize the
 | |
|     // HRESULT_FACILITY and HRESULT_CODE macros.
 | |
|     if (mHResult & FACILITY_NT_BIT) {
 | |
|       return Some(NtStatusToWin32Error(
 | |
|           static_cast<NTSTATUS>(mHResult & ~FACILITY_NT_BIT)));
 | |
|     }
 | |
| 
 | |
|     return Nothing();
 | |
|   }
 | |
| 
 | |
|   // Not all HRESULTs are convertible to NTSTATUS, so we use Maybe
 | |
|   Maybe<NTSTATUS> AsNtStatus() const {
 | |
|     if (mHResult == S_OK) {
 | |
|       return Some(STATUS_SUCCESS);
 | |
|     }
 | |
| 
 | |
|     // The NTSTATUS facility is a special case and thus does not utilize the
 | |
|     // HRESULT_FACILITY and HRESULT_CODE macros.
 | |
|     if (mHResult & FACILITY_NT_BIT) {
 | |
|       return Some(static_cast<NTSTATUS>(mHResult & ~FACILITY_NT_BIT));
 | |
|     }
 | |
| 
 | |
|     return Nothing();
 | |
|   }
 | |
| 
 | |
|   constexpr bool operator==(const WindowsError& aOther) const {
 | |
|     return mHResult == aOther.mHResult;
 | |
|   }
 | |
| 
 | |
|   constexpr bool operator!=(const WindowsError& aOther) const {
 | |
|     return mHResult != aOther.mHResult;
 | |
|   }
 | |
| 
 | |
|   static DWORD NtStatusToWin32Error(NTSTATUS aNtStatus) {
 | |
|     static const StaticDynamicallyLinkedFunctionPtr<
 | |
|         decltype(&RtlNtStatusToDosError)>
 | |
|         pRtlNtStatusToDosError(L"ntdll.dll", "RtlNtStatusToDosError");
 | |
| 
 | |
|     MOZ_ASSERT(!!pRtlNtStatusToDosError);
 | |
|     if (!pRtlNtStatusToDosError) {
 | |
|       return ERROR_UNIDENTIFIED_ERROR;
 | |
|     }
 | |
| 
 | |
|     return pRtlNtStatusToDosError(aNtStatus);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   // We store the error code as an HRESULT because they can encode both Win32
 | |
|   // error codes and NTSTATUS codes.
 | |
|   HRESULT mHResult;
 | |
| };
 | |
| 
 | |
| namespace detail {
 | |
| template <>
 | |
| struct UnusedZero<WindowsError> {
 | |
|   using StorageType = WindowsError;
 | |
| 
 | |
|   static constexpr bool value = true;
 | |
|   static constexpr StorageType nullValue = WindowsError::FromHResult(S_OK);
 | |
| 
 | |
|   static constexpr void AssertValid(StorageType aValue) {}
 | |
|   static constexpr const WindowsError& Inspect(const StorageType& aValue) {
 | |
|     return aValue;
 | |
|   }
 | |
|   static constexpr WindowsError Unwrap(StorageType aValue) { return aValue; }
 | |
|   static constexpr StorageType Store(WindowsError aValue) { return aValue; }
 | |
| };
 | |
| }  // namespace detail
 | |
| 
 | |
| enum DetourResultCode : uint32_t {
 | |
|   RESULT_OK = 0,
 | |
|   INTERCEPTOR_MOD_NULL,
 | |
|   INTERCEPTOR_MOD_INACCESSIBLE,
 | |
|   INTERCEPTOR_PROC_NULL,
 | |
|   INTERCEPTOR_PROC_INACCESSIBLE,
 | |
|   DETOUR_PATCHER_RESERVE_FOR_MODULE_PE_ERROR,
 | |
|   DETOUR_PATCHER_RESERVE_FOR_MODULE_TEXT_ERROR,
 | |
|   DETOUR_PATCHER_RESERVE_FOR_MODULE_RESERVE_ERROR,
 | |
|   DETOUR_PATCHER_DO_RESERVE_ERROR,
 | |
|   DETOUR_PATCHER_NEXT_TRAMPOLINE_ERROR,
 | |
|   DETOUR_PATCHER_INVALID_TRAMPOLINE,
 | |
|   DETOUR_PATCHER_WRITE_POINTER_ERROR,
 | |
|   DETOUR_PATCHER_CREATE_TRAMPOLINE_ERROR,
 | |
|   FUNCHOOKCROSSPROCESS_COPYSTUB_ERROR,
 | |
|   MMPOLICY_RESERVE_INVALIDARG,
 | |
|   MMPOLICY_RESERVE_ZERO_RESERVATIONSIZE,
 | |
|   MMPOLICY_RESERVE_CREATEFILEMAPPING,
 | |
|   MMPOLICY_RESERVE_MAPVIEWOFFILE,
 | |
|   MMPOLICY_RESERVE_NOBOUND_RESERVE_ERROR,
 | |
|   MMPOLICY_RESERVE_FINDREGION_INVALIDLEN,
 | |
|   MMPOLICY_RESERVE_FINDREGION_INVALIDRANGE,
 | |
|   MMPOLICY_RESERVE_FINDREGION_VIRTUALQUERY_ERROR,
 | |
|   MMPOLICY_RESERVE_FINDREGION_NO_FREE_REGION,
 | |
|   MMPOLICY_RESERVE_FINAL_RESERVE_ERROR,
 | |
| };
 | |
| 
 | |
| #if defined(NIGHTLY_BUILD)
 | |
| struct DetourError {
 | |
|   // We have a 16-bytes buffer, but only minimum bytes to detour per
 | |
|   // architecture are copied.  See CreateTrampoline in PatcherDetour.h.
 | |
|   DetourResultCode mErrorCode;
 | |
|   uint8_t mOrigBytes[16];
 | |
|   explicit DetourError(DetourResultCode aError)
 | |
|       : mErrorCode(aError), mOrigBytes{} {}
 | |
|   DetourError(DetourResultCode aError, DWORD aWin32Error)
 | |
|       : mErrorCode(aError), mOrigBytes{} {
 | |
|     static_assert(sizeof(mOrigBytes) >= sizeof(aWin32Error),
 | |
|                   "Can't fit a DWORD in mOrigBytes");
 | |
|     *reinterpret_cast<DWORD*>(mOrigBytes) = aWin32Error;
 | |
|   }
 | |
|   operator WindowsError() const {
 | |
|     return WindowsError::FromHResult(mErrorCode);
 | |
|   }
 | |
| };
 | |
| #endif  // defined(NIGHTLY_BUILD)
 | |
| 
 | |
| template <typename T>
 | |
| using WindowsErrorResult = Result<T, WindowsError>;
 | |
| 
 | |
| struct LauncherError {
 | |
|   LauncherError(const char* aFile, int aLine, WindowsError aWin32Error)
 | |
|       : mFile(aFile), mLine(aLine), mError(aWin32Error) {}
 | |
| 
 | |
| #if defined(NIGHTLY_BUILD)
 | |
|   LauncherError(const char* aFile, int aLine,
 | |
|                 const Maybe<DetourError>& aDetourError)
 | |
|       : mFile(aFile),
 | |
|         mLine(aLine),
 | |
|         mError(aDetourError.isSome() ? aDetourError.value()
 | |
|                                      : WindowsError::CreateGeneric()),
 | |
|         mDetourError(aDetourError) {}
 | |
| #endif  // defined(NIGHTLY_BUILD)
 | |
| 
 | |
|   const char* mFile;
 | |
|   int mLine;
 | |
|   WindowsError mError;
 | |
| #if defined(NIGHTLY_BUILD)
 | |
|   Maybe<DetourError> mDetourError;
 | |
| #endif  // defined(NIGHTLY_BUILD)
 | |
| 
 | |
|   bool operator==(const LauncherError& aOther) const {
 | |
|     return mError == aOther.mError;
 | |
|   }
 | |
| 
 | |
|   bool operator!=(const LauncherError& aOther) const {
 | |
|     return mError != aOther.mError;
 | |
|   }
 | |
| 
 | |
|   bool operator==(const WindowsError& aOther) const { return mError == aOther; }
 | |
| 
 | |
|   bool operator!=(const WindowsError& aOther) const { return mError != aOther; }
 | |
| };
 | |
| 
 | |
| #if defined(MOZ_USE_LAUNCHER_ERROR)
 | |
| 
 | |
| template <typename T>
 | |
| using LauncherResult = Result<T, LauncherError>;
 | |
| 
 | |
| template <typename T>
 | |
| using LauncherResultWithLineInfo = LauncherResult<T>;
 | |
| 
 | |
| using WindowsErrorType = LauncherError;
 | |
| 
 | |
| #else
 | |
| 
 | |
| template <typename T>
 | |
| using LauncherResult = WindowsErrorResult<T>;
 | |
| 
 | |
| template <typename T>
 | |
| using LauncherResultWithLineInfo = Result<T, LauncherError>;
 | |
| 
 | |
| using WindowsErrorType = WindowsError;
 | |
| 
 | |
| #endif  // defined(MOZ_USE_LAUNCHER_ERROR)
 | |
| 
 | |
| using LauncherVoidResult = LauncherResult<Ok>;
 | |
| 
 | |
| using LauncherVoidResultWithLineInfo = LauncherResultWithLineInfo<Ok>;
 | |
| 
 | |
| #if defined(MOZ_USE_LAUNCHER_ERROR)
 | |
| 
 | |
| #  define LAUNCHER_ERROR_GENERIC()           \
 | |
|     ::mozilla::Err(::mozilla::LauncherError( \
 | |
|         __FILE__, __LINE__, ::mozilla::WindowsError::CreateGeneric()))
 | |
| 
 | |
| #  if defined(NIGHTLY_BUILD)
 | |
| #    define LAUNCHER_ERROR_FROM_DETOUR_ERROR(err) \
 | |
|       ::mozilla::Err(::mozilla::LauncherError(__FILE__, __LINE__, err))
 | |
| #  else
 | |
| #    define LAUNCHER_ERROR_FROM_DETOUR_ERROR(err) LAUNCHER_ERROR_GENERIC()
 | |
| #  endif  // defined(NIGHTLY_BUILD)
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_WIN32(err)     \
 | |
|     ::mozilla::Err(::mozilla::LauncherError( \
 | |
|         __FILE__, __LINE__, ::mozilla::WindowsError::FromWin32Error(err)))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_LAST()         \
 | |
|     ::mozilla::Err(::mozilla::LauncherError( \
 | |
|         __FILE__, __LINE__, ::mozilla::WindowsError::FromLastError()))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_NTSTATUS(ntstatus) \
 | |
|     ::mozilla::Err(::mozilla::LauncherError(     \
 | |
|         __FILE__, __LINE__, ::mozilla::WindowsError::FromNtStatus(ntstatus)))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_HRESULT(hresult) \
 | |
|     ::mozilla::Err(::mozilla::LauncherError(   \
 | |
|         __FILE__, __LINE__, ::mozilla::WindowsError::FromHResult(hresult)))
 | |
| 
 | |
| // This macro wraps the supplied WindowsError with a LauncherError
 | |
| #  define LAUNCHER_ERROR_FROM_MOZ_WINDOWS_ERROR(err) \
 | |
|     ::mozilla::Err(::mozilla::LauncherError(__FILE__, __LINE__, err))
 | |
| 
 | |
| #else
 | |
| 
 | |
| #  define LAUNCHER_ERROR_GENERIC() \
 | |
|     ::mozilla::Err(::mozilla::WindowsError::CreateGeneric())
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_DETOUR_ERROR(err) LAUNCHER_ERROR_GENERIC()
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_WIN32(err) \
 | |
|     ::mozilla::Err(::mozilla::WindowsError::FromWin32Error(err))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_LAST() \
 | |
|     ::mozilla::Err(::mozilla::WindowsError::FromLastError())
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_NTSTATUS(ntstatus) \
 | |
|     ::mozilla::Err(::mozilla::WindowsError::FromNtStatus(ntstatus))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_HRESULT(hresult) \
 | |
|     ::mozilla::Err(::mozilla::WindowsError::FromHResult(hresult))
 | |
| 
 | |
| #  define LAUNCHER_ERROR_FROM_MOZ_WINDOWS_ERROR(err) ::mozilla::Err(err)
 | |
| 
 | |
| #endif  // defined(MOZ_USE_LAUNCHER_ERROR)
 | |
| 
 | |
| // How long to wait for a created process to become available for input,
 | |
| // to prevent that process's windows being forced to the background.
 | |
| // This is used across update, restart, and the launcher.
 | |
| const DWORD kWaitForInputIdleTimeoutMS = 10 * 1000;
 | |
| 
 | |
| /**
 | |
|  * Wait for a child GUI process to become "idle." Idle means that the process
 | |
|  * has created its message queue and has begun waiting for user input.
 | |
|  *
 | |
|  * Note that this must only be used when the child process is going to display
 | |
|  * GUI! Otherwise you're going to be waiting for a very long time ;-)
 | |
|  *
 | |
|  * @return true if we successfully waited for input idle;
 | |
|  *         false if we timed out or failed to wait.
 | |
|  */
 | |
| inline bool WaitForInputIdle(HANDLE aProcess,
 | |
|                              DWORD aTimeoutMs = kWaitForInputIdleTimeoutMS) {
 | |
|   const DWORD kSleepTimeMs = 10;
 | |
|   const DWORD waitStart = aTimeoutMs == INFINITE ? 0 : ::GetTickCount();
 | |
|   DWORD elapsed = 0;
 | |
| 
 | |
|   while (true) {
 | |
|     if (aTimeoutMs != INFINITE) {
 | |
|       elapsed = ::GetTickCount() - waitStart;
 | |
|     }
 | |
| 
 | |
|     if (elapsed >= aTimeoutMs) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     // ::WaitForInputIdle() doesn't always set the last-error code on failure
 | |
|     ::SetLastError(ERROR_SUCCESS);
 | |
| 
 | |
|     DWORD waitResult = ::WaitForInputIdle(aProcess, aTimeoutMs - elapsed);
 | |
|     if (!waitResult) {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     if (waitResult == WAIT_FAILED &&
 | |
|         ::GetLastError() == ERROR_NOT_GUI_PROCESS) {
 | |
|       ::Sleep(kSleepTimeMs);
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| enum class PathType {
 | |
|   eNtPath,
 | |
|   eDosPath,
 | |
| };
 | |
| 
 | |
| class FileUniqueId final {
 | |
|  public:
 | |
|   explicit FileUniqueId(const wchar_t* aPath, PathType aPathType)
 | |
|       : mId(FILE_ID_INFO()) {
 | |
|     if (!aPath) {
 | |
|       mId = LAUNCHER_ERROR_FROM_HRESULT(E_INVALIDARG);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     nsAutoHandle file;
 | |
| 
 | |
|     switch (aPathType) {
 | |
|       default:
 | |
|         mId = LAUNCHER_ERROR_FROM_HRESULT(E_INVALIDARG);
 | |
|         MOZ_ASSERT_UNREACHABLE("Unhandled PathType");
 | |
|         return;
 | |
| 
 | |
|       case PathType::eNtPath: {
 | |
|         UNICODE_STRING unicodeString;
 | |
|         ::RtlInitUnicodeString(&unicodeString, aPath);
 | |
|         OBJECT_ATTRIBUTES objectAttributes;
 | |
|         InitializeObjectAttributes(&objectAttributes, &unicodeString,
 | |
|                                    OBJ_CASE_INSENSITIVE, nullptr, nullptr);
 | |
|         IO_STATUS_BLOCK ioStatus = {};
 | |
|         HANDLE ntHandle;
 | |
|         NTSTATUS status = ::NtOpenFile(
 | |
|             &ntHandle, SYNCHRONIZE | FILE_READ_ATTRIBUTES, &objectAttributes,
 | |
|             &ioStatus, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
 | |
|             FILE_SYNCHRONOUS_IO_NONALERT | FILE_OPEN_FOR_BACKUP_INTENT);
 | |
|         // We don't need to check |ntHandle| for INVALID_HANDLE_VALUE here,
 | |
|         // as that value is set by the Win32 layer.
 | |
|         if (!NT_SUCCESS(status)) {
 | |
|           mId = LAUNCHER_ERROR_FROM_NTSTATUS(status);
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         file.own(ntHandle);
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       case PathType::eDosPath: {
 | |
|         file.own(::CreateFileW(
 | |
|             aPath, 0, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
 | |
|             nullptr, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, nullptr));
 | |
|         if (file == INVALID_HANDLE_VALUE) {
 | |
|           mId = LAUNCHER_ERROR_FROM_LAST();
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     GetId(file);
 | |
|   }
 | |
| 
 | |
|   explicit FileUniqueId(const nsAutoHandle& aFile) : mId(FILE_ID_INFO()) {
 | |
|     GetId(aFile);
 | |
|   }
 | |
| 
 | |
|   ~FileUniqueId() = default;
 | |
| 
 | |
|   bool IsError() const { return mId.isErr(); }
 | |
| 
 | |
|   const WindowsErrorType& GetError() const { return mId.inspectErr(); }
 | |
| 
 | |
|   FileUniqueId(FileUniqueId&& aOther) = default;
 | |
|   FileUniqueId& operator=(FileUniqueId&& aOther) = delete;
 | |
| 
 | |
|   bool operator==(const FileUniqueId& aOther) const {
 | |
|     return mId.isOk() && aOther.mId.isOk() &&
 | |
|            !memcmp(&mId.inspect(), &aOther.mId.inspect(), sizeof(FILE_ID_INFO));
 | |
|   }
 | |
| 
 | |
|   bool operator!=(const FileUniqueId& aOther) const {
 | |
|     return !((*this) == aOther);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   void GetId(const nsAutoHandle& aFile) {
 | |
|     FILE_ID_INFO fileIdInfo = {};
 | |
|     if (IsWin8OrLater()) {
 | |
|       if (::GetFileInformationByHandleEx(aFile.get(), FileIdInfo, &fileIdInfo,
 | |
|                                          sizeof(fileIdInfo))) {
 | |
|         mId = fileIdInfo;
 | |
|         return;
 | |
|       }
 | |
|       // Only NTFS and ReFS support FileIdInfo. So we have to fallback if
 | |
|       // GetFileInformationByHandleEx failed.
 | |
|     }
 | |
| 
 | |
|     BY_HANDLE_FILE_INFORMATION info = {};
 | |
|     if (!::GetFileInformationByHandle(aFile.get(), &info)) {
 | |
|       mId = LAUNCHER_ERROR_FROM_LAST();
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     fileIdInfo.VolumeSerialNumber = info.dwVolumeSerialNumber;
 | |
|     memcpy(&fileIdInfo.FileId.Identifier[0], &info.nFileIndexLow,
 | |
|            sizeof(DWORD));
 | |
|     memcpy(&fileIdInfo.FileId.Identifier[sizeof(DWORD)], &info.nFileIndexHigh,
 | |
|            sizeof(DWORD));
 | |
|     mId = fileIdInfo;
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   LauncherResult<FILE_ID_INFO> mId;
 | |
| };
 | |
| 
 | |
| class MOZ_RAII AutoVirtualProtect final {
 | |
|  public:
 | |
|   AutoVirtualProtect(void* aAddress, size_t aLength, DWORD aProtFlags,
 | |
|                      HANDLE aTargetProcess = ::GetCurrentProcess())
 | |
|       : mAddress(aAddress),
 | |
|         mLength(aLength),
 | |
|         mTargetProcess(aTargetProcess),
 | |
|         mPrevProt(0),
 | |
|         mError(WindowsError::CreateSuccess()) {
 | |
|     if (!::VirtualProtectEx(aTargetProcess, aAddress, aLength, aProtFlags,
 | |
|                             &mPrevProt)) {
 | |
|       mError = WindowsError::FromLastError();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ~AutoVirtualProtect() {
 | |
|     if (mError.IsFailure()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     ::VirtualProtectEx(mTargetProcess, mAddress, mLength, mPrevProt,
 | |
|                        &mPrevProt);
 | |
|   }
 | |
| 
 | |
|   explicit operator bool() const { return mError.IsSuccess(); }
 | |
| 
 | |
|   WindowsError GetError() const { return mError; }
 | |
| 
 | |
|   DWORD PrevProt() const { return mPrevProt; }
 | |
| 
 | |
|   AutoVirtualProtect(const AutoVirtualProtect&) = delete;
 | |
|   AutoVirtualProtect(AutoVirtualProtect&&) = delete;
 | |
|   AutoVirtualProtect& operator=(const AutoVirtualProtect&) = delete;
 | |
|   AutoVirtualProtect& operator=(AutoVirtualProtect&&) = delete;
 | |
| 
 | |
|  private:
 | |
|   void* mAddress;
 | |
|   size_t mLength;
 | |
|   HANDLE mTargetProcess;
 | |
|   DWORD mPrevProt;
 | |
|   WindowsError mError;
 | |
| };
 | |
| 
 | |
| inline UniquePtr<wchar_t[]> GetFullModulePath(HMODULE aModule) {
 | |
|   DWORD bufLen = MAX_PATH;
 | |
|   mozilla::UniquePtr<wchar_t[]> buf;
 | |
|   DWORD retLen;
 | |
| 
 | |
|   while (true) {
 | |
|     buf = mozilla::MakeUnique<wchar_t[]>(bufLen);
 | |
|     retLen = ::GetModuleFileNameW(aModule, buf.get(), bufLen);
 | |
|     if (!retLen) {
 | |
|       return nullptr;
 | |
|     }
 | |
| 
 | |
|     if (retLen == bufLen && ::GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
 | |
|       bufLen *= 2;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   // Upon success, retLen *excludes* the null character
 | |
|   ++retLen;
 | |
| 
 | |
|   // Since we're likely to have a bunch of unused space in buf, let's
 | |
|   // reallocate a string to the actual size of the file name.
 | |
|   auto result = mozilla::MakeUnique<wchar_t[]>(retLen);
 | |
|   if (wcscpy_s(result.get(), retLen, buf.get())) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| inline UniquePtr<wchar_t[]> GetFullBinaryPath() {
 | |
|   return GetFullModulePath(nullptr);
 | |
| }
 | |
| 
 | |
| // Generates the install directory without a trailing path separator.
 | |
| inline bool GetInstallDirectory(UniquePtr<wchar_t[]>& installPath) {
 | |
|   installPath = GetFullBinaryPath();
 | |
|   // It's not safe to use PathRemoveFileSpecW with strings longer than MAX_PATH
 | |
|   // (including null terminator).
 | |
|   if (wcslen(installPath.get()) >= MAX_PATH) {
 | |
|     return false;
 | |
|   }
 | |
|   ::PathRemoveFileSpecW(installPath.get());
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| class ModuleVersion final {
 | |
|  public:
 | |
|   constexpr ModuleVersion() : mVersion(0ULL) {}
 | |
| 
 | |
|   explicit ModuleVersion(const VS_FIXEDFILEINFO& aFixedInfo)
 | |
|       : mVersion((static_cast<uint64_t>(aFixedInfo.dwFileVersionMS) << 32) |
 | |
|                  static_cast<uint64_t>(aFixedInfo.dwFileVersionLS)) {}
 | |
| 
 | |
|   explicit ModuleVersion(const uint64_t aVersion) : mVersion(aVersion) {}
 | |
| 
 | |
|   ModuleVersion(const ModuleVersion& aOther) : mVersion(aOther.mVersion) {}
 | |
| 
 | |
|   uint64_t AsInteger() const { return mVersion; }
 | |
| 
 | |
|   operator uint64_t() const { return AsInteger(); }
 | |
| 
 | |
|   std::tuple<uint16_t, uint16_t, uint16_t, uint16_t> AsTuple() const {
 | |
|     uint16_t major = static_cast<uint16_t>((mVersion >> 48) & 0xFFFFU);
 | |
|     uint16_t minor = static_cast<uint16_t>((mVersion >> 32) & 0xFFFFU);
 | |
|     uint16_t patch = static_cast<uint16_t>((mVersion >> 16) & 0xFFFFU);
 | |
|     uint16_t build = static_cast<uint16_t>(mVersion & 0xFFFFU);
 | |
| 
 | |
|     return {major, minor, patch, build};
 | |
|   }
 | |
| 
 | |
|   explicit operator bool() const { return !!mVersion; }
 | |
| 
 | |
|   bool operator<(const ModuleVersion& aOther) const {
 | |
|     return mVersion < aOther.mVersion;
 | |
|   }
 | |
| 
 | |
|   bool operator<(const uint64_t& aOther) const { return mVersion < aOther; }
 | |
| 
 | |
|   ModuleVersion& operator=(const uint64_t aIntVersion) {
 | |
|     mVersion = aIntVersion;
 | |
|     return *this;
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   uint64_t mVersion;
 | |
| };
 | |
| 
 | |
| inline LauncherResult<ModuleVersion> GetModuleVersion(
 | |
|     const wchar_t* aModuleFullPath) {
 | |
|   DWORD verInfoLen = ::GetFileVersionInfoSizeW(aModuleFullPath, nullptr);
 | |
|   if (!verInfoLen) {
 | |
|     return LAUNCHER_ERROR_FROM_LAST();
 | |
|   }
 | |
| 
 | |
|   auto verInfoBuf = MakeUnique<BYTE[]>(verInfoLen);
 | |
|   if (!::GetFileVersionInfoW(aModuleFullPath, 0, verInfoLen,
 | |
|                              verInfoBuf.get())) {
 | |
|     return LAUNCHER_ERROR_FROM_LAST();
 | |
|   }
 | |
| 
 | |
|   UINT fixedInfoLen;
 | |
|   VS_FIXEDFILEINFO* fixedInfo = nullptr;
 | |
|   if (!::VerQueryValueW(verInfoBuf.get(), L"\\",
 | |
|                         reinterpret_cast<LPVOID*>(&fixedInfo), &fixedInfoLen)) {
 | |
|     // VerQueryValue may fail if the resource does not exist. This is not an
 | |
|     // error; we'll return 0 in this case.
 | |
|     return ModuleVersion(0ULL);
 | |
|   }
 | |
| 
 | |
|   return ModuleVersion(*fixedInfo);
 | |
| }
 | |
| 
 | |
| inline LauncherResult<ModuleVersion> GetModuleVersion(HMODULE aModule) {
 | |
|   UniquePtr<wchar_t[]> fullPath(GetFullModulePath(aModule));
 | |
|   if (!fullPath) {
 | |
|     return LAUNCHER_ERROR_GENERIC();
 | |
|   }
 | |
| 
 | |
|   return GetModuleVersion(fullPath.get());
 | |
| }
 | |
| 
 | |
| #if defined(MOZILLA_INTERNAL_API)
 | |
| inline LauncherResult<ModuleVersion> GetModuleVersion(nsIFile* aFile) {
 | |
|   if (!aFile) {
 | |
|     return LAUNCHER_ERROR_FROM_HRESULT(E_INVALIDARG);
 | |
|   }
 | |
| 
 | |
|   nsAutoString fullPath;
 | |
|   nsresult rv = aFile->GetPath(fullPath);
 | |
|   if (NS_FAILED(rv)) {
 | |
|     return LAUNCHER_ERROR_GENERIC();
 | |
|   }
 | |
| 
 | |
|   return GetModuleVersion(fullPath.get());
 | |
| }
 | |
| #endif  // defined(MOZILLA_INTERNAL_API)
 | |
| 
 | |
| struct CoTaskMemFreeDeleter {
 | |
|   void operator()(void* aPtr) { ::CoTaskMemFree(aPtr); }
 | |
| };
 | |
| 
 | |
| inline LauncherResult<TOKEN_ELEVATION_TYPE> GetElevationType(
 | |
|     const nsAutoHandle& aToken) {
 | |
|   DWORD retLen;
 | |
|   TOKEN_ELEVATION_TYPE elevationType;
 | |
|   if (!::GetTokenInformation(aToken.get(), TokenElevationType, &elevationType,
 | |
|                              sizeof(elevationType), &retLen)) {
 | |
|     return LAUNCHER_ERROR_FROM_LAST();
 | |
|   }
 | |
| 
 | |
|   return elevationType;
 | |
| }
 | |
| 
 | |
| inline bool HasPackageIdentity() {
 | |
|   if (!IsWin8OrLater()) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   HMODULE kernel32Dll = ::GetModuleHandleW(L"kernel32");
 | |
|   if (!kernel32Dll) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   typedef LONG(WINAPI * GetCurrentPackageIdProc)(UINT32*, BYTE*);
 | |
|   GetCurrentPackageIdProc pGetCurrentPackageId =
 | |
|       (GetCurrentPackageIdProc)::GetProcAddress(kernel32Dll,
 | |
|                                                 "GetCurrentPackageId");
 | |
| 
 | |
|   // If there was any package identity to retrieve, we get
 | |
|   // ERROR_INSUFFICIENT_BUFFER. If there had been no package identity it
 | |
|   // would instead return APPMODEL_ERROR_NO_PACKAGE.
 | |
|   UINT32 packageNameSize = 0;
 | |
|   return pGetCurrentPackageId &&
 | |
|          (pGetCurrentPackageId(&packageNameSize, nullptr) ==
 | |
|           ERROR_INSUFFICIENT_BUFFER);
 | |
| }
 | |
| 
 | |
| inline UniquePtr<wchar_t[]> GetPackageFamilyName() {
 | |
|   HMODULE kernel32Dll = ::GetModuleHandleW(L"kernel32");
 | |
|   if (!kernel32Dll) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   typedef LONG(WINAPI * GetCurrentPackageFamilyNameProc)(UINT32*, PWSTR);
 | |
|   GetCurrentPackageFamilyNameProc pGetCurrentPackageFamilyName =
 | |
|       (GetCurrentPackageFamilyNameProc)::GetProcAddress(
 | |
|           kernel32Dll, "GetCurrentPackageFamilyName");
 | |
|   if (!pGetCurrentPackageFamilyName) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   UINT32 packageNameSize = 0;
 | |
|   if (pGetCurrentPackageFamilyName(&packageNameSize, nullptr) !=
 | |
|       ERROR_INSUFFICIENT_BUFFER) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   UniquePtr<wchar_t[]> packageIdentity = MakeUnique<wchar_t[]>(packageNameSize);
 | |
|   if (pGetCurrentPackageFamilyName(&packageNameSize, packageIdentity.get()) !=
 | |
|       ERROR_SUCCESS) {
 | |
|     return nullptr;
 | |
|   }
 | |
| 
 | |
|   return packageIdentity;
 | |
| }
 | |
| 
 | |
| // This implementation is equivalent to PathGetDriveNumber[AW].
 | |
| // We define our own version because using PathGetDriveNumber
 | |
| // delay-loads shlwapi.dll, which may fail when the process is
 | |
| // sandboxed.
 | |
| template <typename T>
 | |
| int MozPathGetDriveNumber(const T* aPath) {
 | |
|   const auto ToDriveNumber = [](const T* aPath) -> int {
 | |
|     if (*aPath == '\0' || *(aPath + 1) != ':') {
 | |
|       return -1;
 | |
|     }
 | |
| 
 | |
|     T c = *aPath;
 | |
|     return (c >= 'A' && c <= 'Z')   ? c - 'A'
 | |
|            : (c >= 'a' && c <= 'z') ? c - 'a'
 | |
|                                     : -1;
 | |
|   };
 | |
| 
 | |
|   if (!aPath) {
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   if (*aPath == '\\' && *(aPath + 1) == '\\' && *(aPath + 2) == '?' &&
 | |
|       *(aPath + 3) == '\\') {
 | |
|     return ToDriveNumber(aPath + 4);
 | |
|   }
 | |
| 
 | |
|   return ToDriveNumber(aPath);
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla
 | |
| 
 | |
| #endif  // mozilla_WinHeaderOnlyUtils_h
 |