1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_bridges.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include "bridges/cpp_uno/shared/vtables.hxx"
28*b1cdbd2cSJim Jagielski 
29*b1cdbd2cSJim Jagielski #include "osl/diagnose.h"
30*b1cdbd2cSJim Jagielski #include "sal/types.h"
31*b1cdbd2cSJim Jagielski #include "typelib/typedescription.h"
32*b1cdbd2cSJim Jagielski 
33*b1cdbd2cSJim Jagielski #include <algorithm>
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski namespace
36*b1cdbd2cSJim Jagielski {
37*b1cdbd2cSJim Jagielski 
38*b1cdbd2cSJim Jagielski /**
39*b1cdbd2cSJim Jagielski  *  Calculates the number of vtables associated with an interface type.
40*b1cdbd2cSJim Jagielski  *
41*b1cdbd2cSJim Jagielski  * <p>Multiple-inheritance C++ classes have more than one vtable.</p>
42*b1cdbd2cSJim Jagielski  *
43*b1cdbd2cSJim Jagielski  * @param type a non-null pointer to an interface type description
44*b1cdbd2cSJim Jagielski  * @return the number of vtables associated with the given interface type
45*b1cdbd2cSJim Jagielski  */
getVtableCount(typelib_InterfaceTypeDescription const * type)46*b1cdbd2cSJim Jagielski sal_Int32 getVtableCount(typelib_InterfaceTypeDescription const * type) {
47*b1cdbd2cSJim Jagielski     sal_Int32 n = 0;
48*b1cdbd2cSJim Jagielski     for (sal_Int32 i = 0; i < type->nBaseTypes; ++i) {
49*b1cdbd2cSJim Jagielski         n += getVtableCount(type->ppBaseTypes[i]);
50*b1cdbd2cSJim Jagielski     }
51*b1cdbd2cSJim Jagielski     return std::max< sal_Int32 >(n, 1);
52*b1cdbd2cSJim Jagielski }
53*b1cdbd2cSJim Jagielski 
54*b1cdbd2cSJim Jagielski /**
55*b1cdbd2cSJim Jagielski  * Maps a local member index to a local function index.
56*b1cdbd2cSJim Jagielski  *
57*b1cdbd2cSJim Jagielski  * <p><em>Local</em> members/functions are those not inherited from any base
58*b1cdbd2cSJim Jagielski  * types.  The number of <em>functions</em> is potentially larger than the
59*b1cdbd2cSJim Jagielski  * number of <em>members</em>, as each read&ndash;write attribute member counts
60*b1cdbd2cSJim Jagielski  * as two functions.</p>
61*b1cdbd2cSJim Jagielski  *
62*b1cdbd2cSJim Jagielski  * @param type a non-null pointer to an interface type description
63*b1cdbd2cSJim Jagielski  * @param localMember a local member index, relative to the given interface type
64*b1cdbd2cSJim Jagielski  * @return the local function index corresponding to the given local member
65*b1cdbd2cSJim Jagielski  *     index, relative to the given interface type
66*b1cdbd2cSJim Jagielski  */
mapLocalMemberToLocalFunction(typelib_InterfaceTypeDescription * type,sal_Int32 localMember)67*b1cdbd2cSJim Jagielski sal_Int32 mapLocalMemberToLocalFunction(
68*b1cdbd2cSJim Jagielski     typelib_InterfaceTypeDescription * type, sal_Int32 localMember)
69*b1cdbd2cSJim Jagielski {
70*b1cdbd2cSJim Jagielski     typelib_typedescription_complete(
71*b1cdbd2cSJim Jagielski         reinterpret_cast< typelib_TypeDescription ** >(&type));
72*b1cdbd2cSJim Jagielski     sal_Int32 localMemberOffset = type->nAllMembers - type->nMembers;
73*b1cdbd2cSJim Jagielski     sal_Int32 localFunctionOffset = type->nMapFunctionIndexToMemberIndex
74*b1cdbd2cSJim Jagielski         - bridges::cpp_uno::shared::getLocalFunctions(type);
75*b1cdbd2cSJim Jagielski     return type->pMapMemberIndexToFunctionIndex[localMemberOffset + localMember]
76*b1cdbd2cSJim Jagielski         - localFunctionOffset;
77*b1cdbd2cSJim Jagielski }
78*b1cdbd2cSJim Jagielski 
79*b1cdbd2cSJim Jagielski // Since on Solaris we compile with --instances=static, getVtableSlot cannot be
80*b1cdbd2cSJim Jagielski // a template function, with explicit instantiates for
81*b1cdbd2cSJim Jagielski // T = typelib_InterfaceAttributeTypeDescription and
82*b1cdbd2cSJim Jagielski // T = typelib_InterfaceMethodTypeDescription in this file; hence, there are two
83*b1cdbd2cSJim Jagielski // overloaded versions of getVtableSlot that both delegate to this template
84*b1cdbd2cSJim Jagielski // function:
doGetVtableSlot(T const * ifcMember)85*b1cdbd2cSJim Jagielski template< typename T > bridges::cpp_uno::shared::VtableSlot doGetVtableSlot(
86*b1cdbd2cSJim Jagielski     T const * ifcMember)
87*b1cdbd2cSJim Jagielski {
88*b1cdbd2cSJim Jagielski     bridges::cpp_uno::shared::VtableSlot slot;
89*b1cdbd2cSJim Jagielski     slot.offset = 0;
90*b1cdbd2cSJim Jagielski     T * member = const_cast< T * >(ifcMember);
91*b1cdbd2cSJim Jagielski     while (member->pBaseRef != 0) {
92*b1cdbd2cSJim Jagielski         OSL_ASSERT(member->nIndex < member->pInterface->nBaseTypes);
93*b1cdbd2cSJim Jagielski         for (sal_Int32 i = 0; i < member->nIndex; ++i) {
94*b1cdbd2cSJim Jagielski             slot.offset += getVtableCount(member->pInterface->ppBaseTypes[i]);
95*b1cdbd2cSJim Jagielski         }
96*b1cdbd2cSJim Jagielski         typelib_TypeDescription * desc = 0;
97*b1cdbd2cSJim Jagielski         typelib_typedescriptionreference_getDescription(
98*b1cdbd2cSJim Jagielski             &desc, member->pBaseRef);
99*b1cdbd2cSJim Jagielski         OSL_ASSERT(
100*b1cdbd2cSJim Jagielski             desc != 0 && desc->eTypeClass == member->aBase.aBase.eTypeClass);
101*b1cdbd2cSJim Jagielski         if (member != ifcMember) {
102*b1cdbd2cSJim Jagielski             typelib_typedescription_release(&member->aBase.aBase);
103*b1cdbd2cSJim Jagielski         }
104*b1cdbd2cSJim Jagielski         member = reinterpret_cast< T * >(desc);
105*b1cdbd2cSJim Jagielski     }
106*b1cdbd2cSJim Jagielski     slot.index
107*b1cdbd2cSJim Jagielski         = bridges::cpp_uno::shared::getPrimaryFunctions(
108*b1cdbd2cSJim Jagielski             member->pInterface->pBaseTypeDescription)
109*b1cdbd2cSJim Jagielski         + mapLocalMemberToLocalFunction(member->pInterface, member->nIndex);
110*b1cdbd2cSJim Jagielski     if (member != ifcMember) {
111*b1cdbd2cSJim Jagielski         typelib_typedescription_release(&member->aBase.aBase);
112*b1cdbd2cSJim Jagielski     }
113*b1cdbd2cSJim Jagielski     return slot;
114*b1cdbd2cSJim Jagielski }
115*b1cdbd2cSJim Jagielski 
116*b1cdbd2cSJim Jagielski }
117*b1cdbd2cSJim Jagielski 
118*b1cdbd2cSJim Jagielski namespace bridges { namespace cpp_uno { namespace shared {
119*b1cdbd2cSJim Jagielski 
getLocalFunctions(typelib_InterfaceTypeDescription const * type)120*b1cdbd2cSJim Jagielski sal_Int32 getLocalFunctions(typelib_InterfaceTypeDescription const * type) {
121*b1cdbd2cSJim Jagielski     return type->nMembers == 0
122*b1cdbd2cSJim Jagielski         ? 0
123*b1cdbd2cSJim Jagielski         : (type->nMapFunctionIndexToMemberIndex
124*b1cdbd2cSJim Jagielski            - type->pMapMemberIndexToFunctionIndex[
125*b1cdbd2cSJim Jagielski                type->nAllMembers - type->nMembers]);
126*b1cdbd2cSJim Jagielski }
127*b1cdbd2cSJim Jagielski 
getPrimaryFunctions(typelib_InterfaceTypeDescription * type)128*b1cdbd2cSJim Jagielski sal_Int32 getPrimaryFunctions(typelib_InterfaceTypeDescription * type) {
129*b1cdbd2cSJim Jagielski     sal_Int32 n = 0;
130*b1cdbd2cSJim Jagielski     for (; type != 0; type = type->pBaseTypeDescription) {
131*b1cdbd2cSJim Jagielski         typelib_typedescription_complete(
132*b1cdbd2cSJim Jagielski             reinterpret_cast< typelib_TypeDescription ** >(&type));
133*b1cdbd2cSJim Jagielski         n += getLocalFunctions(type);
134*b1cdbd2cSJim Jagielski     }
135*b1cdbd2cSJim Jagielski     return n;
136*b1cdbd2cSJim Jagielski }
137*b1cdbd2cSJim Jagielski 
getVtableSlot(typelib_InterfaceAttributeTypeDescription const * ifcMember)138*b1cdbd2cSJim Jagielski VtableSlot getVtableSlot(
139*b1cdbd2cSJim Jagielski     typelib_InterfaceAttributeTypeDescription const * ifcMember)
140*b1cdbd2cSJim Jagielski {
141*b1cdbd2cSJim Jagielski     return doGetVtableSlot(ifcMember);
142*b1cdbd2cSJim Jagielski }
143*b1cdbd2cSJim Jagielski 
getVtableSlot(typelib_InterfaceMethodTypeDescription const * ifcMember)144*b1cdbd2cSJim Jagielski VtableSlot getVtableSlot(
145*b1cdbd2cSJim Jagielski     typelib_InterfaceMethodTypeDescription const * ifcMember)
146*b1cdbd2cSJim Jagielski {
147*b1cdbd2cSJim Jagielski     return doGetVtableSlot(ifcMember);
148*b1cdbd2cSJim Jagielski }
149*b1cdbd2cSJim Jagielski 
150*b1cdbd2cSJim Jagielski } } }
151