1*cdf0e10cSrcweir /*************************************************************************
2*cdf0e10cSrcweir  *
3*cdf0e10cSrcweir  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*cdf0e10cSrcweir  *
5*cdf0e10cSrcweir  * Copyright 2000, 2010 Oracle and/or its affiliates.
6*cdf0e10cSrcweir  *
7*cdf0e10cSrcweir  * OpenOffice.org - a multi-platform office productivity suite
8*cdf0e10cSrcweir  *
9*cdf0e10cSrcweir  * This file is part of OpenOffice.org.
10*cdf0e10cSrcweir  *
11*cdf0e10cSrcweir  * OpenOffice.org is free software: you can redistribute it and/or modify
12*cdf0e10cSrcweir  * it under the terms of the GNU Lesser General Public License version 3
13*cdf0e10cSrcweir  * only, as published by the Free Software Foundation.
14*cdf0e10cSrcweir  *
15*cdf0e10cSrcweir  * OpenOffice.org is distributed in the hope that it will be useful,
16*cdf0e10cSrcweir  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17*cdf0e10cSrcweir  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*cdf0e10cSrcweir  * GNU Lesser General Public License version 3 for more details
19*cdf0e10cSrcweir  * (a copy is included in the LICENSE file that accompanied this code).
20*cdf0e10cSrcweir  *
21*cdf0e10cSrcweir  * You should have received a copy of the GNU Lesser General Public License
22*cdf0e10cSrcweir  * version 3 along with OpenOffice.org.  If not, see
23*cdf0e10cSrcweir  * <http://www.openoffice.org/license.html>
24*cdf0e10cSrcweir  * for a copy of the LGPLv3 License.
25*cdf0e10cSrcweir  *
26*cdf0e10cSrcweir  ************************************************************************/
27*cdf0e10cSrcweir 
28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
29*cdf0e10cSrcweir #include "precompiled_bridges.hxx"
30*cdf0e10cSrcweir 
31*cdf0e10cSrcweir #include "bridges/cpp_uno/shared/vtables.hxx"
32*cdf0e10cSrcweir 
33*cdf0e10cSrcweir #include "osl/diagnose.h"
34*cdf0e10cSrcweir #include "sal/types.h"
35*cdf0e10cSrcweir #include "typelib/typedescription.h"
36*cdf0e10cSrcweir 
37*cdf0e10cSrcweir #include <algorithm>
38*cdf0e10cSrcweir 
39*cdf0e10cSrcweir namespace
40*cdf0e10cSrcweir {
41*cdf0e10cSrcweir 
42*cdf0e10cSrcweir /**
43*cdf0e10cSrcweir  *  Calculates the number of vtables associated with an interface type.
44*cdf0e10cSrcweir  *
45*cdf0e10cSrcweir  * <p>Multiple-inheritance C++ classes have more than one vtable.</p>
46*cdf0e10cSrcweir  *
47*cdf0e10cSrcweir  * @param type a non-null pointer to an interface type description
48*cdf0e10cSrcweir  * @return the number of vtables associated with the given interface type
49*cdf0e10cSrcweir  */
50*cdf0e10cSrcweir sal_Int32 getVtableCount(typelib_InterfaceTypeDescription const * type) {
51*cdf0e10cSrcweir     sal_Int32 n = 0;
52*cdf0e10cSrcweir     for (sal_Int32 i = 0; i < type->nBaseTypes; ++i) {
53*cdf0e10cSrcweir         n += getVtableCount(type->ppBaseTypes[i]);
54*cdf0e10cSrcweir     }
55*cdf0e10cSrcweir     return std::max< sal_Int32 >(n, 1);
56*cdf0e10cSrcweir }
57*cdf0e10cSrcweir 
58*cdf0e10cSrcweir /**
59*cdf0e10cSrcweir  * Maps a local member index to a local function index.
60*cdf0e10cSrcweir  *
61*cdf0e10cSrcweir  * <p><em>Local</em> members/functions are those not inherited from any base
62*cdf0e10cSrcweir  * types.  The number of <em>functions</em> is potentially larger than the
63*cdf0e10cSrcweir  * number of <em>members</em>, as each read&ndash;write attribute member counts
64*cdf0e10cSrcweir  * as two functions.</p>
65*cdf0e10cSrcweir  *
66*cdf0e10cSrcweir  * @param type a non-null pointer to an interface type description
67*cdf0e10cSrcweir  * @param localMember a local member index, relative to the given interface type
68*cdf0e10cSrcweir  * @return the local function index corresponding to the given local member
69*cdf0e10cSrcweir  *     index, relative to the given interface type
70*cdf0e10cSrcweir  */
71*cdf0e10cSrcweir sal_Int32 mapLocalMemberToLocalFunction(
72*cdf0e10cSrcweir     typelib_InterfaceTypeDescription * type, sal_Int32 localMember)
73*cdf0e10cSrcweir {
74*cdf0e10cSrcweir     typelib_typedescription_complete(
75*cdf0e10cSrcweir         reinterpret_cast< typelib_TypeDescription ** >(&type));
76*cdf0e10cSrcweir     sal_Int32 localMemberOffset = type->nAllMembers - type->nMembers;
77*cdf0e10cSrcweir     sal_Int32 localFunctionOffset = type->nMapFunctionIndexToMemberIndex
78*cdf0e10cSrcweir         - bridges::cpp_uno::shared::getLocalFunctions(type);
79*cdf0e10cSrcweir     return type->pMapMemberIndexToFunctionIndex[localMemberOffset + localMember]
80*cdf0e10cSrcweir         - localFunctionOffset;
81*cdf0e10cSrcweir }
82*cdf0e10cSrcweir 
83*cdf0e10cSrcweir // Since on Solaris we compile with --instances=static, getVtableSlot cannot be
84*cdf0e10cSrcweir // a template function, with explicit instantiates for
85*cdf0e10cSrcweir // T = typelib_InterfaceAttributeTypeDescription and
86*cdf0e10cSrcweir // T = typelib_InterfaceMethodTypeDescription in this file; hence, there are two
87*cdf0e10cSrcweir // overloaded versions of getVtableSlot that both delegate to this template
88*cdf0e10cSrcweir // function:
89*cdf0e10cSrcweir template< typename T > bridges::cpp_uno::shared::VtableSlot doGetVtableSlot(
90*cdf0e10cSrcweir     T const * ifcMember)
91*cdf0e10cSrcweir {
92*cdf0e10cSrcweir     bridges::cpp_uno::shared::VtableSlot slot;
93*cdf0e10cSrcweir     slot.offset = 0;
94*cdf0e10cSrcweir     T * member = const_cast< T * >(ifcMember);
95*cdf0e10cSrcweir     while (member->pBaseRef != 0) {
96*cdf0e10cSrcweir         OSL_ASSERT(member->nIndex < member->pInterface->nBaseTypes);
97*cdf0e10cSrcweir         for (sal_Int32 i = 0; i < member->nIndex; ++i) {
98*cdf0e10cSrcweir             slot.offset += getVtableCount(member->pInterface->ppBaseTypes[i]);
99*cdf0e10cSrcweir         }
100*cdf0e10cSrcweir         typelib_TypeDescription * desc = 0;
101*cdf0e10cSrcweir         typelib_typedescriptionreference_getDescription(
102*cdf0e10cSrcweir             &desc, member->pBaseRef);
103*cdf0e10cSrcweir         OSL_ASSERT(
104*cdf0e10cSrcweir             desc != 0 && desc->eTypeClass == member->aBase.aBase.eTypeClass);
105*cdf0e10cSrcweir         if (member != ifcMember) {
106*cdf0e10cSrcweir             typelib_typedescription_release(&member->aBase.aBase);
107*cdf0e10cSrcweir         }
108*cdf0e10cSrcweir         member = reinterpret_cast< T * >(desc);
109*cdf0e10cSrcweir     }
110*cdf0e10cSrcweir     slot.index
111*cdf0e10cSrcweir         = bridges::cpp_uno::shared::getPrimaryFunctions(
112*cdf0e10cSrcweir             member->pInterface->pBaseTypeDescription)
113*cdf0e10cSrcweir         + mapLocalMemberToLocalFunction(member->pInterface, member->nIndex);
114*cdf0e10cSrcweir     if (member != ifcMember) {
115*cdf0e10cSrcweir         typelib_typedescription_release(&member->aBase.aBase);
116*cdf0e10cSrcweir     }
117*cdf0e10cSrcweir     return slot;
118*cdf0e10cSrcweir }
119*cdf0e10cSrcweir 
120*cdf0e10cSrcweir }
121*cdf0e10cSrcweir 
122*cdf0e10cSrcweir namespace bridges { namespace cpp_uno { namespace shared {
123*cdf0e10cSrcweir 
124*cdf0e10cSrcweir sal_Int32 getLocalFunctions(typelib_InterfaceTypeDescription const * type) {
125*cdf0e10cSrcweir     return type->nMembers == 0
126*cdf0e10cSrcweir         ? 0
127*cdf0e10cSrcweir         : (type->nMapFunctionIndexToMemberIndex
128*cdf0e10cSrcweir            - type->pMapMemberIndexToFunctionIndex[
129*cdf0e10cSrcweir                type->nAllMembers - type->nMembers]);
130*cdf0e10cSrcweir }
131*cdf0e10cSrcweir 
132*cdf0e10cSrcweir sal_Int32 getPrimaryFunctions(typelib_InterfaceTypeDescription * type) {
133*cdf0e10cSrcweir     sal_Int32 n = 0;
134*cdf0e10cSrcweir     for (; type != 0; type = type->pBaseTypeDescription) {
135*cdf0e10cSrcweir         typelib_typedescription_complete(
136*cdf0e10cSrcweir             reinterpret_cast< typelib_TypeDescription ** >(&type));
137*cdf0e10cSrcweir         n += getLocalFunctions(type);
138*cdf0e10cSrcweir     }
139*cdf0e10cSrcweir     return n;
140*cdf0e10cSrcweir }
141*cdf0e10cSrcweir 
142*cdf0e10cSrcweir VtableSlot getVtableSlot(
143*cdf0e10cSrcweir     typelib_InterfaceAttributeTypeDescription const * ifcMember)
144*cdf0e10cSrcweir {
145*cdf0e10cSrcweir     return doGetVtableSlot(ifcMember);
146*cdf0e10cSrcweir }
147*cdf0e10cSrcweir 
148*cdf0e10cSrcweir VtableSlot getVtableSlot(
149*cdf0e10cSrcweir     typelib_InterfaceMethodTypeDescription const * ifcMember)
150*cdf0e10cSrcweir {
151*cdf0e10cSrcweir     return doGetVtableSlot(ifcMember);
152*cdf0e10cSrcweir }
153*cdf0e10cSrcweir 
154*cdf0e10cSrcweir } } }
155