2020-09-30 17:12:29 +02:00

609 lines
18 KiB
C++

//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1994.
//
// File: inv16.cxx
//
// Contents: 32->16 Call thunking
//
// History: 25-Feb-94 DrewB Created
//
//----------------------------------------------------------------------------
#include "headers.cxx"
#pragma hdrstop
#include "..\..\ole232\inplace\inplace.h" // We need CFrameFilter for
// WinWord 6 Hack
//+---------------------------------------------------------------------------
//
// Function: InvokeOn16, public
//
// Synopsis: Sets up the THUNKINFO and starts thunking for a 32->16 call
//
// Arguments: [iidx] - Custom interface or known interface index
// [dwMethod] - Method index
// [pvStack32] - 32-bit stack
//
// Returns: Appropriate status code
//
// History: 25-Feb-94 DrewB Created
//
//----------------------------------------------------------------------------
#if DBG == 1
extern "C"
{
ULONG InvokeOn16_count = 0;
ULONG InvokeOn16_break = 0;
int _iInvokeOn16BreakIidx = -1;
int _iInvokeOn16BreakMethod = -1;
};
#endif
// InvokeOn16 uses a lot of local variables so allocate its locals
// rather than declaring them on the stack. This saves roughly
// 150 bytes of stack per call
struct INVOKE16RECORD
{
THOP CONST *pThop;
THOP CONST * CONST *ppThop;
UINT uiThop;
VTBLFN UNALIGNED CONST *pvfnVtbl;
VPVOID vpvThis16;
VPVOID vpvVtbl16;
VPVOID UNALIGNED CONST *pvpvThis16;
DWORD dwStack16[MAX_PARAMS];
THUNKINFO ti;
THUNK3216OBJ *ptoThis32;
ThreadData *ptd;
};
DWORD InvokeOn16(IIDIDX iidx, DWORD dwMethod, LPVOID pvStack32)
{
// NOTE: Do not declare local variables in this routine
// except for debug builds
INVOKE16RECORD *pir;
DWORD dwResult;
#if DBG == 1
ULONG ulInvokeOn16_count = ++InvokeOn16_count;
if (InvokeOn16_count == InvokeOn16_break)
{
DebugBreak();
}
thkDebugOut((DEB_ITRACE, "%sInvokeOn16(0x%x, 0x%x, %p)\n",
NestingLevelString(), iidx, dwMethod, pvStack32));
#endif
pir = (INVOKE16RECORD *)STACKALLOC32(sizeof(INVOKE16RECORD));
if (pir == NULL)
{
// BUGBUG - This error isn't guaranteed to mean anything for
// this call. Not much else we can do, though
return (DWORD)E_OUTOFMEMORY;
}
// pvStack32 is a pointer to an array of arguments from the
// 32-bit call. It's always laid out with the first
// argument low and increasing from there
pir->ti.s32.pbStart = (BYTE *)pvStack32;
pir->ti.s32.pbCurrent = pir->ti.s32.pbStart;
pir->ti.s16.pbStart = (BYTE *)pir->dwStack16;
pir->ti.s16.pbCurrent = pir->ti.s16.pbStart;
pir->ti.scResult = S_OK;
pir->ti.fResultThunked = FALSE;
pir->ptd = TlsThkGetData();
if (pir->ptd == NULL)
{
thkDebugOut((DEB_WARN, "WARNING: InvokeOn16: Call refused\n"));
STACKFREE32(pir, sizeof(INVOKE16RECORD));
return (DWORD)E_FAIL;
}
pir->ti.pThkMgr = pir->ptd->pCThkMgr;
thkAssert(pir->ti.pThkMgr != NULL);
thkAssert(iidx >= 0 && iidx < THI_COUNT);
// For each interface there is an array of thop strings, one for
// each method. The IUnknown methods don't have thop strings so
// bias the thop string pointer to account for that
thkAssert(dwMethod >= SMI_COUNT);
pir->ppThop = athopiInterfaceThopis[iidx].ppThops-SMI_COUNT;
pir->uiThop = athopiInterfaceThopis[iidx].uiSize;
// Methods are cdecl so we need to move upwards in memory to
// get to the next parameter
pir->ti.s16.iDir = 1;
// We need to look up the appropriate method pointer by
// looking in the 16-bit object's vtable
GET_STACK32(&pir->ti, pir->ptoThis32, THUNK3216OBJ *);
thkDebugOut((DEB_INVOKES,
"InvokeOn16: ptoThis32 = %08lX\n", pir->ptoThis32 ));
if ( pir->ptoThis32->grfFlags & PROXYFLAG_CLEANEDUP )
{
thkDebugOut((DEB_WARN,
"InvokeOn16: Attempt to call %s::%s"
"on cleaned-up proxy %08lX for 16-bit object %08lX\n",
inInterfaceNames[iidx].pszInterface,
inInterfaceNames[iidx].ppszMethodNames[dwMethod],
pir->ptoThis32, pir->ptoThis32->vpvThis16));
STACKFREE32(pir, sizeof(INVOKE16RECORD));
return (DWORD)E_FAIL;
}
// check PROXYFLAG_CLEANEDUP before calling DebugValidateProxy3216.
// Otherwise we might get asserts on checked OLE.
DebugValidateProxy3216(pir->ptoThis32);
pir->ti.dwCallerProxy = (DWORD)pir->ptoThis32;
pir->vpvThis16 = pir->ptoThis32->vpvThis16;
pir->pvpvThis16 = (VPVOID UNALIGNED *)
GetReadPtr16(&pir->ti, pir->vpvThis16, sizeof(VPVOID));
if (pir->pvpvThis16 == NULL)
{
dwResult = pir->ti.scResult;
STACKFREE32(pir, sizeof(INVOKE16RECORD));
return dwResult;
}
pir->vpvVtbl16 = *pir->pvpvThis16;
pir->pvfnVtbl = (VTBLFN UNALIGNED *)
GetReadPtr16(&pir->ti, pir->vpvVtbl16, sizeof(VPVOID)*pir->uiThop);
WOWRELVDMPTR(pir->vpvThis16);
if (pir->pvfnVtbl == NULL)
{
dwResult = pir->ti.scResult;
STACKFREE32(pir, sizeof(INVOKE16RECORD));
return dwResult;
}
// Push the 16-bit this pointer on the stack first
TO_STACK16(&pir->ti, pir->vpvThis16, VPVOID);
thkAssert(dwMethod < pir->uiThop);
pir->pThop = pir->ppThop[dwMethod];
thkAssert(pir->pThop != NULL);
pir->ti.pThop = pir->pThop;
pir->ti.pvfn = pir->pvfnVtbl[dwMethod];
pir->ti.iidx = iidx;
pir->ti.dwMethod = dwMethod;
pir->ti.this32 = (IUnknown *)pir->ptoThis32;
WOWRELVDMPTR(pir->vpvVtbl16);
thkDebugOut((DEB_INVOKES, "%s#(%04X):InvokeOn16 on %p:%p, %s::%s\n",
NestingLevelString(), ulInvokeOn16_count,
pir->vpvThis16, pir->ti.pvfn,
inInterfaceNames[iidx].pszInterface,
inInterfaceNames[iidx].ppszMethodNames[dwMethod]));
DebugIncrementNestingLevel();
pir->ti.pThkMgr->SetThkState(THKSTATE_INVOKETHKIN16);
#if DBG == 1
SStackRecord sr;
RecordStackState16(&sr);
#endif
#if DBG == 1
if ((_iInvokeOn16BreakIidx > 0 && _iInvokeOn16BreakIidx == (int)iidx) &&
(_iInvokeOn16BreakMethod < 0 ||
_iInvokeOn16BreakMethod == (int)dwMethod))
{
DebugBreak();
}
#endif
dwResult = EXECUTE_THOP3216(&pir->ti);
#if DBG == 1
if ( !pir->ti.fResultThunked && FAILED(dwResult) )
{
thkDebugOut((DEB_FAILURES,
"InvokeOn16 probable failure %s::%s sc = %08lX\n",
inInterfaceNames[iidx].pszInterface,
inInterfaceNames[iidx].ppszMethodNames[dwMethod],
dwResult));
}
CheckStackState16(&sr);
#endif
pir->ti.pThkMgr->SetThkState(THKSTATE_NOCALL);
DebugDecrementNestingLevel();
thkDebugOut((DEB_INVOKES,
"%s#(%04X):InvokeOn16 on %p:%p, %s::%s returns 0x%08lX\n",
NestingLevelString(), ulInvokeOn16_count,
pir->vpvThis16, pir->ti.pvfn,
inInterfaceNames[iidx].pszInterface,
inInterfaceNames[iidx].ppszMethodNames[dwMethod],
dwResult));
STACKFREE32(pir, sizeof(INVOKE16RECORD));
return dwResult;
}
//+---------------------------------------------------------------------------
//
// Function: Call3216, private
//
// Synopsis: Sets up stack and transitions to 16-bit
//
// Arguments: [pvfn] - Function to call
// [pbStack] - Stack in 32-bits
// [cbStack] - Size of stack
//
// Returns: Appropriate status code
//
// History: 04-Mar-94 DrewB Created
//
//----------------------------------------------------------------------------
#if DBG == 1
extern "C" ULONG Call3216_count = 0;
extern "C" ULONG Call3216_break = 0;
#endif
DWORD Call3216(VPVOID pvfn, BYTE *pbStack, UINT cbStack)
{
#if DBG == 1
ULONG ulCall3216_count = ++Call3216_count;
if (Call3216_count == Call3216_break)
{
DebugBreak();
}
#endif
VPVOID vpvStack16;
DWORD dwResult;
void *pvStack16;
dwResult = (DWORD)S_OK;
if (cbStack <= WCB16_MAX_CBARGS)
{
thkDebugOut((DEB_ITRACE, "CallbackTo16Ex(%p, %lu, %p) #(%x)\n",
pvfn, cbStack, pbStack, ulCall3216_count));
// pbStack must have at least WCB16_MAX_CBARGS bytes of valid memory
// since 16V always copies that many bytes
// In our case pbStack is from InvokeOn16 which should be large enough
thkAssert(MAX_PARAMS*sizeof(DWORD) >= WCB16_MAX_CBARGS);
if (!CallbackTo16Ex(pvfn, WCB16_CDECL, cbStack, pbStack,
&dwResult))
{
dwResult = (DWORD)E_UNEXPECTED;
}
}
else
{
CALLDATA UNALIGNED *pcd;
UINT cbAlloc;
cbAlloc = cbStack+sizeof(CALLDATA);
vpvStack16 = STACKALLOC16(cbAlloc);
if (vpvStack16 == 0)
{
dwResult = (DWORD)E_OUTOFMEMORY;
}
else
{
pvStack16 = (void *)WOWFIXVDMPTR(vpvStack16, cbAlloc);
pcd = (CALLDATA UNALIGNED *)((BYTE *)pvStack16+cbStack);
pcd->vpfn = (DWORD)pvfn;
pcd->vpvStack16 = vpvStack16;
pcd->cbStack = cbStack;
memcpy(pvStack16, pbStack, cbStack);
WOWRELVDMPTR(vpvStack16);
thkDebugOut((DEB_ITRACE, "CallbackTo16(%p, (%p, %p, %lu)) #(%x)\n",
gdata16Data.fnCallStub16, pvfn, vpvStack16,
cbStack, ulCall3216_count));
dwResult = CallbackTo16(gdata16Data.fnCallStub16,
vpvStack16+cbStack);
STACKFREE16(vpvStack16, cbAlloc);
}
}
return dwResult;
}
//+---------------------------------------------------------------------------
//
// Function: ThunkCall3216, public
//
// Synopsis: Sets up the 16-bit stack and makes a 32->16 call
//
// Arguments: [pti] - Thunk state info
//
// Returns: Appropriate status code
//
// History: 25-Feb-94 DrewB Created
// 08-Aug-94 AlexT Add IOleClientSite::OnShowWindow code
//
//----------------------------------------------------------------------------
#if DBG == 1
extern "C" ULONG ThunkCall3216_count = 0;
extern "C" ULONG ThunkCall3216_break = 0;
#endif
DWORD ThunkCall3216(THUNKINFO *pti)
{
DWORD dwReturn;
UINT cbStack;
DWORD dwCallerTID;
HRESULT hrCaller;
#if DBG == 1
ULONG ulThunkCall3216_count = ++ThunkCall3216_count;
thkAssert( (ThunkCall3216_count != ThunkCall3216_break) &&
"Break Count Hit");
#endif
thkAssert(*pti->pThop == THOP_END);
pti->pThop++;
thkAssert(*pti->pThop == THOP_ROUTINEINDEX);
pti->pThop++;
thkDebugOut((DEB_ITRACE, "ThunkCall3216 #(%x) %p, index %d\n",
ulThunkCall3216_count, pti->pvfn, *pti->pThop));
cbStack = pti->s16.pbCurrent-pti->s16.pbStart;
// The this pointer should always be on the stack
thkAssert(cbStack >= sizeof(VPVOID));
//
// Hacks for specific interface member functions.
// The placement of these hacks here is by no means an optimal solution.
// It just happens to be convienient for now since everything goes through
// here. This section is for pre-processing.
//
if ( IIDIDX_IS_INDEX(pti->iidx) )
{
switch( IIDIDX_INDEX(pti->iidx) )
{
case THI_IOleClientSite:
#define METHOD_ONSHOWWINDOW 7
if ( pti->dwMethod == METHOD_ONSHOWWINDOW )
{
//
// Here we merge the input queues for the sole reason so that
// we can link the object's window activations into the calling
// thread's window z-order.
//
hrCaller = CoGetCallerTID( &dwCallerTID );
if ( hrCaller == S_FALSE )
{
AttachThreadInput( dwCallerTID, GetCurrentThreadId(),
TRUE );
}
}
break;
case THI_IOleObject:
#define METHOD_DOVERB 11
if ( pti->dwMethod == METHOD_DOVERB )
{
//
// Here we merge the input queues for the sole reason so
// that we can link the object's window activations into
// the calling thread's window z-order.
//
hrCaller = CoGetCallerTID( &dwCallerTID );
if ( hrCaller == S_FALSE )
{
AttachThreadInput( dwCallerTID, GetCurrentThreadId(),
TRUE );
}
}
break;
default:
break;
}
}
pti->pThkMgr->SetThkState(THKSTATE_NOCALL);
dwReturn = Call3216((VPVOID)pti->pvfn, pti->s16.pbStart, cbStack);
pti->pThkMgr->SetThkState(THKSTATE_INVOKETHKOUT16);
//
// Hacks for specific interface member functions.
// Again, the placement of these is by no means an optimal solution.
// They can be moved as long as they have the same effect for just these
// interfaces. This section is for post-processing.
//
if ( IIDIDX_IS_INDEX(pti->iidx) )
{
switch( IIDIDX_INDEX(pti->iidx) )
{
case THI_IOleClientSite:
if ( pti->dwMethod == METHOD_ONSHOWWINDOW )
{
//
// Unmerge the input queues afterward.
//
if ( hrCaller == S_FALSE )
{
AttachThreadInput( dwCallerTID, GetCurrentThreadId(),
FALSE );
}
}
break;
case THI_IOleObject:
if ( pti->dwMethod == METHOD_DOVERB )
{
//
// Unmerge the input queues afterward.
//
if ( hrCaller == S_FALSE )
{
AttachThreadInput( dwCallerTID, GetCurrentThreadId(),
FALSE );
}
}
#define METHOD_GETCLIENTSITE 4
if ( pti->dwMethod == METHOD_GETCLIENTSITE )
{
//
// Excel 5.0a needs to perform some special processing
// on the way out of a IOleObject::GetClientSite call.
// See CTHKMGR.CXX and APINOT.CXX for more details.
//
if ( TlsThkGetAppCompatFlags() & OACF_CLIENTSITE_REF )
{
//
// Tell the thkmgr that we are thunking a bad
// IOleObject::GetClientSite reference on the way out.
//
thkDebugOut((DEB_WARN,"TC3216: OACF_CLIENTSITE_REF used: "
"Setting to clientsite thunk state\n"));
pti->pThkMgr->SetThkState(
THKSTATE_INVOKETHKOUT16_CLIENTSITE);
}
}
break;
case THI_IOleInPlaceFrame:
#define METHOD_REMOVEMENUS 11
//
// Winword 6.0a didn't call OleSetMenuDescriptor(NULL)
// during its IOleInPlaceFrame::RemoveMenus. This leaves
// OLE's frame filter in place. The frame filter holds references
// to some objects so everybody's refcounting gets thrown off
// Here, when we see a RemoveMenus call completing we force
// the OleSetMenuDescriptor(NULL) call to occur. This shuts
// down the frame filter and corrects the reference counts.
//
// There is one other hack necessary: Word unsubclasses the
// window itself directly rather than going through
// OleSetMenuDescriptor. Therefore the frame filter code
// is patched to only unhook if it detects that it is the
// current hook
//
// See APINOT.CXX for more hack code.
//
if (pti->dwMethod == METHOD_REMOVEMENUS)
{
if ( TlsThkGetAppCompatFlags() & OACF_RESETMENU )
{
HRESULT hr;
HWND hwnd;
LPOLEINPLACEFRAME lpoipf;
pti->pThkMgr->SetThkState(THKSTATE_NOCALL);
lpoipf = (LPOLEINPLACEFRAME)pti->this32;
hr = lpoipf->GetWindow( &hwnd );
pti->pThkMgr->SetThkState(THKSTATE_INVOKETHKOUT16);
if ( FAILED(hr) )
{
break;
}
thkDebugOut((DEB_WARN,
"TC3216: OACF_RESETMENU used: "
"Setting menu descriptor "
"to NULL on %p\n", hwnd));
OleSetMenuDescriptor(NULL, hwnd, NULL, NULL, NULL);
}
}
break;
default:
break;
}
}
if ( !pti->fResultThunked )
{
dwReturn = TransformHRESULT_1632( dwReturn );
#if DBG == 1
if (FAILED(dwReturn) )
{
thkDebugOut((DEB_FAILURES,
"Call3216 pvfn = %08lX Probably failed hr = %08lX\n",
pti->pvfn, dwReturn));
}
#endif
}
thkDebugOut((DEB_ITRACE,
"ThunkCall3216 #(%x) returns 0x%08lX\n",
ulThunkCall3216_count,
dwReturn));
return dwReturn;
}
//+---------------------------------------------------------------------------
//
// Function: SetOwnerPublicHMEM16, public
//
// Synopsis: Changes the 16-bit memory handle into a public selector, owned
// by nobody. This prevents any app from taking it away when it
// is cleaned up.
//
// Arguments: [hmem] - 16-bit memory handle
//
// Returns: Appropriate status code
//
// History: 13-Jul-94 BobDay Created it
//
//----------------------------------------------------------------------------
void SetOwnerPublicHMEM16( DWORD hmem )
{
CallbackTo16(gdata16Data.fnSetOwnerPublic16, hmem );
}