office-gobmx/bridges/source/cpp_uno/gcc3_linux_riscv64/cpp2uno.cxx
Sakura286 de0a269a6e riscv64 bridge: replace some preprocessor directives to macros for debugging
Change-Id: Ic53d4a462e12e8448b14e750c4ef2824385b8f28
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/165502
Tested-by: Jenkins
Reviewed-by: Stephan Bergmann <stephan.bergmann@allotropia.de>
2024-04-02 14:16:01 +02:00

740 lines
30 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
/*
* This file is part of the LibreOffice project.
*
* 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/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <com/sun/star/uno/genfunc.hxx>
#include <sal/log.hxx>
#include <typelib/typedescription.hxx>
#include <uno/data.h>
#include <osl/endian.h>
#include "bridge.hxx"
#include "cppinterfaceproxy.hxx"
#include "types.hxx"
#include "vtablefactory.hxx"
#include "call.hxx"
#include "share.hxx"
#include "abi.hxx"
#include <cstring>
#include <typeinfo>
using namespace com::sun::star::uno;
namespace CPPU_CURRENT_NAMESPACE
{
bool is_complex_struct(const typelib_TypeDescription* type)
{
const typelib_CompoundTypeDescription* p
= reinterpret_cast<const typelib_CompoundTypeDescription*>(type);
for (sal_Int32 i = 0; i < p->nMembers; ++i)
{
if (p->ppTypeRefs[i]->eTypeClass == typelib_TypeClass_STRUCT
|| p->ppTypeRefs[i]->eTypeClass == typelib_TypeClass_EXCEPTION)
{
typelib_TypeDescription* t = 0;
TYPELIB_DANGER_GET(&t, p->ppTypeRefs[i]);
bool b = is_complex_struct(t);
TYPELIB_DANGER_RELEASE(t);
if (b)
{
return true;
}
}
else if (!bridges::cpp_uno::shared::isSimpleType(p->ppTypeRefs[i]->eTypeClass))
return true;
}
if (p->pBaseTypeDescription != 0)
return is_complex_struct(&p->pBaseTypeDescription->aBase);
return false;
}
bool return_in_hidden_param(typelib_TypeDescriptionReference* pTypeRef)
{
if (bridges::cpp_uno::shared::isSimpleType(pTypeRef))
return false;
else if (pTypeRef->eTypeClass == typelib_TypeClass_STRUCT
|| pTypeRef->eTypeClass == typelib_TypeClass_EXCEPTION)
{
typelib_TypeDescription* pTypeDescr = 0;
TYPELIB_DANGER_GET(&pTypeDescr, pTypeRef);
//A Composite Type not larger than 16 bytes is returned in up to two GPRs
bool bRet = pTypeDescr->nSize > 16 || is_complex_struct(pTypeDescr);
TYPELIB_DANGER_RELEASE(pTypeDescr);
return bRet;
}
return true;
}
}
namespace
{
static sal_Int32
cpp2uno_call(bridges::cpp_uno::shared::CppInterfaceProxy* pThis,
const typelib_TypeDescription* pMemberTypeDescr,
typelib_TypeDescriptionReference* pReturnTypeRef, // 0 indicates void return
sal_Int32 nParams, typelib_MethodParameter* pParams, void** gpreg, void** fpreg,
void** ovrflw, sal_uInt64* pRegisterReturn /* space for register return */)
{
BRIDGE_LOG("In cpp2uno_call, pThis = %p, pMemberTypeDescr = %p, pReturnTypeRef = %p\n", pThis,
pMemberTypeDescr, pReturnTypeRef);
BRIDGE_LOG("In cpp2uno_call, nParams = %d, pParams = %p, pRegisterReturn = %p\n", nParams,
pParams, pRegisterReturn);
BRIDGE_LOG("In cpp2uno_call, gpreg = %p, fpreg = %p, ovrflw = %p\n", gpreg, fpreg, ovrflw);
unsigned int nr_gpr = 0;
unsigned int nr_fpr = 0;
BRIDGE_LOG("cpp2uno_call:begin\n");
// return
typelib_TypeDescription* pReturnTypeDescr = 0;
if (pReturnTypeRef)
TYPELIB_DANGER_GET(&pReturnTypeDescr, pReturnTypeRef);
void* pUnoReturn = 0;
void* pCppReturn = 0; // complex return ptr: if != 0 && != pUnoReturn, reconversion need
if (pReturnTypeDescr)
{
if (CPPU_CURRENT_NAMESPACE::return_in_hidden_param(pReturnTypeRef))
{
pCppReturn = *gpreg++; // complex return via ptr (pCppReturn)
nr_gpr++;
pUnoReturn = (bridges::cpp_uno::shared::relatesToInterfaceType(pReturnTypeDescr)
? alloca(pReturnTypeDescr->nSize)
: pCppReturn); // direct way
BRIDGE_LOG("cpp2uno_call:complexreturn\n");
}
else
{
pUnoReturn = pRegisterReturn; // direct way for simple types
BRIDGE_LOG("cpp2uno_call:simplereturn\n");
}
}
// pop this
// TODO: Is it really essential to pop?
gpreg++;
nr_gpr++;
// stack space
static_assert(sizeof(void*) == sizeof(sal_Int64), "### unexpected size!");
// parameters
void** pUnoArgs = (void**)alloca(4 * sizeof(void*) * nParams);
void** pCppArgs = pUnoArgs + nParams;
// indices of values this have to be converted (interface conversion cpp<=>uno)
sal_Int32* pTempIndices = (sal_Int32*)(pUnoArgs + (2 * nParams));
// type descriptions for reconversions
typelib_TypeDescription** ppTempParamTypeDescr
= (typelib_TypeDescription**)(pUnoArgs + (3 * nParams));
sal_Int32 nTempIndices = 0;
BRIDGE_LOG("cpp2uno_call:nParams=%d\n", nParams);
for (sal_Int32 nPos = 0; nPos < nParams; ++nPos)
{
const typelib_MethodParameter& rParam = pParams[nPos];
typelib_TypeDescription* pParamTypeDescr = 0;
TYPELIB_DANGER_GET(&pParamTypeDescr, rParam.pTypeRef);
if (!rParam.bOut && bridges::cpp_uno::shared::isSimpleType(pParamTypeDescr)) // value
{
BRIDGE_LOG("cpp2uno_call:Param %u, type %u\n", nPos, pParamTypeDescr->eTypeClass);
switch (pParamTypeDescr->eTypeClass)
{
case typelib_TypeClass_FLOAT:
case typelib_TypeClass_DOUBLE:
if (nr_fpr < MAX_FP_REGS)
{
BRIDGE_LOG("cpp2uno_call:fpr=%p\n", *fpreg);
pCppArgs[nPos] = pUnoArgs[nPos] = fpreg++;
nr_fpr++;
}
else if (nr_gpr < MAX_GP_REGS)
{
BRIDGE_LOG("cpp2uno_call:fpr=%p\n", *gpreg);
pCppArgs[nPos] = pUnoArgs[nPos] = gpreg++;
nr_gpr++;
}
else
{
BRIDGE_LOG("cpp2uno_call:fpr=%p\n", *ovrflw);
pCppArgs[nPos] = pUnoArgs[nPos] = ovrflw++;
}
break;
default:
if (nr_gpr < MAX_GP_REGS)
{
BRIDGE_LOG("cpp2uno_call:gpr=%p\n", *gpreg);
pCppArgs[nPos] = pUnoArgs[nPos] = gpreg++;
nr_gpr++;
}
else
{
BRIDGE_LOG("cpp2uno_call:gpr=%p\n", *ovrflw);
pCppArgs[nPos] = pUnoArgs[nPos] = ovrflw++;
}
break;
}
// no longer needed
TYPELIB_DANGER_RELEASE(pParamTypeDescr);
}
else // ptr to complex value | ref
{
BRIDGE_LOG("cpp2uno_call:ptr|ref\n");
void* pCppStack;
if (nr_gpr < MAX_GP_REGS)
{
pCppArgs[nPos] = pCppStack = *gpreg++;
nr_gpr++;
}
else
{
pCppArgs[nPos] = pCppStack = *ovrflw++;
}
BRIDGE_LOG("cpp2uno_call:pCppStack=%p\n", pCppStack);
if (!rParam.bIn) // is pure out
{
// uno out is unconstructed mem!
pUnoArgs[nPos] = alloca(pParamTypeDescr->nSize);
pTempIndices[nTempIndices] = nPos;
// will be released at reconversion
ppTempParamTypeDescr[nTempIndices++] = pParamTypeDescr;
}
// is in/inout
else if (bridges::cpp_uno::shared::relatesToInterfaceType(pParamTypeDescr))
{
uno_copyAndConvertData(pUnoArgs[nPos] = alloca(pParamTypeDescr->nSize), pCppStack,
pParamTypeDescr, pThis->getBridge()->getCpp2Uno());
pTempIndices[nTempIndices] = nPos; // has to be reconverted
// will be released at reconversion
ppTempParamTypeDescr[nTempIndices++] = pParamTypeDescr;
BRIDGE_LOG("cpp2uno_call:related to interface,%p,%d,pUnoargs[%d]=%p\n", pCppStack,
pParamTypeDescr->nSize, nPos, pUnoArgs[nPos]);
}
else // direct way
{
pUnoArgs[nPos] = pCppStack;
BRIDGE_LOG("cpp2uno_call:direct,pUnoArgs[%d]=%p\n", nPos, pUnoArgs[nPos]);
// no longer needed
TYPELIB_DANGER_RELEASE(pParamTypeDescr);
}
}
}
BRIDGE_LOG("cpp2uno_call2,%p,unoargs=%p\n", pThis->getUnoI()->pDispatcher, pUnoArgs);
BRIDGE_LOG("pMemberTypeDescr=%p,pUnoReturn=%p\n", pMemberTypeDescr, pUnoReturn);
// ExceptionHolder
uno_Any aUnoExc; // Any will be constructed by callee
uno_Any* pUnoExc = &aUnoExc;
BRIDGE_LOG("pThis=%p,pThis->getUnoI()=%p,pMemberTypeDescr=%p\npUnoReturn=%p,pUnoArgs=%p", pThis,
pThis->getUnoI(), pMemberTypeDescr, pUnoReturn, pUnoArgs);
// invoke uno dispatch call
(*pThis->getUnoI()->pDispatcher)(pThis->getUnoI(), pMemberTypeDescr, pUnoReturn, pUnoArgs,
&pUnoExc);
BRIDGE_LOG("cpp2uno_call2,after dispatch\n");
// in case an exception occurred...
if (pUnoExc)
{
// destruct temporary in/inout params
for (; nTempIndices--;)
{
sal_Int32 nIndex = pTempIndices[nTempIndices];
if (pParams[nIndex].bIn) // is in/inout => was constructed
uno_destructData(pUnoArgs[nIndex], ppTempParamTypeDescr[nTempIndices], 0);
TYPELIB_DANGER_RELEASE(ppTempParamTypeDescr[nTempIndices]);
}
if (pReturnTypeDescr)
TYPELIB_DANGER_RELEASE(pReturnTypeDescr);
CPPU_CURRENT_NAMESPACE::raiseException(&aUnoExc, pThis->getBridge()->getUno2Cpp());
// has to destruct the any
// is here for dummy
return typelib_TypeClass_VOID;
}
else // else no exception occurred...
{
// temporary params
for (; nTempIndices--;)
{
sal_Int32 nIndex = pTempIndices[nTempIndices];
typelib_TypeDescription* pParamTypeDescr = ppTempParamTypeDescr[nTempIndices];
if (pParams[nIndex].bOut) // inout/out
{
// convert and assign
uno_destructData(pCppArgs[nIndex], pParamTypeDescr, cpp_release);
uno_copyAndConvertData(pCppArgs[nIndex], pUnoArgs[nIndex], pParamTypeDescr,
pThis->getBridge()->getUno2Cpp());
}
// destroy temp uno param
uno_destructData(pUnoArgs[nIndex], pParamTypeDescr, 0);
TYPELIB_DANGER_RELEASE(pParamTypeDescr);
}
//void* retout = nullptr; // avoid false -Werror=maybe-uninitialized
// return
sal_Int32 returnType = 0;
if (pReturnTypeDescr)
{
if (!bridges::cpp_uno::shared::relatesToInterfaceType(pReturnTypeDescr))
{
const bool isSigned = true;
switch (pReturnTypeDescr == nullptr ? typelib_TypeClass_VOID
: pReturnTypeDescr->eTypeClass)
{
// Sometimes we need to return a smaller type into a larger type.
//
// For example, in pyuno.cxx:PyUNO_bool(), an int(32bit) is returned
// in type Py_ssize_t(64bit)
// We assume that this 32bit int was put in low 32 bit of register a0.
// The bridge may return with high 32 bit uncleaned and compiler might
// directly bind this register to 64 bit variable.
//
// This bug occurs when build pyuno with gcc-12 with -O2.
// https://bugs.documentfoundation.org/show_bug.cgi?id=155937
//
// So we need to clean the higher bits in bridge.
case typelib_TypeClass_BOOLEAN:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn),
!isSigned, 1);
break;
case typelib_TypeClass_BYTE:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn), isSigned,
1);
break;
case typelib_TypeClass_CHAR:
case typelib_TypeClass_UNSIGNED_SHORT:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn),
!isSigned, 2);
break;
case typelib_TypeClass_SHORT:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn), isSigned,
2);
break;
case typelib_TypeClass_UNSIGNED_LONG:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn),
!isSigned, 4);
break;
case typelib_TypeClass_LONG:
abi_riscv64::extIntBits(pRegisterReturn,
reinterpret_cast<sal_uInt64*>(pUnoReturn), isSigned,
4);
break;
// TODO: check the source of the enum type.
case typelib_TypeClass_ENUM:
case typelib_TypeClass_UNSIGNED_HYPER:
case typelib_TypeClass_HYPER:
std::memcpy(reinterpret_cast<char*>(pRegisterReturn), pUnoReturn, 8);
break;
case typelib_TypeClass_FLOAT:
std::memcpy(reinterpret_cast<char*>(pRegisterReturn), pUnoReturn, 4);
std::memset(reinterpret_cast<char*>(pRegisterReturn) + 4, 0xFF, 4);
break;
case typelib_TypeClass_DOUBLE:
std::memcpy(reinterpret_cast<char*>(pRegisterReturn), pUnoReturn, 8);
break;
case typelib_TypeClass_STRUCT:
case typelib_TypeClass_EXCEPTION:
abi_riscv64::splitUNOStruct(
pReturnTypeDescr, reinterpret_cast<sal_uInt64*>(pRegisterReturn),
reinterpret_cast<sal_uInt64*>(pUnoReturn), returnType);
break;
case typelib_TypeClass_VOID:
break;
default:
if (pUnoReturn)
{
std::memcpy(pRegisterReturn, pUnoReturn, 16);
}
BRIDGE_LOG("Unhandled Type: %d\n", pReturnTypeDescr->eTypeClass);
}
}
else
{
uno_copyAndConvertData(pCppReturn, pUnoReturn, pReturnTypeDescr,
pThis->getBridge()->getUno2Cpp());
// destroy temp uno return
uno_destructData(pUnoReturn, pReturnTypeDescr, 0);
// complex return ptr is set to return reg
*(void**)pRegisterReturn = pCppReturn;
}
TYPELIB_DANGER_RELEASE(pReturnTypeDescr);
}
return returnType;
}
}
/**
* is called on incoming vtable calls
* (called by asm snippets)
*/
sal_Int32 cpp_vtable_call(sal_Int32 nFunctionIndex, sal_Int32 nVtableOffset, void** gpreg,
void** fpreg, void** ovrflw,
sal_uInt64* pRegisterReturn /* space for register return */)
{
static_assert(sizeof(sal_Int64) == sizeof(void*), "### unexpected!");
BRIDGE_LOG("in cpp_vtable_call nFunctionIndex is %d\n", nFunctionIndex);
BRIDGE_LOG("in cpp_vtable_call nVtableOffset is %d\n", nVtableOffset);
BRIDGE_LOG("in cpp_vtable_call gp=%p, fp=%p, ov=%p\n", gpreg, fpreg, ovrflw);
// gpreg: [ret *], this, [other gpr params]
// fpreg: [fpr params]
// ovrflw: [gpr or fpr params (properly aligned)]
void* pThis;
if (nFunctionIndex & 0x80000000)
{
nFunctionIndex &= 0x7fffffff;
pThis = gpreg[1];
}
else
{
pThis = gpreg[0];
}
BRIDGE_LOG("cpp_vtable_call, pThis=%p, nFunctionIndex=%d, nVtableOffset=%d\n", pThis,
nFunctionIndex, nVtableOffset);
pThis = static_cast<char*>(pThis) - nVtableOffset;
bridges::cpp_uno::shared::CppInterfaceProxy* pCppI
= bridges::cpp_uno::shared::CppInterfaceProxy::castInterfaceToProxy(pThis);
BRIDGE_LOG("cpp_vtable_call, pCppI=%p\n", pCppI);
typelib_InterfaceTypeDescription* pTypeDescr = pCppI->getTypeDescr();
if (nFunctionIndex >= pTypeDescr->nMapFunctionIndexToMemberIndex)
{
SAL_WARN("bridges", "illegal " << OUString::unacquired(&pTypeDescr->aBase.pTypeName)
<< " vtable index " << nFunctionIndex << "/"
<< pTypeDescr->nMapFunctionIndexToMemberIndex);
throw RuntimeException(("illegal " + OUString::unacquired(&pTypeDescr->aBase.pTypeName)
+ " vtable index " + OUString::number(nFunctionIndex) + "/"
+ OUString::number(pTypeDescr->nMapFunctionIndexToMemberIndex)),
(XInterface*)pThis);
}
// determine called method
sal_Int32 nMemberPos = pTypeDescr->pMapFunctionIndexToMemberIndex[nFunctionIndex];
assert(nMemberPos < pTypeDescr->nAllMembers);
TypeDescription aMemberDescr(pTypeDescr->ppAllMembers[nMemberPos]);
sal_Int32 eRet;
switch (aMemberDescr.get()->eTypeClass)
{
case typelib_TypeClass_INTERFACE_ATTRIBUTE:
{
BRIDGE_LOG("cpp_vtable_call interface attribute\n");
typelib_TypeDescriptionReference* pAttrTypeRef
= reinterpret_cast<typelib_InterfaceAttributeTypeDescription*>(aMemberDescr.get())
->pAttributeTypeRef;
if (pTypeDescr->pMapMemberIndexToFunctionIndex[nMemberPos] == nFunctionIndex)
{
// is GET method
eRet = cpp2uno_call(pCppI, aMemberDescr.get(), pAttrTypeRef, 0, 0, // no params
gpreg, fpreg, ovrflw, pRegisterReturn);
}
else
{
// is SET method
typelib_MethodParameter aParam;
aParam.pTypeRef = pAttrTypeRef;
aParam.bIn = sal_True;
aParam.bOut = sal_False;
eRet = cpp2uno_call(pCppI, aMemberDescr.get(),
0, // indicates void return
1, &aParam, gpreg, fpreg, ovrflw, pRegisterReturn);
}
break;
}
case typelib_TypeClass_INTERFACE_METHOD:
{
BRIDGE_LOG("cpp_vtable_call interface method\n");
// is METHOD
switch (nFunctionIndex)
{
case 1: // acquire()
BRIDGE_LOG("cpp_vtable_call method acquire\n");
pCppI->acquireProxy(); // non virtual call!
eRet = 0;
break;
case 2: // release()
BRIDGE_LOG("cpp_vtable_call method release\n");
pCppI->releaseProxy(); // non virtual call!
eRet = 0;
break;
case 0: // queryInterface() opt
{
BRIDGE_LOG("cpp_vtable_call method query interface opt\n");
typelib_TypeDescription* pTD = 0;
TYPELIB_DANGER_GET(&pTD, reinterpret_cast<Type*>(gpreg[2])->getTypeLibType());
if (pTD)
{
XInterface* pInterface = 0;
(*pCppI->getBridge()->getCppEnv()->getRegisteredInterface)(
pCppI->getBridge()->getCppEnv(), (void**)&pInterface,
pCppI->getOid().pData,
reinterpret_cast<typelib_InterfaceTypeDescription*>(pTD));
if (pInterface)
{
::uno_any_construct(reinterpret_cast<uno_Any*>(gpreg[0]), &pInterface,
pTD, cpp_acquire);
pInterface->release();
TYPELIB_DANGER_RELEASE(pTD);
reinterpret_cast<void**>(pRegisterReturn)[0] = gpreg[0];
eRet = 0;
break;
}
TYPELIB_DANGER_RELEASE(pTD);
}
[[fallthrough]];
} // else perform queryInterface()
default:
BRIDGE_LOG("cpp_vtable_call method query interface\n");
typelib_InterfaceMethodTypeDescription* pMethodTD
= reinterpret_cast<typelib_InterfaceMethodTypeDescription*>(
aMemberDescr.get());
eRet = cpp2uno_call(pCppI, aMemberDescr.get(), pMethodTD->pReturnTypeRef,
pMethodTD->nParams, pMethodTD->pParams, gpreg, fpreg,
ovrflw, pRegisterReturn);
}
break;
}
default:
{
BRIDGE_LOG("cpp_vtable_call no member\n");
throw RuntimeException("no member description found!", (XInterface*)pThis);
}
}
return eRet;
}
extern "C" void privateSnippetExecutor(...);
int const codeSnippetSize = 0x6c;
unsigned char* codeSnippet(unsigned char* code, sal_Int32 functionIndex, sal_Int32 vtableOffset,
bool bHasHiddenParam)
{
BRIDGE_LOG("in codeSnippet functionIndex is %d\n", functionIndex);
BRIDGE_LOG("in codeSnippet vtableOffset is %d\n", vtableOffset);
BRIDGE_LOG("in codeSnippet privateSnippetExecutor is %lx\n",
(unsigned long)privateSnippetExecutor);
BRIDGE_LOG("in codeSnippet cpp_vtable_call is %lx\n", (unsigned long)cpp_vtable_call);
if (bHasHiddenParam)
functionIndex |= 0x80000000;
unsigned int* p = (unsigned int*)code;
assert((((unsigned long)code) & 0x3) == 0); //aligned to 4 otherwise a mistake
/* generate this code */
/*
It is complex to load a 64bit address because you cannot load
an unsigned number to register on RISC-V.
# load functionIndex to t4
00000eb7 lui t4,0x0
000eee93 ori t4,t4,0x0
# load privateSnippetExecutor to t0
000002b7 lui t0,0x0
02429293 slli t0,t0,36
00000337 lui t1,0x0
01431313 slli t1,t1,20
0062e2b3 or t0,t0,t1
00000337 lui t1,0x0
00431313 slli t1,t1,4
0062e2b3 or t0,t0,t1
00000337 lui t1,0x0
00c35313 srli t1,t1,12
0062e2b3 or t0,t0,t1
# load cpp_vtable_call to t6
00000fb7 lui t6,0x0
024f9f93 slli t6,t6,36
00000337 lui t1,0x0
01431313 slli t1,t1,20
006fefb3 or t6,t6,t1
00000337 lui t1,0x0
00431313 slli t1,t1,4
006fefb3 or t6,t6,t1
00000337 lui t1,0x0
00c35313 srli t1,t1,12
006fefb3 or t6,t6,t1
# load vtableOffset to t5
00000f37 lui t5,0x0
000f6f13 ori t5,t5,0x0
# jump to privateSnippetExecutor
00028067 jalr zero,t0,0x0
*/
*p++ = 0x00000eb7 | ((functionIndex)&0xfffff000);
*p++ = 0x000eee93 | ((functionIndex << 20) & 0xfff00000);
// load privateSnippetExecutor to t0
unsigned long functionEntry = ((unsigned long)privateSnippetExecutor);
*p++ = 0x000002b7 | ((functionEntry >> 36) & 0x000000000ffff000);
*p++ = 0x02429293;
*p++ = 0x00000337 | ((functionEntry >> 20) & 0x000000000ffff000);
*p++ = 0x01431313;
*p++ = 0x0062e2b3;
*p++ = 0x00000337 | ((functionEntry >> 4) & 0x000000000ffff000);
*p++ = 0x00431313;
*p++ = 0x0062e2b3;
*p++ = 0x00000337 | ((functionEntry << 12) & 0x000000000ffff000);
*p++ = 0x00c35313;
*p++ = 0x0062e2b3;
// load cpp_vtable_call to t6
functionEntry = (unsigned long)cpp_vtable_call;
*p++ = 0x00000fb7 | ((functionEntry >> 36) & 0x000000000ffff000);
*p++ = 0x024f9f93;
*p++ = 0x00000337 | ((functionEntry >> 20) & 0x000000000ffff000);
*p++ = 0x01431313;
*p++ = 0x006fefb3;
*p++ = 0x00000337 | ((functionEntry >> 4) & 0x000000000ffff000);
*p++ = 0x00431313;
*p++ = 0x006fefb3;
*p++ = 0x00000337 | ((functionEntry << 12) & 0x000000000ffff000);
*p++ = 0x00c35313;
*p++ = 0x006fefb3;
// load vtableOffset to t5
*p++ = 0x00000f37 | ((vtableOffset)&0xfffff000);
*p++ = 0x000f6f13 | ((vtableOffset << 20) & 0xfff00000);
// jump to privateSnippetExecutor
*p++ = 0x00028067;
return (code + codeSnippetSize);
}
}
void bridges::cpp_uno::shared::VtableFactory::flushCode(unsigned char const*, unsigned char const*)
{
asm volatile("fence" :::);
}
struct bridges::cpp_uno::shared::VtableFactory::Slot
{
void const* fn;
};
bridges::cpp_uno::shared::VtableFactory::Slot*
bridges::cpp_uno::shared::VtableFactory::mapBlockToVtable(void* block)
{
return static_cast<Slot*>(block) + 2;
}
std::size_t bridges::cpp_uno::shared::VtableFactory::getBlockSize(sal_Int32 slotCount)
{
return (slotCount + 2) * sizeof(Slot) + slotCount * codeSnippetSize;
}
namespace
{
// Some dummy type whose RTTI is used in the synthesized proxy vtables to make uses of dynamic_cast
// on such proxy objects not crash:
struct ProxyRtti
{
};
}
bridges::cpp_uno::shared::VtableFactory::Slot*
bridges::cpp_uno::shared::VtableFactory::initializeBlock(void* block, sal_Int32 slotCount,
sal_Int32,
typelib_InterfaceTypeDescription*)
{
Slot* slots = mapBlockToVtable(block);
slots[-2].fn = 0; //null
slots[-1].fn = &typeid(ProxyRtti);
return slots + slotCount;
}
unsigned char* bridges::cpp_uno::shared::VtableFactory::addLocalFunctions(
Slot** slots, unsigned char* code, sal_PtrDiff writetoexecdiff,
typelib_InterfaceTypeDescription const* type, sal_Int32 functionOffset, sal_Int32 functionCount,
sal_Int32 vtableOffset)
{
(*slots) -= functionCount;
Slot* s = *slots;
BRIDGE_LOG("in addLocalFunctions functionOffset is %d\n", functionOffset);
BRIDGE_LOG("in addLocalFunctions vtableOffset is %d\n", vtableOffset);
BRIDGE_LOG("nMembers=%d\n", type->nMembers);
for (sal_Int32 i = 0; i < type->nMembers; ++i)
{
typelib_TypeDescription* member = 0;
TYPELIB_DANGER_GET(&member, type->ppMembers[i]);
assert(member != 0);
switch (member->eTypeClass)
{
case typelib_TypeClass_INTERFACE_ATTRIBUTE:
// Getter:
(s++)->fn = code + writetoexecdiff;
code = codeSnippet(
code, functionOffset++, vtableOffset,
CPPU_CURRENT_NAMESPACE::return_in_hidden_param(
reinterpret_cast<typelib_InterfaceAttributeTypeDescription*>(member)
->pAttributeTypeRef));
// Setter:
if (!reinterpret_cast<typelib_InterfaceAttributeTypeDescription*>(member)
->bReadOnly)
{
(s++)->fn = code + writetoexecdiff;
code = codeSnippet(code, functionOffset++, vtableOffset, false);
}
break;
case typelib_TypeClass_INTERFACE_METHOD:
(s++)->fn = code + writetoexecdiff;
code = codeSnippet(
code, functionOffset++, vtableOffset,
CPPU_CURRENT_NAMESPACE::return_in_hidden_param(
reinterpret_cast<typelib_InterfaceMethodTypeDescription*>(member)
->pReturnTypeRef));
break;
default:
assert(false);
break;
}
TYPELIB_DANGER_RELEASE(member);
}
return code;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s cinkeys+=0=break: */