398 lines
12 KiB
C++
398 lines
12 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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/MainThreadHandoff.h"
|
|
|
|
#include "mozilla/Move.h"
|
|
#include "mozilla/mscom/InterceptorLog.h"
|
|
#include "mozilla/mscom/Registration.h"
|
|
#include "mozilla/mscom/Utils.h"
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsProxyRelease.h"
|
|
|
|
using mozilla::DebugOnly;
|
|
|
|
namespace {
|
|
|
|
class HandoffRunnable : public mozilla::Runnable
|
|
{
|
|
public:
|
|
explicit HandoffRunnable(ICallFrame* aCallFrame, IUnknown* aTargetInterface)
|
|
: mCallFrame(aCallFrame)
|
|
, mTargetInterface(aTargetInterface)
|
|
, mResult(E_UNEXPECTED)
|
|
{
|
|
}
|
|
|
|
NS_IMETHOD Run() override
|
|
{
|
|
mResult = mCallFrame->Invoke(mTargetInterface);
|
|
return NS_OK;
|
|
}
|
|
|
|
HRESULT GetResult() const
|
|
{
|
|
return mResult;
|
|
}
|
|
|
|
private:
|
|
ICallFrame* mCallFrame;
|
|
IUnknown* mTargetInterface;
|
|
HRESULT mResult;
|
|
};
|
|
|
|
} // anonymous namespace
|
|
|
|
namespace mozilla {
|
|
namespace mscom {
|
|
|
|
/* static */ HRESULT
|
|
MainThreadHandoff::Create(IInterceptorSink** aOutput)
|
|
{
|
|
*aOutput = nullptr;
|
|
MainThreadHandoff* handoff = new MainThreadHandoff();
|
|
HRESULT hr = handoff->QueryInterface(IID_IInterceptorSink, (void**) aOutput);
|
|
handoff->Release();
|
|
return hr;
|
|
}
|
|
|
|
MainThreadHandoff::MainThreadHandoff()
|
|
: mRefCnt(1)
|
|
{
|
|
}
|
|
|
|
MainThreadHandoff::~MainThreadHandoff()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
}
|
|
|
|
HRESULT
|
|
MainThreadHandoff::QueryInterface(REFIID riid, void** ppv)
|
|
{
|
|
IUnknown* punk = nullptr;
|
|
if (!ppv) {
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (riid == IID_IUnknown || riid == IID_ICallFrameEvents ||
|
|
riid == IID_IInterceptorSink) {
|
|
punk = static_cast<IInterceptorSink*>(this);
|
|
} else if (riid == IID_ICallFrameWalker) {
|
|
punk = static_cast<ICallFrameWalker*>(this);
|
|
}
|
|
|
|
*ppv = punk;
|
|
if (!punk) {
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
punk->AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
ULONG
|
|
MainThreadHandoff::AddRef()
|
|
{
|
|
return (ULONG) InterlockedIncrement((LONG*)&mRefCnt);
|
|
}
|
|
|
|
ULONG
|
|
MainThreadHandoff::Release()
|
|
{
|
|
ULONG newRefCnt = (ULONG) InterlockedDecrement((LONG*)&mRefCnt);
|
|
if (newRefCnt == 0) {
|
|
// It is possible for the last Release() call to happen off-main-thread.
|
|
// If so, we need to dispatch an event to delete ourselves.
|
|
if (NS_IsMainThread()) {
|
|
delete this;
|
|
} else {
|
|
// We need to delete this object on the main thread, but we aren't on the
|
|
// main thread right now, so we send a reference to ourselves to the main
|
|
// thread to be re-released there.
|
|
RefPtr<MainThreadHandoff> self = this;
|
|
NS_ReleaseOnMainThread(self.forget());
|
|
}
|
|
}
|
|
return newRefCnt;
|
|
}
|
|
|
|
HRESULT
|
|
MainThreadHandoff::OnCall(ICallFrame* aFrame)
|
|
{
|
|
// (1) Get info about the method call
|
|
HRESULT hr;
|
|
IID iid;
|
|
ULONG method;
|
|
hr = aFrame->GetIIDAndMethod(&iid, &method);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
RefPtr<IInterceptor> interceptor;
|
|
hr = mInterceptor->Resolve(IID_IInterceptor,
|
|
(void**)getter_AddRefs(interceptor));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
InterceptorTargetPtr targetInterface;
|
|
hr = interceptor->GetTargetForIID(iid, targetInterface);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
// (2) Execute the method call syncrhonously on the main thread
|
|
RefPtr<HandoffRunnable> handoffInfo(new HandoffRunnable(aFrame,
|
|
targetInterface.get()));
|
|
MainThreadInvoker invoker;
|
|
if (!invoker.Invoke(do_AddRef(handoffInfo))) {
|
|
MOZ_ASSERT(false);
|
|
return E_UNEXPECTED;
|
|
}
|
|
hr = handoffInfo->GetResult();
|
|
MOZ_ASSERT(SUCCEEDED(hr));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
// (3) Log *before* wrapping outputs so that the log will contain pointers to
|
|
// the true target interface, not the wrapped ones.
|
|
InterceptorLog::Event(aFrame, targetInterface.get());
|
|
|
|
// (4) Scan the function call for outparams that contain interface pointers.
|
|
// Those will need to be wrapped with MainThreadHandoff so that they too will
|
|
// be exeuted on the main thread.
|
|
|
|
hr = aFrame->GetReturnValue();
|
|
if (FAILED(hr)) {
|
|
// If the call resulted in an error then there's not going to be anything
|
|
// that needs to be wrapped.
|
|
return S_OK;
|
|
}
|
|
|
|
// (5) Unfortunately ICallFrame::WalkFrame does not correctly handle array
|
|
// outparams. Instead, we find out whether anybody has called
|
|
// mscom::RegisterArrayData to supply array parameter information and use it
|
|
// if available. This is a terrible hack, but it works for the short term. In
|
|
// the longer term we want to be able to use COM proxy/stub metadata to
|
|
// resolve array information for us.
|
|
const ArrayData* arrayData = FindArrayData(iid, method);
|
|
if (arrayData) {
|
|
hr = FixArrayElements(aFrame, *arrayData);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
} else {
|
|
// (6) Scan the outputs looking for any outparam interfaces that need wrapping.
|
|
// NB: WalkFrame does not correctly handle array outparams. It processes the
|
|
// first element of an array but not the remaining elements (if any).
|
|
hr = aFrame->WalkFrame(CALLFRAME_WALK_OUT, this);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static PVOID
|
|
ResolveArrayPtr(VARIANT& aVariant)
|
|
{
|
|
if (!(aVariant.vt & VT_BYREF)) {
|
|
return nullptr;
|
|
}
|
|
return aVariant.byref;
|
|
}
|
|
|
|
static PVOID*
|
|
ResolveInterfacePtr(PVOID aArrayPtr, VARTYPE aVartype, LONG aIndex)
|
|
{
|
|
if (aVartype != (VT_VARIANT | VT_BYREF)) {
|
|
IUnknown** ifaceArray = reinterpret_cast<IUnknown**>(aArrayPtr);
|
|
return reinterpret_cast<PVOID*>(&ifaceArray[aIndex]);
|
|
}
|
|
VARIANT* variantArray = reinterpret_cast<VARIANT*>(aArrayPtr);
|
|
VARIANT& element = variantArray[aIndex];
|
|
return &element.byref;
|
|
}
|
|
|
|
HRESULT
|
|
MainThreadHandoff::FixArrayElements(ICallFrame* aFrame,
|
|
const ArrayData& aArrayData)
|
|
{
|
|
// Extract the array length
|
|
VARIANT paramVal;
|
|
VariantInit(¶mVal);
|
|
HRESULT hr = aFrame->GetParam(aArrayData.mLengthParamIndex, ¶mVal);
|
|
MOZ_ASSERT(SUCCEEDED(hr) &&
|
|
(paramVal.vt == (VT_I4 | VT_BYREF) ||
|
|
paramVal.vt == (VT_UI4 | VT_BYREF)));
|
|
if (FAILED(hr) || (paramVal.vt != (VT_I4 | VT_BYREF) &&
|
|
paramVal.vt != (VT_UI4 | VT_BYREF))) {
|
|
return hr;
|
|
}
|
|
|
|
const LONG arrayLength = *(paramVal.plVal);
|
|
if (!arrayLength) {
|
|
// Nothing to do
|
|
return S_OK;
|
|
}
|
|
|
|
// Extract the array parameter
|
|
VariantInit(¶mVal);
|
|
PVOID arrayPtr = nullptr;
|
|
hr = aFrame->GetParam(aArrayData.mArrayParamIndex, ¶mVal);
|
|
if (hr == DISP_E_BADVARTYPE) {
|
|
// ICallFrame::GetParam is not able to coerce the param into a VARIANT.
|
|
// That's ok, we can try to do it ourselves.
|
|
CALLFRAMEPARAMINFO paramInfo;
|
|
hr = aFrame->GetParamInfo(aArrayData.mArrayParamIndex, ¶mInfo);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
PVOID stackBase = aFrame->GetStackLocation();
|
|
if (aArrayData.mFlag == ArrayData::Flag::eAllocatedByServer) {
|
|
// In order for the server to allocate the array's buffer and store it in
|
|
// an outparam, the parameter must be typed as Type***. Since the base
|
|
// of the array is Type*, we must dereference twice.
|
|
arrayPtr = **reinterpret_cast<PVOID**>(reinterpret_cast<PBYTE>(stackBase) +
|
|
paramInfo.stackOffset);
|
|
} else {
|
|
// We dereference because we need to obtain the value of a parameter
|
|
// from a stack offset. This pointer is the base of the array.
|
|
arrayPtr = *reinterpret_cast<PVOID*>(reinterpret_cast<PBYTE>(stackBase) +
|
|
paramInfo.stackOffset);
|
|
}
|
|
} else if (FAILED(hr)) {
|
|
return hr;
|
|
} else {
|
|
arrayPtr = ResolveArrayPtr(paramVal);
|
|
}
|
|
|
|
MOZ_ASSERT(arrayPtr);
|
|
if (!arrayPtr) {
|
|
return DISP_E_BADVARTYPE;
|
|
}
|
|
|
|
// We walk the elements of the array and invoke OnWalkInterface to wrap each
|
|
// one, just as ICallFrame::WalkFrame would do.
|
|
for (LONG index = 0; index < arrayLength; ++index) {
|
|
hr = OnWalkInterface(aArrayData.mArrayParamIid,
|
|
ResolveInterfacePtr(arrayPtr, paramVal.vt, index),
|
|
FALSE, TRUE);
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT
|
|
MainThreadHandoff::SetInterceptor(IWeakReference* aInterceptor)
|
|
{
|
|
mInterceptor = aInterceptor;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT
|
|
MainThreadHandoff::OnWalkInterface(REFIID aIid, PVOID* aInterface,
|
|
BOOL aIsInParam, BOOL aIsOutParam)
|
|
{
|
|
MOZ_ASSERT(aInterface && aIsOutParam);
|
|
if (!aInterface || !aIsOutParam) {
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// Adopt aInterface for the time being. We can't touch its refcount off
|
|
// the main thread, so we'll use STAUniquePtr so that we can safely
|
|
// Release() it if necessary.
|
|
STAUniquePtr<IUnknown> origInterface(static_cast<IUnknown*>(*aInterface));
|
|
*aInterface = nullptr;
|
|
|
|
if (!origInterface) {
|
|
// Nothing to wrap.
|
|
return S_OK;
|
|
}
|
|
|
|
// First make sure that aInterface isn't a proxy - we don't want to wrap
|
|
// those.
|
|
if (IsProxy(origInterface.get())) {
|
|
*aInterface = origInterface.release();
|
|
return S_OK;
|
|
}
|
|
|
|
RefPtr<IInterceptor> interceptor;
|
|
HRESULT hr = mInterceptor->Resolve(IID_IInterceptor,
|
|
(void**) getter_AddRefs(interceptor));
|
|
MOZ_ASSERT(SUCCEEDED(hr));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
// Now make sure that origInterface isn't referring to the same IUnknown
|
|
// as an interface that we are already managing. We can determine this by
|
|
// querying (NOT casting!) both objects for IUnknown and then comparing the
|
|
// resulting pointers.
|
|
InterceptorTargetPtr existingTarget;
|
|
hr = interceptor->GetTargetForIID(aIid, existingTarget);
|
|
if (SUCCEEDED(hr)) {
|
|
bool areIUnknownsEqual = false;
|
|
|
|
// This check must be done on the main thread
|
|
auto checkFn = [&existingTarget, &origInterface, &areIUnknownsEqual]() -> void {
|
|
RefPtr<IUnknown> unkExisting;
|
|
HRESULT hrExisting =
|
|
existingTarget->QueryInterface(IID_IUnknown,
|
|
(void**)getter_AddRefs(unkExisting));
|
|
RefPtr<IUnknown> unkNew;
|
|
HRESULT hrNew =
|
|
origInterface->QueryInterface(IID_IUnknown,
|
|
(void**)getter_AddRefs(unkNew));
|
|
areIUnknownsEqual = SUCCEEDED(hrExisting) && SUCCEEDED(hrNew) &&
|
|
unkExisting == unkNew;
|
|
};
|
|
|
|
MainThreadInvoker invoker;
|
|
if (invoker.Invoke(NS_NewRunnableFunction(checkFn)) && areIUnknownsEqual) {
|
|
// The existing interface and the new interface both belong to the same
|
|
// target object. Let's just use the existing one.
|
|
void* intercepted = nullptr;
|
|
hr = interceptor->GetInterceptorForIID(aIid, &intercepted);
|
|
MOZ_ASSERT(SUCCEEDED(hr));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
*aInterface = intercepted;
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
// Now create a new MainThreadHandoff wrapper...
|
|
RefPtr<IInterceptorSink> handoff;
|
|
hr = MainThreadHandoff::Create(getter_AddRefs(handoff));
|
|
MOZ_ASSERT(SUCCEEDED(hr));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
RefPtr<IUnknown> wrapped;
|
|
hr = Interceptor::Create(Move(origInterface), handoff, aIid,
|
|
getter_AddRefs(wrapped));
|
|
MOZ_ASSERT(SUCCEEDED(hr));
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
// And replace the original interface pointer with the wrapped one.
|
|
wrapped.forget(reinterpret_cast<IUnknown**>(aInterface));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
} // namespace mscom
|
|
} // namespace mozilla
|