bump product version to 6.3.0.0.beta1
[LibreOffice.git] / bridges / source / cpp_uno / shared / vtables.cxx
bloba9909eaee5cae73ffc823e84b7afa2af7dfb10d7
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
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>
26 #include <algorithm>
27 #include <cassert>
29 namespace
32 /**
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) {
41 sal_Int32 n = 0;
42 for (sal_Int32 i = 0; i < type->nBaseTypes; ++i) {
43 n += getVtableCount(type->ppBaseTypes[i]);
45 return std::max< sal_Int32 >(n, 1);
48 /**
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&ndash;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
78 // function:
79 template< typename T > bridges::cpp_uno::shared::VtableSlot doGetVtableSlot(
80 T const * ifcMember)
82 bridges::cpp_uno::shared::VtableSlot slot;
83 slot.offset = 0;
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);
93 assert(
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);
100 slot.index
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);
107 return slot;
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) {
123 sal_Int32 n = 0;
124 for (; type != nullptr; type = type->pBaseTypeDescription) {
125 typelib_typedescription_complete(
126 reinterpret_cast< typelib_TypeDescription ** >(&type));
127 n += getLocalFunctions(type);
129 return n;
132 VtableSlot getVtableSlot(
133 typelib_InterfaceAttributeTypeDescription const * ifcMember)
135 return doGetVtableSlot(ifcMember);
138 VtableSlot getVtableSlot(
139 typelib_InterfaceMethodTypeDescription const * ifcMember)
141 return doGetVtableSlot(ifcMember);
144 } } }
146 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */