gecko-dev/ipc/mscom/ActivationContext.cpp

259 строки
6.1 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 http://mozilla.org/MPL/2.0/. */
#include "mozilla/mscom/ActivationContext.h"
#include "mozilla/Assertions.h"
#include "mozilla/DebugOnly.h"
#include "mozilla/mscom/Utils.h"
namespace mozilla {
namespace mscom {
ActivationContext::ActivationContext(WORD aResourceId)
: ActivationContext(reinterpret_cast<HMODULE>(GetContainingModuleHandle()),
aResourceId)
{
}
ActivationContext::ActivationContext(HMODULE aLoadFromModule, WORD aResourceId)
: mActCtx(INVALID_HANDLE_VALUE)
{
ACTCTX actCtx = {sizeof(actCtx)};
actCtx.dwFlags = ACTCTX_FLAG_RESOURCE_NAME_VALID | ACTCTX_FLAG_HMODULE_VALID;
actCtx.lpResourceName = MAKEINTRESOURCE(aResourceId);
actCtx.hModule = aLoadFromModule;
Init(actCtx);
}
void
ActivationContext::Init(ACTCTX& aActCtx)
{
MOZ_ASSERT(mActCtx == INVALID_HANDLE_VALUE);
mActCtx = ::CreateActCtx(&aActCtx);
MOZ_ASSERT(mActCtx != INVALID_HANDLE_VALUE);
}
void
ActivationContext::AddRef()
{
if (mActCtx == INVALID_HANDLE_VALUE) {
return;
}
::AddRefActCtx(mActCtx);
}
ActivationContext::ActivationContext(ActivationContext&& aOther)
: mActCtx(aOther.mActCtx)
{
aOther.mActCtx = INVALID_HANDLE_VALUE;
}
ActivationContext&
ActivationContext::operator=(ActivationContext&& aOther)
{
Release();
mActCtx = aOther.mActCtx;
aOther.mActCtx = INVALID_HANDLE_VALUE;
return *this;
}
ActivationContext::ActivationContext(const ActivationContext& aOther)
: mActCtx(aOther.mActCtx)
{
AddRef();
}
ActivationContext&
ActivationContext::operator=(const ActivationContext& aOther)
{
Release();
mActCtx = aOther.mActCtx;
AddRef();
return *this;
}
void
ActivationContext::Release()
{
if (mActCtx == INVALID_HANDLE_VALUE) {
return;
}
::ReleaseActCtx(mActCtx);
mActCtx = INVALID_HANDLE_VALUE;
}
ActivationContext::~ActivationContext()
{
Release();
}
#if defined(MOZILLA_INTERNAL_API)
/* static */ Result<uintptr_t,HRESULT>
ActivationContext::GetCurrent()
{
HANDLE actCtx;
if (!::GetCurrentActCtx(&actCtx)) {
return Result<uintptr_t,HRESULT>(HRESULT_FROM_WIN32(::GetLastError()));
}
return reinterpret_cast<uintptr_t>(actCtx);
}
/* static */ HRESULT
ActivationContext::GetCurrentManifestPath(nsAString& aOutManifestPath)
{
aOutManifestPath.Truncate();
SIZE_T bytesNeeded;
BOOL ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr,
nullptr, ActivationContextDetailedInformation,
nullptr, 0, &bytesNeeded);
if (!ok) {
DWORD err = ::GetLastError();
if (err != ERROR_INSUFFICIENT_BUFFER) {
return HRESULT_FROM_WIN32(err);
}
}
auto ctxBuf = MakeUnique<BYTE[]>(bytesNeeded);
ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr, nullptr,
ActivationContextDetailedInformation, ctxBuf.get(),
bytesNeeded, nullptr);
if (!ok) {
return HRESULT_FROM_WIN32(::GetLastError());
}
auto ctxInfo =
reinterpret_cast<ACTIVATION_CONTEXT_DETAILED_INFORMATION*>(ctxBuf.get());
// assemblyIndex is 1-based, and we want the last index, so we can just copy
// ctxInfo->ulAssemblyCount directly.
DWORD assemblyIndex = ctxInfo->ulAssemblyCount;
ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr,
&assemblyIndex,
AssemblyDetailedInformationInActivationContext, nullptr,
0, &bytesNeeded);
if (!ok) {
DWORD err = ::GetLastError();
if (err != ERROR_INSUFFICIENT_BUFFER) {
return HRESULT_FROM_WIN32(err);
}
}
auto assemblyBuf = MakeUnique<BYTE[]>(bytesNeeded);
ok = ::QueryActCtxW(QUERY_ACTCTX_FLAG_USE_ACTIVE_ACTCTX, nullptr,
&assemblyIndex,
AssemblyDetailedInformationInActivationContext,
assemblyBuf.get(), bytesNeeded, &bytesNeeded);
if (!ok) {
return HRESULT_FROM_WIN32(::GetLastError());
}
auto assemblyInfo =
reinterpret_cast<ACTIVATION_CONTEXT_ASSEMBLY_DETAILED_INFORMATION*>(assemblyBuf.get());
aOutManifestPath = nsDependentString(assemblyInfo->lpAssemblyManifestPath,
(assemblyInfo->ulManifestPathLength + 1) / sizeof(wchar_t));
return S_OK;
}
#endif // defined(MOZILLA_INTERNAL_API)
ActivationContextRegion::ActivationContextRegion()
: mActCookie(0)
{
}
ActivationContextRegion::ActivationContextRegion(const ActivationContext& aActCtx)
: mActCtx(aActCtx)
, mActCookie(0)
{
Activate();
}
ActivationContextRegion&
ActivationContextRegion::operator=(const ActivationContext& aActCtx)
{
Deactivate();
mActCtx = aActCtx;
Activate();
return *this;
}
ActivationContextRegion::ActivationContextRegion(ActivationContext&& aActCtx)
: mActCtx(Move(aActCtx))
, mActCookie(0)
{
Activate();
}
ActivationContextRegion&
ActivationContextRegion::operator=(ActivationContext&& aActCtx)
{
Deactivate();
mActCtx = Move(aActCtx);
Activate();
return *this;
}
ActivationContextRegion::ActivationContextRegion(ActivationContextRegion&& aRgn)
: mActCtx(Move(aRgn.mActCtx))
, mActCookie(aRgn.mActCookie)
{
aRgn.mActCookie = 0;
}
ActivationContextRegion&
ActivationContextRegion::operator=(ActivationContextRegion&& aRgn)
{
Deactivate();
mActCtx = Move(aRgn.mActCtx);
mActCookie = aRgn.mActCookie;
aRgn.mActCookie = 0;
return *this;
}
void
ActivationContextRegion::Activate()
{
if (mActCtx.mActCtx == INVALID_HANDLE_VALUE) {
return;
}
BOOL activated = ::ActivateActCtx(mActCtx.mActCtx, &mActCookie);
MOZ_DIAGNOSTIC_ASSERT(activated);
}
bool
ActivationContextRegion::Deactivate()
{
if (!mActCookie) {
return true;
}
BOOL deactivated = ::DeactivateActCtx(0, mActCookie);
MOZ_DIAGNOSTIC_ASSERT(deactivated);
if (deactivated) {
mActCookie = 0;
}
return !!deactivated;
}
ActivationContextRegion::~ActivationContextRegion()
{
Deactivate();
}
} // namespace mscom
} // namespace mozilla