1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include <vtables.hxx>
23 #include <sal/types.h>
24 #include <typelib/typedescription.h>
33 * Calculates the number of vtables associated with an interface type.
35 * <p>Multiple-inheritance C++ classes have more than one vtable.</p>
37 * @param type a non-null pointer to an interface type description
38 * @return the number of vtables associated with the given interface type
40 sal_Int32
getVtableCount(typelib_InterfaceTypeDescription
const * type
) {
42 for (sal_Int32 i
= 0; i
< type
->nBaseTypes
; ++i
) {
43 n
+= getVtableCount(type
->ppBaseTypes
[i
]);
45 return std::max
< sal_Int32
>(n
, 1);
49 * Maps a local member index to a local function index.
51 * <p><em>Local</em> members/functions are those not inherited from any base
52 * types. The number of <em>functions</em> is potentially larger than the
53 * number of <em>members</em>, as each read–write attribute member counts
54 * as two functions.</p>
56 * @param type a non-null pointer to an interface type description
57 * @param localMember a local member index, relative to the given interface type
58 * @return the local function index corresponding to the given local member
59 * index, relative to the given interface type
61 sal_Int32
mapLocalMemberToLocalFunction(
62 typelib_InterfaceTypeDescription
* type
, sal_Int32 localMember
)
64 typelib_typedescription_complete(
65 reinterpret_cast< typelib_TypeDescription
** >(&type
));
66 sal_Int32 localMemberOffset
= type
->nAllMembers
- type
->nMembers
;
67 sal_Int32 localFunctionOffset
= type
->nMapFunctionIndexToMemberIndex
68 - bridges::cpp_uno::shared::getLocalFunctions(type
);
69 return type
->pMapMemberIndexToFunctionIndex
[localMemberOffset
+ localMember
]
70 - localFunctionOffset
;
73 // Since on Solaris we compile with --instances=static, getVtableSlot cannot be
74 // a template function, with explicit instantiates for
75 // T = typelib_InterfaceAttributeTypeDescription and
76 // T = typelib_InterfaceMethodTypeDescription in this file; hence, there are two
77 // overloaded versions of getVtableSlot that both delegate to this template
79 template< typename T
> bridges::cpp_uno::shared::VtableSlot
doGetVtableSlot(
82 bridges::cpp_uno::shared::VtableSlot slot
;
84 T
* member
= const_cast< T
* >(ifcMember
);
85 while (member
->pBaseRef
!= 0) {
86 assert(member
->nIndex
< member
->pInterface
->nBaseTypes
);
87 for (sal_Int32 i
= 0; i
< member
->nIndex
; ++i
) {
88 slot
.offset
+= getVtableCount(member
->pInterface
->ppBaseTypes
[i
]);
90 typelib_TypeDescription
* desc
= nullptr;
91 typelib_typedescriptionreference_getDescription(
92 &desc
, member
->pBaseRef
);
94 desc
!= nullptr && desc
->eTypeClass
== member
->aBase
.aBase
.eTypeClass
);
95 if (member
!= ifcMember
) {
96 typelib_typedescription_release(&member
->aBase
.aBase
);
98 member
= reinterpret_cast< T
* >(desc
);
101 = bridges::cpp_uno::shared::getPrimaryFunctions(
102 member
->pInterface
->pBaseTypeDescription
)
103 + mapLocalMemberToLocalFunction(member
->pInterface
, member
->nIndex
);
104 if (member
!= ifcMember
) {
105 typelib_typedescription_release(&member
->aBase
.aBase
);
112 namespace bridges
{ namespace cpp_uno
{ namespace shared
{
114 sal_Int32
getLocalFunctions(typelib_InterfaceTypeDescription
const * type
) {
115 return type
->nMembers
== 0
117 : (type
->nMapFunctionIndexToMemberIndex
118 - type
->pMapMemberIndexToFunctionIndex
[
119 type
->nAllMembers
- type
->nMembers
]);
122 sal_Int32
getPrimaryFunctions(typelib_InterfaceTypeDescription
* type
) {
124 for (; type
!= nullptr; type
= type
->pBaseTypeDescription
) {
125 typelib_typedescription_complete(
126 reinterpret_cast< typelib_TypeDescription
** >(&type
));
127 n
+= getLocalFunctions(type
);
132 VtableSlot
getVtableSlot(
133 typelib_InterfaceAttributeTypeDescription
const * ifcMember
)
135 return doGetVtableSlot(ifcMember
);
138 VtableSlot
getVtableSlot(
139 typelib_InterfaceMethodTypeDescription
const * ifcMember
)
141 return doGetVtableSlot(ifcMember
);
146 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */