diff options
Diffstat (limited to 'src/VBox/Main/idl/apiwrap-server.xsl')
-rw-r--r-- | src/VBox/Main/idl/apiwrap-server.xsl | 2592 |
1 files changed, 2592 insertions, 0 deletions
diff --git a/src/VBox/Main/idl/apiwrap-server.xsl b/src/VBox/Main/idl/apiwrap-server.xsl new file mode 100644 index 00000000..baa629ae --- /dev/null +++ b/src/VBox/Main/idl/apiwrap-server.xsl @@ -0,0 +1,2592 @@ +<?xml version="1.0"?> + +<!-- + apiwrap-server.xsl: + XSLT stylesheet that generates C++ API wrappers (server side) from + VirtualBox.xidl. +--> +<!-- + Copyright (C) 2010-2022 Oracle and/or its affiliates. + + This file is part of VirtualBox base platform packages, as + available from https://www.virtualbox.org. + + This program is free software; you can redistribute it and/or + modify it under the terms of the GNU General Public License + as published by the Free Software Foundation, in version 3 of the + License. + + This program is distributed in the hope that it will be useful, but + WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, see <https://www.gnu.org/licenses>. + + SPDX-License-Identifier: GPL-3.0-only +--> + +<xsl:stylesheet + version="1.0" + xmlns:xsl="http://www.w3.org/1999/XSL/Transform" + xmlns:exsl="http://exslt.org/common" + extension-element-prefixes="exsl"> + +<xsl:output method="text"/> + +<xsl:strip-space elements="*"/> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + global XSLT variables + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:variable name="G_xsltFilename" select="'apiwrap-server.xsl'"/> + +<xsl:variable name="G_root" select="/"/> + +<xsl:include href="typemap-shared.inc.xsl"/> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + Keys for more efficiently looking up of types. + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:key name="G_keyEnumsByName" match="//enum[@name]" use="@name"/> +<xsl:key name="G_keyInterfacesByName" match="//interface[@name]" use="@name"/> + +<!-- - - - - - - - - - - - - - - - - - - - - - - +templates for file separation + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="interface" mode="startfile"> + <xsl:param name="file"/> + + <xsl:call-template name="xsltprocNewlineOutputHack"/> + <xsl:value-of select="concat($G_sNewLine, '// ##### BEGINFILE "', $file, '"', $G_sNewLine)"/> +</xsl:template> + +<xsl:template match="interface" mode="endfile"> + <xsl:param name="file"/> + + <xsl:value-of select="concat($G_sNewLine, '// ##### ENDFILE "', $file, '"', $G_sNewLine)"/> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - +templates for file headers/footers + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template name="fileheader"> + <xsl:param name="class"/> + <xsl:param name="name"/> + <xsl:param name="type"/> + + <xsl:text>/** @file +</xsl:text> + <xsl:value-of select="concat(' * VirtualBox API class wrapper ', $type, ' for I', $class, '.')"/> + <xsl:text> + * + * DO NOT EDIT! This is a generated file. + * Generated from: src/VBox/Main/idl/VirtualBox.xidl + * Generator: src/VBox/Main/idl/apiwrap-server.xsl + */ + +/* + * Copyright (C) 2010-2022 Oracle and/or its affiliates. + * + * This file is part of VirtualBox base platform packages, as + * available from https://www.virtualbox.org. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, in version 3 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses>. + * + * SPDX-License-Identifier: GPL-3.0-only + */ + +</xsl:text> +</xsl:template> + +<!-- Emits COM_INTERFACE_ENTRY statements for the current interface node and whatever it inherits from. --> +<xsl:template name="emitCOMInterfaces"> + <xsl:value-of select="concat(' COM_INTERFACE_ENTRY(', @name, ')' , $G_sNewLine)"/> + <!-- now recurse to emit all base interfaces --> + <xsl:variable name="extends" select="@extends"/> + <xsl:if test="$extends and not($extends='$unknown') and not($extends='$errorinfo')"> + <xsl:for-each select="key('G_keyInterfacesByName', $extends)"> + <xsl:call-template name="emitCOMInterfaces"/> + </xsl:for-each> + </xsl:if> +</xsl:template> + +<xsl:template match="interface" mode="classheader"> + <xsl:param name="addinterfaces"/> + <xsl:value-of select="concat('#ifndef ', substring(@name, 2), 'Wrap_H_', $G_sNewLine)"/> + <xsl:value-of select="concat('#define ', substring(@name, 2), 'Wrap_H_')"/> + <xsl:text> +#ifndef RT_WITHOUT_PRAGMA_ONCE +# pragma once +#endif + +#include "VirtualBoxBase.h" +#include "Wrapper.h" + +</xsl:text> + <xsl:value-of select="concat('class ATL_NO_VTABLE ', substring(@name, 2), 'Wrap')"/> + <xsl:text> + : public VirtualBoxBase +</xsl:text> + <xsl:value-of select="concat(' , VBOX_SCRIPTABLE_IMPL(', @name, ')')"/> + <xsl:value-of select="$G_sNewLine"/> + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <xsl:value-of select="concat(' , VBOX_SCRIPTABLE_IMPL(', text(), ')')"/> + <xsl:value-of select="$G_sNewLine"/> + </xsl:for-each> + <xsl:text>{ + Q_OBJECT + +public: +</xsl:text> + <xsl:value-of select="concat(' VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT(', substring(@name, 2), 'Wrap, ', @name, ')', $G_sNewLine)"/> + <xsl:value-of select="concat(' DECLARE_NOT_AGGREGATABLE(', substring(@name, 2), 'Wrap)', $G_sNewLine)"/> + <xsl:text> DECLARE_PROTECT_FINAL_CONSTRUCT() + +</xsl:text> + <xsl:value-of select="concat(' BEGIN_COM_MAP(', substring(@name, 2), 'Wrap)', $G_sNewLine)"/> + <xsl:text> COM_INTERFACE_ENTRY(ISupportErrorInfo) +</xsl:text> + <xsl:call-template name="emitCOMInterfaces"/> + <xsl:value-of select="concat(' COM_INTERFACE_ENTRY2(IDispatch, ', @name, ')', $G_sNewLine)"/> + <xsl:variable name="manualAddInterfaces"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'manualaddinterfaces'"/> + </xsl:call-template> + </xsl:variable> + <xsl:if test="not($manualAddInterfaces = 'true')"> + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <!-- This is super tricky, as the for-each switches to the node set, + which means the normal document isn't available any more. We get + the data we need, uses a for-each to switch document and then a + key() to look up the interface by name. --> + <xsl:variable name="addifname"> + <xsl:value-of select="string(.)"/> + </xsl:variable> + <xsl:for-each select="$G_root"> + <xsl:for-each select="key('G_keyInterfacesByName', $addifname)"> + <xsl:call-template name="emitCOMInterfaces"/> + </xsl:for-each> + </xsl:for-each> + </xsl:for-each> + </xsl:if> + <xsl:value-of select="concat(' VBOX_TWEAK_INTERFACE_ENTRY(', @name, ')', $G_sNewLine)"/> + <xsl:text> END_COM_MAP() + +</xsl:text> + <xsl:value-of select="concat(' DECLARE_COMMON_CLASS_METHODS(', substring(@name, 2), 'Wrap)', $G_sNewLine)"/> +</xsl:template> + +<xsl:template match="interface" mode="classfooter"> + <xsl:param name="addinterfaces"/> + <xsl:if test="@wrap-gen-hook = 'yes'"> + <xsl:text> +public: + virtual void i_callHook(const char *a_pszFunction) { RT_NOREF_PV(a_pszFunction); } +</xsl:text> + </xsl:if> + <xsl:text> +private: + DECLARE_CLS_COPY_CTOR_ASSIGN_NOOP(</xsl:text> + <xsl:value-of select="concat(substring(@name, 2),'Wrap')"/> + <xsl:text>); /* Shuts up MSC warning C4625. */ + +}; + +</xsl:text> + <xsl:value-of select="concat('#endif // !', substring(@name, 2), 'Wrap_H_', $G_sNewLine)"/> +</xsl:template> + +<xsl:template match="interface" mode="codeheader"> + <xsl:param name="addinterfaces"/> + <xsl:value-of select="concat('#define LOG_GROUP LOG_GROUP_MAIN_', translate(substring(@name, 2), $G_lowerCase, $G_upperCase), $G_sNewLine, $G_sNewLine)"/> + <xsl:value-of select="concat('#include "', substring(@name, 2), 'Wrap.h"', $G_sNewLine)"/> + <xsl:text>#include "LoggingNew.h" +#ifdef VBOX_WITH_DTRACE_R3_MAIN +# include "dtrace/VBoxAPI.h" +#endif + +</xsl:text> +</xsl:template> + +<xsl:template name="emitISupports"> + <xsl:param name="classname"/> + <xsl:param name="extends"/> + <xsl:param name="addinterfaces"/> + <xsl:param name="depth"/> + <xsl:param name="interfacelist"/> + + <xsl:choose> + <xsl:when test="$extends and not($extends='$unknown') and not($extends='$errorinfo')"> + <xsl:variable name="newextends" select="key('G_keyInterfacesByName', $extends)/@extends"/> + <xsl:variable name="newiflist" select="concat($interfacelist, ', ', $extends)"/> + <xsl:call-template name="emitISupports"> + <xsl:with-param name="classname" select="$classname"/> + <xsl:with-param name="extends" select="$newextends"/> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + <xsl:with-param name="depth" select="$depth + 1"/> + <xsl:with-param name="interfacelist" select="$newiflist"/> + </xsl:call-template> + </xsl:when> + <xsl:otherwise> + <xsl:variable name="addinterfaces_ns" select="exsl:node-set($addinterfaces)"/> + <xsl:choose> + <xsl:when test="count($addinterfaces_ns/token) > 0"> + <xsl:variable name="addifname" select="$addinterfaces_ns/token[1]"/> + <xsl:variable name="addif" select="key('G_keyInterfacesByName', $addifname)/@extends"/> + <xsl:variable name="newextends" select="$addif/@extends"/> + <xsl:variable name="newaddinterfaces" select="$addinterfaces_ns/token[position() > 1]"/> + <xsl:variable name="newiflist" select="concat($interfacelist, ', ', $addifname)"/> + <xsl:call-template name="emitISupports"> + <xsl:with-param name="classname" select="$classname"/> + <xsl:with-param name="extends" select="$newextends"/> + <xsl:with-param name="addinterfaces" select="$newaddinterfaces"/> + <xsl:with-param name="depth" select="$depth + 1"/> + <xsl:with-param name="interfacelist" select="$newiflist"/> + </xsl:call-template> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="concat('NS_IMPL_THREADSAFE_ISUPPORTS', $depth, '_CI(', $classname, ', ', $interfacelist, ')', $G_sNewLine)"/> + </xsl:otherwise> + </xsl:choose> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<xsl:template match="interface" mode="codefooter"> + <xsl:param name="addinterfaces"/> + <xsl:text>#ifdef VBOX_WITH_XPCOM +</xsl:text> + <xsl:value-of select="concat('NS_DECL_CLASSINFO(', substring(@name, 2), 'Wrap)', $G_sNewLine)"/> + + <xsl:variable name="manualAddInterfaces"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'manualaddinterfaces'"/> + </xsl:call-template> + </xsl:variable> + <xsl:choose> + <xsl:when test="$manualAddInterfaces = 'true'"> + <xsl:variable name="nulladdinterfaces"></xsl:variable> + <xsl:call-template name="emitISupports"> + <xsl:with-param name="classname" select="concat(substring(@name, 2), 'Wrap')"/> + <xsl:with-param name="extends" select="@extends"/> + <xsl:with-param name="addinterfaces" select="$nulladdinterfaces"/> + <xsl:with-param name="depth" select="1"/> + <xsl:with-param name="interfacelist" select="@name"/> + </xsl:call-template> + </xsl:when> + <xsl:otherwise> + <xsl:call-template name="emitISupports"> + <xsl:with-param name="classname" select="concat(substring(@name, 2), 'Wrap')"/> + <xsl:with-param name="extends" select="@extends"/> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + <xsl:with-param name="depth" select="1"/> + <xsl:with-param name="interfacelist" select="@name"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> + <xsl:text>#endif // VBOX_WITH_XPCOM +</xsl:text> +</xsl:template> + + +<!-- - - - - - - - - - - - - - - - - - - - - - - + templates for dealing with names and parameters + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template name="tospace"> + <xsl:param name="str"/> + <xsl:value-of select="translate($str, 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_', ' ')"/> +</xsl:template> + +<xsl:template name="checkoption"> + <xsl:param name="optionlist"/> + <xsl:param name="option"/> + <xsl:value-of select="string-length($option) > 0 and contains(concat(',', translate($optionlist, ' ', ''), ','), concat(',', $option, ','))"/> +</xsl:template> + +<xsl:template name="getattrlist"> + <xsl:param name="val"/> + <xsl:param name="separator" select="','"/> + + <xsl:if test="$val and $val != ''"> + <xsl:choose> + <xsl:when test="contains($val,$separator)"> + <token> + <xsl:value-of select="substring-before($val,$separator)"/> + </token> + <xsl:call-template name="getattrlist"> + <xsl:with-param name="val" select="substring-after($val,$separator)"/> + <xsl:with-param name="separator" select="$separator"/> + </xsl:call-template> + </xsl:when> + <xsl:otherwise> + <token><xsl:value-of select="$val"/></token> + </xsl:otherwise> + </xsl:choose> + </xsl:if> +</xsl:template> + +<xsl:template name="translatepublictype"> + <xsl:param name="type"/> + <xsl:param name="dir"/> + <xsl:param name="mod"/> + + <xsl:choose> + <xsl:when test="$type='wstring' or $type='uuid'"> + <xsl:if test="$dir='in'"> + <xsl:text>IN_</xsl:text> + </xsl:if> + <xsl:text>BSTR</xsl:text> + </xsl:when> + + <xsl:when test="$type='$unknown'"> + <xsl:text>IUnknown *</xsl:text> + </xsl:when> + + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:value-of select="concat($type, ' *')"/> + </xsl:when> + + <xsl:when test="count(key('G_keyEnumsByName', $type)) > 0"> + <xsl:value-of select="concat($type, '_T')"/> + </xsl:when> + + <!-- Micro optimizations: Put off wraptypefield calculation as long as possible; Check interfaces before enums. --> + <xsl:otherwise> + <!-- get C++ glue type from IDL type from table in typemap-shared.inc.xsl --> + <xsl:variable name="gluetypefield" select="exsl:node-set($G_aSharedTypes)/type[@idlname=$type]/@gluename"/> + <xsl:choose> + <xsl:when test="string-length($gluetypefield)"> + <xsl:value-of select="$gluetypefield"/> + </xsl:when> + + <xsl:otherwise> + <xsl:call-template name="fatalError"> + <xsl:with-param name="msg" select="concat('translatepublictype: Type "', $type, '" is not supported.')"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> + </xsl:otherwise> + </xsl:choose> + <xsl:if test="$mod='ptr'"> + <xsl:text> *</xsl:text> + </xsl:if> +</xsl:template> + +<xsl:template name="translatewrappedtype"> + <xsl:param name="type"/> + <xsl:param name="dir"/> + <xsl:param name="mod"/> + <xsl:param name="safearray"/> + + <xsl:choose> + <xsl:when test="$type='wstring'"> + <xsl:if test="$dir='in' and not($safearray='yes')"> + <xsl:text>const </xsl:text> + </xsl:if> + <xsl:text>com::Utf8Str &</xsl:text> + </xsl:when> + + <xsl:when test="$type='uuid'"> + <xsl:if test="$dir='in'"> + <xsl:text>const </xsl:text> + </xsl:if> + <xsl:text>com::Guid &</xsl:text> + </xsl:when> + + <xsl:when test="$type='$unknown'"> + <xsl:if test="$dir='in' and not($safearray='yes')"> + <xsl:text>const </xsl:text> + </xsl:if> + <xsl:text>ComPtr<IUnknown> &</xsl:text> + </xsl:when> + + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:if test="$dir='in' and not($safearray='yes')"> + <xsl:text>const </xsl:text> + </xsl:if> + <xsl:value-of select="concat('ComPtr<', $type, '> &')"/> + </xsl:when> + + <xsl:when test="count(key('G_keyEnumsByName', $type)) > 0"> + <xsl:value-of select="concat($type, '_T')"/> + </xsl:when> + + <!-- Micro optimizations: Put off wraptypefield calculation as long as possible; Check interfaces before enums. --> + <xsl:otherwise> + <!-- get C++ wrap type from IDL type from table in typemap-shared.inc.xsl --> + <xsl:variable name="wraptypefield" select="exsl:node-set($G_aSharedTypes)/type[@idlname=$type]/@gluename"/> + <xsl:choose> + <xsl:when test="string-length($wraptypefield)"> + <xsl:value-of select="$wraptypefield"/> + </xsl:when> + + <xsl:otherwise> + <xsl:call-template name="fatalError"> + <xsl:with-param name="msg" select="concat('translatewrappedtype: Type "', $type, '" is not supported.')"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> + </xsl:otherwise> + </xsl:choose> + <xsl:if test="$mod='ptr'"> + <xsl:text> *</xsl:text> + </xsl:if> +</xsl:template> + +<xsl:template name="translatefmtspectype"> + <xsl:param name="type"/> + <xsl:param name="dir"/> + <xsl:param name="mod"/> + <xsl:param name="safearray"/> + <xsl:param name="isref"/> + + <!-- get C format string for IDL type from table in typemap-shared.inc.xsl --> + <xsl:variable name="wrapfmt" select="exsl:node-set($G_aSharedTypes)/type[@idlname=$type]/@gluefmt"/> + <xsl:choose> + <xsl:when test="$mod='ptr' or ($isref='yes' and $dir!='in')"> + <xsl:text>%p</xsl:text> + </xsl:when> + <xsl:when test="$safearray='yes'"> + <xsl:text>%zu</xsl:text> + </xsl:when> + <xsl:when test="string-length($wrapfmt)"> + <xsl:value-of select="$wrapfmt"/> + </xsl:when> + <xsl:when test="$type='$unknown'"> + <xsl:text>%p</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyEnumsByName', $type)) > 0"> + <xsl:text>%RU32</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>%p</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:call-template name="fatalError"> + <xsl:with-param name="msg" select="concat('translatefmtcpectype: Type "', $type, '" is not supported.')"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<xsl:template name="translatedtracetype"> + <xsl:param name="type"/> + <xsl:param name="dir"/> + <xsl:param name="mod"/> + + <!-- get dtrace probe type from IDL type from table in typemap-shared.inc.xsl --> + <xsl:variable name="dtracetypefield" select="exsl:node-set($G_aSharedTypes)/type[@idlname=$type]/@dtracename"/> + <xsl:choose> + <xsl:when test="string-length($dtracetypefield)"> + <xsl:value-of select="$dtracetypefield"/> + </xsl:when> + <xsl:when test="$type='$unknown'"> + <!-- <xsl:text>struct IUnknown *</xsl:text> --> + <xsl:text>void *</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyEnumsByName', $type)) > 0"> + <!-- <xsl:value-of select="concat($type, '_T')"/> - later we can emit enums into dtrace the library --> + <xsl:text>int</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <!-- + <xsl:value-of select="concat('struct ', $type, ' *')"/> + --> + <xsl:text>void *</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:call-template name="fatalError"> + <xsl:with-param name="msg" select="concat('translatedtracetype Type "', $type, '" is not supported.')"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> + <xsl:if test="$mod='ptr'"> + <xsl:text> *</xsl:text> + </xsl:if> +</xsl:template> + + +<!-- - - - - - - - - - - - - - - - - - - - - - - + templates for handling entire interfaces and their contents + - - - - - - - - - - - - - - - - - - - - - - --> + +<!-- Called on interface node. --> +<xsl:template name="emitInterface"> + <!-- sources, headers and dtrace-probes all needs attribute lists --> + <xsl:variable name="addinterfaces"> + <xsl:call-template name="getattrlist"> + <xsl:with-param name="val" select="@wrap-hint-server-addinterfaces"/> + </xsl:call-template> + </xsl:variable> + + <!-- interface sanity check, prevents crashes --> + <xsl:if test="(count(attribute) + count(method) + sum(@reservedMethods[number()= number()]) + sum(@reservedAttributes[number()= number()])) = 0"> + <xsl:message terminate="yes"> + Interface <xsl:value-of select="@name"/> is empty which causes midl generated proxy + stubs to crash. Please add a dummy:<xsl:value-of select="$G_sNewLine"/> + <attribute name="midlDoesNotLikeEmptyInterfaces" readonly="yes" type="boolean"/> + </xsl:message> + </xsl:if> + + <xsl:choose> + <xsl:when test="$generating = 'sources'"> + <xsl:if test="(position() mod 2) = $reminder"> + <xsl:call-template name="emitCode"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:call-template> + </xsl:if> + </xsl:when> + <xsl:when test="$generating = 'headers'"> + <xsl:call-template name="emitHeader"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:call-template> + </xsl:when> + <xsl:when test="$generating = 'dtrace-probes'"> + <xsl:call-template name="emitDTraceProbes"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:call-template> + </xsl:when> + <xsl:otherwise><xsl:message terminate="yes">Otherwise oops in emitInterface</xsl:message></xsl:otherwise> + </xsl:choose> +</xsl:template> + +<!-- Called on a method param or attribute node. --> +<xsl:template name="emitPublicParameter"> + <xsl:param name="dir"/> + + <xsl:variable name="gluetype"> + <xsl:call-template name="translatepublictype"> + <xsl:with-param name="type" select="@type"/> + <xsl:with-param name="dir" select="$dir"/> + <xsl:with-param name="mod" select="@mod"/> + </xsl:call-template> + </xsl:variable> + + <xsl:choose> + <xsl:when test="@safearray='yes'"> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>ComSafeArrayIn(</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>ComSafeArrayOut(</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:value-of select="$gluetype"/> + <xsl:text>, a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:text>)</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="$gluetype"/> + <xsl:if test="substring($gluetype,string-length($gluetype))!='*'"> + <xsl:text> </xsl:text> + </xsl:if> + <xsl:if test="$dir != 'in'"> + <xsl:text>*</xsl:text> + </xsl:if> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<xsl:template match="attribute/@type | param/@type" mode="wrapped"> + <xsl:param name="dir"/> + + <xsl:variable name="wraptype"> + <xsl:call-template name="translatewrappedtype"> + <xsl:with-param name="type" select="."/> + <xsl:with-param name="dir" select="$dir"/> + <xsl:with-param name="mod" select="../@mod"/> + <xsl:with-param name="safearray" select="../@safearray"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="lastchar"> + <xsl:value-of select="substring($wraptype, string-length($wraptype))"/> + </xsl:variable> + + <xsl:choose> + <xsl:when test="../@safearray='yes'"> + <xsl:if test="$dir='in'"> + <xsl:text>const </xsl:text> + </xsl:if> + <xsl:text>std::vector<</xsl:text> + <xsl:choose> + <xsl:when test="$lastchar = '&'"> + <xsl:variable name="wraptype2"> + <xsl:value-of select="substring($wraptype, 1, string-length($wraptype)-2)"/> + </xsl:variable> + <xsl:value-of select="$wraptype2"/> + <xsl:if test="substring($wraptype2,string-length($wraptype2)) = '>'"> + <xsl:text> </xsl:text> + </xsl:if> + </xsl:when> + <xsl:when test="lastchar = '>'"> + <xsl:value-of select="concat($wraptype, ' ')"/> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="$wraptype"/> + </xsl:otherwise> + </xsl:choose> + <xsl:text>> &</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="$wraptype"/> + <xsl:if test="$lastchar != '&'"> + <xsl:if test="$lastchar != '*'"> + <xsl:text> </xsl:text> + </xsl:if> + <xsl:if test="$dir != 'in'"> + <xsl:text>*</xsl:text> + </xsl:if> + </xsl:if> + </xsl:otherwise> + </xsl:choose> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="attribute/@type | param/@type" mode="logparamtext"> + <xsl:param name="dir"/> + <xsl:param name="isref"/> + + <xsl:if test="$isref!='yes' and ($dir='out' or $dir='ret')"> + <xsl:text>*</xsl:text> + </xsl:if> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + <xsl:text>=</xsl:text> + <xsl:call-template name="translatefmtspectype"> + <xsl:with-param name="type" select="."/> + <xsl:with-param name="dir" select="$dir"/> + <xsl:with-param name="mod" select="../@mod"/> + <xsl:with-param name="safearray" select="../@safearray"/> + <xsl:with-param name="isref" select="$isref"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="attribute/@type | param/@type" mode="logparamval"> + <xsl:param name="dir"/> + <xsl:param name="isref"/> + + <xsl:choose> + <xsl:when test="../@safearray='yes' and $isref!='yes'"> + <xsl:text>ComSafeArraySize(</xsl:text> + <xsl:if test="$isref!='yes' and $dir!='in'"> + <xsl:text>*</xsl:text> + </xsl:if> + </xsl:when> + <xsl:when test="$isref!='yes' and $dir!='in'"> + <xsl:text>*</xsl:text> + </xsl:when> + </xsl:choose> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + <xsl:choose> + <xsl:when test="../@safearray='yes' and $isref!='yes'"> + <xsl:text>)</xsl:text> + </xsl:when> + </xsl:choose> +</xsl:template> + +<!-- Emits the DTrace probe parameter value (using tmps), invoked on param or attribute node. --> +<xsl:template name="emitDTraceParamValue"> + <xsl:param name="dir"/> + + <xsl:variable name="viatmpvar"> + <xsl:for-each select="@type"> + <xsl:call-template name="paramconversionviatmp"> + <xsl:with-param name="dir" select="$dir"/> + </xsl:call-template> + </xsl:for-each> + </xsl:variable> + + <xsl:variable name="type" select="@type"/> + <xsl:choose> + <!-- Doesn't help to inline paramconversionviatmp: <xsl:when test="$type = 'wstring' or $type = '$unknown' or $type = 'uuid' or @safearray = 'yes' or count(key('G_keyInterfacesByName', $type)) > 0"> --> + <xsl:when test="$viatmpvar = 'yes'"> + <xsl:variable name="tmpname"> + <xsl:text>Tmp</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + + <xsl:choose> + <xsl:when test="@safearray = 'yes'"> + <xsl:text>(uint32_t)</xsl:text> + <xsl:value-of select="$tmpname"/> + <xsl:text>.array().size(), </xsl:text> + <!-- Later: + <xsl:value-of select="concat($tmpname, '.array().data(), ')"/> + --> + <xsl:text>NULL /*for now*/</xsl:text> + </xsl:when> + <xsl:when test="$type = 'wstring'"> + <xsl:value-of select="$tmpname"/> + <xsl:text>.str().c_str()</xsl:text> + </xsl:when> + <xsl:when test="$type = 'uuid'"> + <xsl:value-of select="$tmpname"/> + <xsl:text>.uuid().toStringCurly().c_str()</xsl:text> + </xsl:when> + <xsl:when test="$type = '$unknown'"> + <xsl:text>(void *)</xsl:text> + <xsl:value-of select="$tmpname"/> + <xsl:text>.ptr()</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>(void *)</xsl:text> + <xsl:value-of select="$tmpname"/> + <xsl:text>.ptr()</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="$tmpname"/> + </xsl:otherwise> + </xsl:choose> + </xsl:when> + + <xsl:otherwise> + <xsl:if test="$dir != 'in'"> + <xsl:text>*</xsl:text> + </xsl:if> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + + <xsl:if test="$type = 'boolean'"> + <xsl:text> != FALSE</xsl:text> + </xsl:if> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<!-- +Same as emitDTraceParamValue except no temporary variables are used (they are out of scope). +Note! There are two other instances of this code with different @dir values, see below. +--> +<xsl:template name="emitDTraceParamValNoTmp"> + <!-- To speed this up, the logic of paramconversionviatmp has been duplicated/inlined here. --> + <xsl:variable name="type" select="@type"/> + <xsl:choose> + <xsl:when test="@safearray = 'yes'"> + <xsl:text>0, 0</xsl:text> + </xsl:when> + <xsl:when test="$type = 'wstring' or $type = '$unknown' or $type = 'uuid' or count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>0</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:if test="@dir != 'in'"> + <xsl:text>*</xsl:text> + </xsl:if> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:if test="$type = 'boolean'"> + <xsl:text> != FALSE</xsl:text> + </xsl:if> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<!-- Copy of emitDTraceParamValNoTmp with @dir = 'in' for speeding up the code (noticable difference). --> +<xsl:template name="emitDTraceParamValNoTmp-DirIn"> + <xsl:variable name="type" select="@type"/> + <xsl:choose> + <xsl:when test="@safearray = 'yes'"> + <xsl:text>0, 0</xsl:text> + </xsl:when> + <xsl:when test="$type = 'wstring' or $type = '$unknown' or $type = 'uuid' or count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>0</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:if test="$type = 'boolean'"> + <xsl:text> != FALSE</xsl:text> + </xsl:if> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<!-- Copy of emitDTraceParamValNoTmp with @dir != 'in' for speeding up attributes (noticable difference). --> +<xsl:template name="emitDTraceParamValNoTmp-DirNotIn"> + <xsl:variable name="type" select="@type"/> + <xsl:choose> + <xsl:when test="@safearray = 'yes'"> + <xsl:text>0, 0</xsl:text> + </xsl:when> + <xsl:when test="$type = 'wstring' or $type = '$unknown' or $type = 'uuid' or count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>0</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>*a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:if test="$type = 'boolean'"> + <xsl:text> != FALSE</xsl:text> + </xsl:if> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +<xsl:template match="attribute/@type | param/@type" mode="dtraceparamdecl"> + <xsl:param name="dir"/> + + <xsl:variable name="gluetype"> + <xsl:call-template name="translatedtracetype"> + <xsl:with-param name="type" select="."/> + <xsl:with-param name="dir" select="$dir"/> + <xsl:with-param name="mod" select="../@mod"/> + </xsl:call-template> + </xsl:variable> + + <!-- Safe arrays get an extra size parameter. --> + <xsl:if test="../@safearray='yes'"> + <xsl:text>uint32_t a_c</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + <xsl:text>, </xsl:text> + </xsl:if> + + <xsl:value-of select="$gluetype"/> + <xsl:choose> + <xsl:when test="../@safearray='yes'"> + <xsl:text> *a_pa</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:if test="substring($gluetype,string-length($gluetype))!='*'"> + <xsl:text> </xsl:text> + </xsl:if> + <xsl:text>a_</xsl:text> + </xsl:otherwise> + </xsl:choose> + + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> +</xsl:template> + +<!-- Call this to determine whether a temporary conversion variable is used for the current parameter. +Returns empty if not needed, non-empty ('yes') if needed. --> +<xsl:template name="paramconversionviatmp"> + <xsl:param name="dir"/> + <xsl:variable name="type" select="."/> + <xsl:choose> + <xsl:when test="$type = 'wstring' or $type = '$unknown' or $type = 'uuid'"> + <xsl:text>yes</xsl:text> + </xsl:when> + <xsl:when test="../@safearray = 'yes'"> + <xsl:text>yes</xsl:text> + </xsl:when> + <xsl:when test="$type = 'boolean' or $type = 'long' or $type = 'long' or $type = 'long long'"/> <!-- XXX: drop this? --> + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>yes</xsl:text> + </xsl:when> + </xsl:choose> +</xsl:template> + +<!-- Call this to get the argument conversion class, if any is needed. --> +<xsl:template name="paramconversionclass"> + <xsl:param name="dir"/> + + <xsl:variable name="type" select="."/> + <xsl:choose> + <xsl:when test="$type='$unknown'"> + <xsl:if test="../@safearray='yes'"> + <xsl:text>Array</xsl:text> + </xsl:if> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>ComTypeInConverter<IUnknown></xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>ComTypeOutConverter<IUnknown></xsl:text> + </xsl:otherwise> + </xsl:choose> + </xsl:when> + + <xsl:when test="$type='wstring'"> + <xsl:if test="../@safearray='yes'"> + <xsl:text>Array</xsl:text> + </xsl:if> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>BSTRInConverter</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>BSTROutConverter</xsl:text> + </xsl:otherwise> + </xsl:choose> + </xsl:when> + + <xsl:when test="$type='uuid'"> + <xsl:if test="../@safearray='yes'"> + <xsl:text>Array</xsl:text> + </xsl:if> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>UuidInConverter</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>UuidOutConverter</xsl:text> + </xsl:otherwise> + </xsl:choose> + </xsl:when> + + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:if test="../@safearray='yes'"> + <xsl:text>Array</xsl:text> + </xsl:if> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>ComTypeInConverter</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>ComTypeOutConverter</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:value-of select="concat('<', $type, '>')"/> + </xsl:when> + + <xsl:when test="../@safearray='yes'"> + <xsl:text>Array</xsl:text> + <xsl:choose> + <xsl:when test="$dir='in'"> + <xsl:text>InConverter</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>OutConverter</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:variable name="gluetype"> + <xsl:call-template name="translatepublictype"> + <xsl:with-param name="type" select="."/> + <xsl:with-param name="dir" select="$dir"/> + <xsl:with-param name="mod" select="../@mod"/> + </xsl:call-template> + </xsl:variable> + <xsl:value-of select="concat('<', $gluetype, '>')"/> + </xsl:when> + </xsl:choose> +</xsl:template> + +<!-- Emits code for converting the parameter to a temporary variable. --> +<xsl:template match="attribute/@type | param/@type" mode="paramvalconversion2tmpvar"> + <xsl:param name="dir"/> + + <xsl:variable name="conversionclass"> + <xsl:call-template name="paramconversionclass"> + <xsl:with-param name="dir" select="$dir"/> + </xsl:call-template> + </xsl:variable> + + <xsl:if test="$conversionclass != ''"> + <xsl:value-of select="$conversionclass"/> + <xsl:text> Tmp</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + <xsl:text>(</xsl:text> + <xsl:if test="../@safearray = 'yes'"> + <xsl:choose> + <xsl:when test="$dir = 'in'"> + <xsl:text>ComSafeArrayInArg(</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>ComSafeArrayOutArg(</xsl:text> + </xsl:otherwise> + </xsl:choose> + </xsl:if> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + <xsl:if test="../@safearray = 'yes'"> + <xsl:text>)</xsl:text> + </xsl:if> + <xsl:text>);</xsl:text> + </xsl:if> + +</xsl:template> + +<!-- Partner to paramvalconversion2tmpvar that emits the parameter when calling call the internal worker method. --> +<xsl:template match="attribute/@type | param/@type" mode="paramvalconversionusingtmp"> + <xsl:param name="dir"/> + + <xsl:variable name="viatmpvar"> + <xsl:call-template name="paramconversionviatmp"> + <xsl:with-param name="dir" select="$dir"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="type" select="."/> + + <xsl:choose> + <xsl:when test="$viatmpvar = 'yes'"> + <xsl:text>Tmp</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + + <xsl:choose> + <xsl:when test="../@safearray='yes'"> + <xsl:text>.array()</xsl:text> + </xsl:when> + <xsl:when test="$type = 'wstring'"> + <xsl:text>.str()</xsl:text> + </xsl:when> + <xsl:when test="$type = 'uuid'"> + <xsl:text>.uuid()</xsl:text> + </xsl:when> + <xsl:when test="$type = '$unknown'"> + <xsl:text>.ptr()</xsl:text> + </xsl:when> + <xsl:when test="count(key('G_keyInterfacesByName', $type)) > 0"> + <xsl:text>.ptr()</xsl:text> + </xsl:when> + <xsl:otherwise><xsl:message terminate="yes">Oops #1</xsl:message></xsl:otherwise> + </xsl:choose> + </xsl:when> + + <xsl:otherwise> + <xsl:text>a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="../@name"/> + </xsl:call-template> + + <!-- Make sure BOOL values we pass down are either TRUE or FALSE. --> + <xsl:if test="$type = 'boolean' and $dir = 'in'"> + <xsl:text> != FALSE</xsl:text> + </xsl:if> + </xsl:otherwise> + </xsl:choose> + +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit attribute + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="attribute" mode="public"> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="attrbasename"> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + + <xsl:value-of select="concat(' STDMETHOD(COMGETTER(', $attrbasename, '))(')"/> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir">out</xsl:with-param> + </xsl:call-template> + <xsl:text>); +</xsl:text> + + <xsl:if test="not(@readonly) or @readonly!='yes'"> + <xsl:value-of select="concat(' STDMETHOD(COMSETTER(', $attrbasename, '))(')"/> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir">in</xsl:with-param> + </xsl:call-template> + <xsl:text>); +</xsl:text> + </xsl:if> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="attribute" mode="wrapped"> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="attrbasename"> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + + <xsl:if test="$attrbasename = 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> //</xsl:text> + </xsl:if> + + <xsl:value-of select="concat(' virtual HRESULT get', $attrbasename, '(')"/> + <xsl:variable name="passAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'passcaller'"/> + </xsl:call-template> + </xsl:variable> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>AutoCaller &aAutoCaller, </xsl:text> + </xsl:if> + <xsl:apply-templates select="@type" mode="wrapped"> + <xsl:with-param name="dir" select="'out'"/> + </xsl:apply-templates> + <xsl:text>) = 0; +</xsl:text> + + <xsl:if test="not(@readonly) or @readonly!='yes'"> + <xsl:value-of select="concat(' virtual HRESULT set', $attrbasename, '(')"/> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>AutoCaller &aAutoCaller, </xsl:text> + </xsl:if> + <xsl:apply-templates select="@type" mode="wrapped"> + <xsl:with-param name="dir" select="'in'"/> + </xsl:apply-templates> + <xsl:text>) = 0; +</xsl:text> + </xsl:if> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="attribute" mode="code"> + <xsl:param name="topclass"/> + <xsl:param name="dtracetopclass"/> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="attrbasename"> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="limitedAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'limitedcaller'"/> + </xsl:call-template> + </xsl:variable> + + <xsl:variable name="dtraceattrname"> + <xsl:choose> + <xsl:when test="@dtracename"> + <xsl:value-of select="@dtracename"/> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="$attrbasename"/> + </xsl:otherwise> + </xsl:choose> + </xsl:variable> + + <xsl:value-of select="concat('STDMETHODIMP ', $topclass, 'Wrap::COMGETTER(', $attrbasename, ')(')"/> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir">out</xsl:with-param> + </xsl:call-template> + <xsl:text>) +{</xsl:text> + <xsl:if test="$attrbasename = 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#if 0 /* This is a dummy attribute */</xsl:text> + </xsl:if> + <xsl:text> + LogRelFlow(("{%p} %s: enter </xsl:text> + <xsl:apply-templates select="@type" mode="logparamtext"> + <xsl:with-param name="dir" select="'out'"/> + <xsl:with-param name="isref" select="'yes'"/> + </xsl:apply-templates> + <xsl:text>\n", this, </xsl:text> + <xsl:value-of select="concat('"', $topclass, '::get', $attrbasename, '", ')"/> + <xsl:apply-templates select="@type" mode="logparamval"> + <xsl:with-param name="dir" select="'out'"/> + <xsl:with-param name="isref" select="'yes'"/> + </xsl:apply-templates> + <xsl:text>)); +</xsl:text> + <xsl:if test="ancestor::interface[@wrap-gen-hook = 'yes']"> + <xsl:text> + i_callHook(__FUNCTION__);</xsl:text> + </xsl:if> +<xsl:text> + // Clear error info, to make in-process calls behave the same as + // cross-apartment calls or out-of-process calls. + VirtualBoxBase::clearError(); + + HRESULT hrc; + + try + { + CheckComArgOutPointerValidThrow(a</xsl:text> + <xsl:value-of select="$attrbasename"/> + <xsl:text>); + </xsl:text> + <xsl:apply-templates select="@type" mode="paramvalconversion2tmpvar"> + <xsl:with-param name="dir" select="'out'"/> + </xsl:apply-templates> + <xsl:if test="$attrbasename != 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_GET_', $dtraceattrname, '_ENTER('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this); +#endif</xsl:text> + </xsl:if> + <xsl:text> + </xsl:text> + <xsl:choose> + <xsl:when test="$limitedAutoCaller = 'true'"> + <xsl:text>AutoLimitedCaller</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>AutoCaller</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:text> autoCaller(this); + hrc = autoCaller.rc(); + if (SUCCEEDED(hrc)) + { +</xsl:text> + <xsl:value-of select="concat(' hrc = get', $attrbasename, '(')"/> + <xsl:variable name="passAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'passcaller'"/> + </xsl:call-template> + </xsl:variable> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>autoCaller, </xsl:text> + </xsl:if> + <xsl:apply-templates select="@type" mode="paramvalconversionusingtmp"> + <xsl:with-param name="dir" select="'out'"/> + </xsl:apply-templates> + <xsl:text>); + }</xsl:text> + <xsl:if test="$attrbasename != 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_GET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 0 /*normal*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValue"> + <xsl:with-param name="dir">out</xsl:with-param> + </xsl:call-template> + <xsl:text>); +#endif</xsl:text> + </xsl:if> + <xsl:text> + } + catch (HRESULT hrc2) + { + hrc = hrc2;</xsl:text> + <xsl:if test="$attrbasename != 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_GET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 1 /*hrc exception*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp-DirNotIn"/> + <xsl:text>); +#endif</xsl:text> + </xsl:if> + <xsl:text> + } + catch (...) + { + hrc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS);</xsl:text> + <xsl:if test="$attrbasename != 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_GET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 9 /*unhandled exception*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp-DirNotIn"/> + <xsl:text>); +#endif</xsl:text> + </xsl:if> + <xsl:text> + } + + LogRelFlow(("{%p} %s: leave </xsl:text> + <xsl:apply-templates select="@type" mode="logparamtext"> + <xsl:with-param name="dir" select="'out'"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + <xsl:text> hrc=%Rhrc\n", this, </xsl:text> + <xsl:value-of select="concat('"', $topclass, '::get', $dtraceattrname, '", ')"/> + <xsl:apply-templates select="@type" mode="logparamval"> + <xsl:with-param name="dir" select="'out'"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + <xsl:text>, hrc)); + return hrc;</xsl:text> + <xsl:if test="$attrbasename = 'MidlDoesNotLikeEmptyInterfaces'"> + <xsl:text> +#else /* dummy attribute */ + NOREF(aMidlDoesNotLikeEmptyInterfaces); + return E_FAIL; +#endif /* dummy attribute */</xsl:text> + </xsl:if> + <xsl:text> +} +</xsl:text> + <xsl:if test="not(@readonly) or @readonly!='yes'"> + <xsl:text> +</xsl:text> + <xsl:value-of select="concat('STDMETHODIMP ', $topclass, 'Wrap::COMSETTER(', $attrbasename, ')(')"/> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir">in</xsl:with-param> + </xsl:call-template> + <!-- @todo check in parameters if possible --> + <xsl:text>) +{ + LogRelFlow(("{%p} %s: enter </xsl:text> + <xsl:apply-templates select="@type" mode="logparamtext"> + <xsl:with-param name="dir" select="'in'"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + <xsl:text>\n", this, </xsl:text> + <xsl:value-of select="concat('"', $topclass, '::set', $attrbasename, '", ')"/> + <xsl:apply-templates select="@type" mode="logparamval"> + <xsl:with-param name="dir" select="'in'"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + <xsl:text>)); +</xsl:text> + <xsl:if test="ancestor::interface[@wrap-gen-hook = 'yes']"> + <xsl:text> + i_callHook(__FUNCTION__);</xsl:text> + </xsl:if> +<xsl:text> + // Clear error info, to make in-process calls behave the same as + // cross-apartment calls or out-of-process calls. + VirtualBoxBase::clearError(); + + HRESULT hrc; + + try + { + </xsl:text> + <xsl:apply-templates select="@type" mode="paramvalconversion2tmpvar"> + <xsl:with-param name="dir" select="'in'"/> + </xsl:apply-templates> + <xsl:text> + +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $topclass, '_SET_', $dtraceattrname, '_ENTER('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, </xsl:text> + <xsl:call-template name="emitDTraceParamValue"> + <xsl:with-param name="dir">in</xsl:with-param> + </xsl:call-template> + <xsl:text>); +#endif + </xsl:text> + <xsl:choose> + <xsl:when test="$limitedAutoCaller = 'true'"> + <xsl:text>AutoLimitedCaller</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>AutoCaller</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:text> autoCaller(this); + hrc = autoCaller.rc(); + if (SUCCEEDED(hrc)) + { +</xsl:text> + <xsl:value-of select="concat(' hrc = set', $attrbasename, '(')"/> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>autoCaller, </xsl:text> + </xsl:if> + <xsl:apply-templates select="@type" mode="paramvalconversionusingtmp"> + <xsl:with-param name="dir" select="'in'"/> + </xsl:apply-templates> + <xsl:text>); + } +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_SET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 0 /*normal*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValue"> + <xsl:with-param name="dir">in</xsl:with-param> + </xsl:call-template> + <xsl:text>); +#endif + } + catch (HRESULT hrc2) + { + hrc = hrc2; +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_SET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 1 /*hrc exception*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp-DirIn"/> + <xsl:text>); +#endif + } + catch (...) + { + hrc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS); +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_SET_', $dtraceattrname, '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 9 /*unhandled exception*/,</xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp-DirIn"/> + <xsl:text>); +#endif + } + + LogRelFlow(("{%p} %s: leave hrc=%Rhrc\n", this, </xsl:text> + <xsl:value-of select="concat('"', $topclass, '::set', $attrbasename, '", ')"/> + <xsl:text>hrc)); + return hrc; +} +</xsl:text> + </xsl:if> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:call-template name="xsltprocNewlineOutputHack"/> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + Emit DTrace probes for the given attribute. + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template match="attribute" mode="dtrace-probes"> + <xsl:param name="topclass"/> + <xsl:param name="dtracetopclass"/> + <xsl:param name="target"/> + + <xsl:variable name="dtraceattrname"> + <xsl:choose> + <xsl:when test="@dtracename"> + <xsl:value-of select="@dtracename"/> + </xsl:when> + <xsl:otherwise> + <!-- attrbasename --> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:otherwise> + </xsl:choose> + </xsl:variable> + + <xsl:if test="@name != 'midlDoesNotLikeEmptyInterfaces'"> + <xsl:text> probe </xsl:text> + <!-- <xsl:value-of select="concat($dtracetopclass, '__get__', $dtraceattrname, '__enter(struct ', $topclass)"/> --> + <xsl:value-of select="concat($dtracetopclass, '__get__', $dtraceattrname, '__enter(void')"/> + <xsl:text> *a_pThis); + probe </xsl:text> + <!-- <xsl:value-of select="concat($dtracetopclass, '__get__', $dtraceattrname, '__return(struct ', $topclass, ' *a_pThis')"/> --> + <xsl:value-of select="concat($dtracetopclass, '__get__', $dtraceattrname, '__return(void *a_pThis')"/> + <xsl:text>, uint32_t a_hrc, int32_t enmWhy, </xsl:text> + <xsl:apply-templates select="@type" mode="dtraceparamdecl"> + <xsl:with-param name="dir">out</xsl:with-param> + </xsl:apply-templates> + <xsl:text>); +</xsl:text> + </xsl:if> + <xsl:if test="(not(@readonly) or @readonly!='yes') and @name != 'midlDoesNotLikeEmptyInterfaces'"> + <xsl:text> probe </xsl:text> + <!-- <xsl:value-of select="concat($topclass, '__set__', $dtraceattrname, '__enter(struct ', $topclass, ' *a_pThis, ')"/>--> + <xsl:value-of select="concat($topclass, '__set__', $dtraceattrname, '__enter(void *a_pThis, ')"/> + <xsl:apply-templates select="@type" mode="dtraceparamdecl"> + <xsl:with-param name="dir" select="'in'"/> + </xsl:apply-templates> + <xsl:text>); + probe </xsl:text> + <!-- <xsl:value-of select="concat($dtracetopclass, '__set__', $dtraceattrname, '__return(struct ', $topclass, ' *a_pThis')"/> --> + <xsl:value-of select="concat($dtracetopclass, '__set__', $dtraceattrname, '__return(void *a_pThis')"/> + <xsl:text>, uint32_t a_hrc, int32_t enmWhy, </xsl:text> + <xsl:apply-templates select="@type" mode="dtraceparamdecl"> + <xsl:with-param name="dir">in</xsl:with-param> + </xsl:apply-templates> + <xsl:text>); +</xsl:text> + </xsl:if> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + Emit all attributes of an interface (current node). + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitAttributes"> + <xsl:param name="topclass"/> + <xsl:param name="dtracetopclass"/> + <xsl:param name="pmode"/> + + <xsl:variable name="name" select="@name"/> + <!-- first recurse to emit all base interfaces --> + <xsl:variable name="extends" select="@extends"/> + <xsl:if test="$extends and not($extends='$unknown') and not($extends='$errorinfo')"> + <xsl:for-each select="key('G_keyInterfacesByName', $extends)"> + <xsl:call-template name="emitAttributes"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:call-template> + </xsl:for-each> + </xsl:if> + + <xsl:choose> + <xsl:when test="$pmode='code'"> + <xsl:text>// +</xsl:text> + <xsl:value-of select="concat('// ', $name, ' properties')"/> + <xsl:text> +// + +</xsl:text> + </xsl:when> + <xsl:when test="$pmode != 'dtrace-probes'"> + <xsl:value-of select="concat($G_sNewLine, ' /** @name ', translate(substring($pmode, 1, 1), $G_lowerCase, $G_upperCase), substring($pmode,2), ' ', $name, ' properties', $G_sNewLine)"/> + <xsl:text> * @{ */ +</xsl:text> + </xsl:when> + </xsl:choose> + <xsl:choose> + <xsl:when test="$pmode='public'"> + <xsl:apply-templates select="./attribute | ./if" mode="public"> + <xsl:with-param name="emitmode" select="'attribute'"/> + </xsl:apply-templates> + <xsl:variable name="reservedAttributes" select="@reservedAttributes"/> + <xsl:if test="$reservedAttributes > 0"> + <!-- tricky way to do a "for" loop without recursion --> + <xsl:for-each select="(//*)[position() <= $reservedAttributes]"> + <xsl:text> STDMETHOD(COMGETTER(InternalAndReservedAttribute</xsl:text> + <xsl:value-of select="concat(position(), $name)"/> + <xsl:text>))(ULONG *aReserved);
</xsl:text> + </xsl:for-each> + </xsl:if> + </xsl:when> + <xsl:when test="$pmode='wrapped'"> + <xsl:apply-templates select="./attribute | ./if" mode="wrapped"> + <xsl:with-param name="emitmode" select="'attribute'"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$pmode='code'"> + <xsl:apply-templates select="./attribute | ./if" mode="code"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="emitmode" select="'attribute'"/> + </xsl:apply-templates> + <xsl:variable name="reservedAttributes" select="@reservedAttributes"/> + <xsl:if test="$reservedAttributes > 0"> + <!-- tricky way to do a "for" loop without recursion --> + <xsl:for-each select="(//*)[position() <= $reservedAttributes]"> + <xsl:value-of select="concat('STDMETHODIMP ', $topclass, 'Wrap::COMGETTER(InternalAndReservedAttribute', position(), $name, ')(ULONG *aReserved)
')"/> + <xsl:text>{ + NOREF(aReserved); + return E_NOTIMPL; +} + +</xsl:text> + </xsl:for-each> + </xsl:if> + </xsl:when> + <xsl:when test="$pmode = 'dtrace-probes'"> + <xsl:apply-templates select="./attribute | ./if" mode="dtrace-probes"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="emitmode" select="'attribute'"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise><xsl:message terminate="yes">Otherwise oops in emitAttributes</xsl:message></xsl:otherwise> + </xsl:choose> + + <!-- close doxygen @name --> + <xsl:if test="($pmode != 'code') and ($pmode != 'dtrace-probes')" > + <xsl:text> /** @} */ +</xsl:text> + </xsl:if> +</xsl:template> + +<xsl:template name="emitTargetBegin"> + <xsl:param name="target"/> + + <xsl:choose> + <xsl:when test="$target = ''"/> + <xsl:when test="$target = 'xpidl'"> + <xsl:text>#ifdef VBOX_WITH_XPCOM +</xsl:text> + </xsl:when> + <xsl:when test="$target = 'midl'"> + <xsl:text>#ifndef VBOX_WITH_XPCOM +</xsl:text> + </xsl:when> + <xsl:otherwise><xsl:message terminate="yes">Otherwise oops in emitTargetBegin: target=<xsl:value-of select="$target"/></xsl:message></xsl:otherwise> + </xsl:choose> +</xsl:template> + +<xsl:template name="emitTargetEnd"> + <xsl:param name="target"/> + + <xsl:choose> + <xsl:when test="$target = ''"/> + <xsl:when test="$target = 'xpidl'"> + <xsl:text>#endif /* VBOX_WITH_XPCOM */ +</xsl:text> + </xsl:when> + <xsl:when test="$target = 'midl'"> + <xsl:text>#endif /* !VBOX_WITH_XPCOM */ +</xsl:text> + </xsl:when> + <xsl:otherwise><xsl:message terminate="yes">Otherwise oops in emitTargetEnd target=<xsl:value-of select="$target"/></xsl:message></xsl:otherwise> + </xsl:choose> +</xsl:template> + + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit method + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="method" mode="public"> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="methodindent"> + <xsl:call-template name="tospace"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + + <xsl:text> STDMETHOD(</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:text>)(</xsl:text> + <xsl:for-each select="param"> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:call-template> + <xsl:if test="not(position()=last())"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodindent"/> + </xsl:if> + </xsl:for-each> + <xsl:text>); +</xsl:text> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="method" mode="wrapped"> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="methodindent"> + <xsl:call-template name="tospace"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + + <xsl:text> virtual HRESULT </xsl:text> + <xsl:call-template name="uncapitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:text>(</xsl:text> + <xsl:variable name="passAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'passcaller'"/> + </xsl:call-template> + </xsl:variable> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>AutoCaller &aAutoCaller</xsl:text> + <xsl:if test="count(param) > 0"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodindent"/> + </xsl:if> + </xsl:if> + <xsl:for-each select="param"> + <xsl:apply-templates select="@type" mode="wrapped"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:apply-templates> + <xsl:if test="not(position()=last())"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodindent"/> + </xsl:if> + </xsl:for-each> + <xsl:text>) = 0; +</xsl:text> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="method" mode="code"> + <xsl:param name="topclass"/> + <xsl:param name="dtracetopclass"/> + <xsl:param name="target"/> + + <xsl:call-template name="emitTargetBegin"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:variable name="methodindent"> + <xsl:call-template name="tospace"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="methodclassindent"> + <xsl:call-template name="tospace"> + <xsl:with-param name="str" select="concat($topclass, @name)"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="methodbasename"> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="limitedAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'limitedcaller'"/> + </xsl:call-template> + </xsl:variable> + <xsl:variable name="dtracemethodname"> + <xsl:choose> + <xsl:when test="@dtracename"> + <xsl:value-of select="@dtracename"/> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="@name"/> + </xsl:otherwise> + </xsl:choose> + </xsl:variable> + <xsl:variable name="dtracenamehack"> <!-- Ugly hack to deal with Session::assignMachine and similar. --> + <xsl:if test="name(..) = 'if'"> + <xsl:value-of select="concat('__', ../@target)"/> + </xsl:if> + </xsl:variable> + + <xsl:value-of select="concat('STDMETHODIMP ', $topclass, 'Wrap::', $methodbasename, '(')"/> + <xsl:for-each select="param"> + <xsl:call-template name="emitPublicParameter"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:call-template> + <xsl:if test="not(position()=last())"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodclassindent"/> + </xsl:if> + </xsl:for-each> + <xsl:text>) +{ + LogRelFlow(("{%p} %s: enter</xsl:text> + <xsl:for-each select="param"> + <xsl:text> </xsl:text> + <xsl:apply-templates select="@type" mode="logparamtext"> + <xsl:with-param name="dir" select="@dir"/> + <xsl:with-param name="isref" select="'yes'"/> + </xsl:apply-templates> + </xsl:for-each> + <xsl:text>\n", this</xsl:text> + <xsl:value-of select="concat(', "', $topclass, '::', @name, '"')"/> + <xsl:for-each select="param"> + <xsl:text>, </xsl:text> + <xsl:apply-templates select="@type" mode="logparamval"> + <xsl:with-param name="dir" select="@dir"/> + <xsl:with-param name="isref" select="'yes'"/> + </xsl:apply-templates> + </xsl:for-each> + <xsl:text>)); +</xsl:text> + <xsl:if test="ancestor::interface[@wrap-gen-hook = 'yes']"> + <xsl:text> + i_callHook(__FUNCTION__);</xsl:text> + </xsl:if> +<xsl:text> + // Clear error info, to make in-process calls behave the same as + // cross-apartment calls or out-of-process calls. + VirtualBoxBase::clearError(); + + HRESULT hrc; + + try + { +</xsl:text> + <!-- @todo check in parameters if possible --> + <xsl:for-each select="param"> + <xsl:if test="@dir!='in'"> + <xsl:text> CheckComArgOutPointerValidThrow(a</xsl:text> + <xsl:call-template name="capitalize"> + <xsl:with-param name="str" select="@name"/> + </xsl:call-template> + <xsl:text>); +</xsl:text> + </xsl:if> + </xsl:for-each> +<xsl:text> +</xsl:text> + <xsl:for-each select="param"> + <xsl:text> + </xsl:text> + <xsl:apply-templates select="@type" mode="paramvalconversion2tmpvar"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:apply-templates> + </xsl:for-each> + <xsl:text> + +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_', $dtracemethodname, substring($dtracenamehack, 2), '_ENTER('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this</xsl:text> + <xsl:for-each select="param[@dir='in']"> + <xsl:text>, </xsl:text> + <xsl:call-template name="emitDTraceParamValue"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:call-template> + </xsl:for-each> + <xsl:text>); +#endif + </xsl:text> + <xsl:choose> + <xsl:when test="$limitedAutoCaller = 'true'"> + <xsl:text>AutoLimitedCaller</xsl:text> + </xsl:when> + <xsl:otherwise> + <xsl:text>AutoCaller</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:text> autoCaller(this); + hrc = autoCaller.rc(); + if (SUCCEEDED(hrc)) + { +</xsl:text> + <xsl:value-of select="concat(' hrc = ', @name, '(')"/> + <xsl:variable name="passAutoCaller"> + <xsl:call-template name="checkoption"> + <xsl:with-param name="optionlist" select="@wrap-hint-server"/> + <xsl:with-param name="option" select="'passcaller'"/> + </xsl:call-template> + </xsl:variable> + <xsl:if test="$passAutoCaller = 'true'"> + <xsl:text>autoCaller</xsl:text> + <xsl:if test="count(param) > 0"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodindent"/> + </xsl:if> + </xsl:if> + <xsl:for-each select="param"> + <xsl:apply-templates select="@type" mode="paramvalconversionusingtmp"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:apply-templates> + <xsl:if test="not(position()=last())"> + <xsl:text>, + </xsl:text> + <xsl:value-of select="$methodindent"/> + </xsl:if> + </xsl:for-each> + <xsl:text>); + } +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_', $dtracemethodname, substring($dtracenamehack, 2), '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 0 /*normal*/</xsl:text> + <xsl:for-each select="param"> + <xsl:text>, </xsl:text> + <xsl:call-template name="emitDTraceParamValue"> + <xsl:with-param name="dir" select="@dir"/> + </xsl:call-template> + </xsl:for-each> + <xsl:text>); +#endif + } + catch (HRESULT hrc2) + { + hrc = hrc2; +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_', $dtracemethodname, substring($dtracenamehack, 2), '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 1 /*hrc exception*/</xsl:text> + <xsl:for-each select="param"> + <xsl:text>, </xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp"/> + </xsl:for-each> + <xsl:text>); +#endif + } + catch (...) + { + hrc = VirtualBoxBase::handleUnexpectedExceptions(this, RT_SRC_POS); +#ifdef VBOX_WITH_DTRACE_R3_MAIN + </xsl:text> + <xsl:value-of select="translate(concat('VBOXAPI_', $dtracetopclass, '_', $dtracemethodname, substring($dtracenamehack, 2), '_RETURN('), $G_lowerCase, $G_upperCase)"/> + <xsl:text>this, hrc, 9 /*unhandled exception*/</xsl:text> + <xsl:for-each select="param"> + <xsl:text>, </xsl:text> + <xsl:call-template name="emitDTraceParamValNoTmp"/> + </xsl:for-each> + <xsl:text>); +#endif + } + + LogRelFlow(("{%p} %s: leave</xsl:text> + <xsl:for-each select="param"> + <xsl:if test="@dir!='in'"> + <xsl:text> </xsl:text> + <xsl:apply-templates select="@type" mode="logparamtext"> + <xsl:with-param name="dir" select="@dir"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + </xsl:if> + </xsl:for-each> + <xsl:text> hrc=%Rhrc\n", this</xsl:text> + <xsl:value-of select="concat(', "', $topclass, '::', @name, '"')"/> + <xsl:for-each select="param"> + <xsl:if test="@dir!='in'"> + <xsl:text>, </xsl:text> + <xsl:apply-templates select="@type" mode="logparamval"> + <xsl:with-param name="dir" select="@dir"/> + <xsl:with-param name="isref" select="''"/> + </xsl:apply-templates> + </xsl:if> + </xsl:for-each> + <xsl:text>, hrc)); + return hrc; +} +</xsl:text> + + <xsl:call-template name="emitTargetEnd"> + <xsl:with-param name="target" select="$target"/> + </xsl:call-template> + + <xsl:text> +</xsl:text> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + Emits the DTrace probes for a method. + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template match="method" mode="dtrace-probes"> + <xsl:param name="topclass"/> + <xsl:param name="dtracetopclass"/> + <xsl:param name="target"/> + + <xsl:variable name="dtracemethodname"> + <xsl:choose> + <xsl:when test="@dtracename"> + <xsl:value-of select="@dtracename"/> + </xsl:when> + <xsl:otherwise> + <xsl:value-of select="@name"/> + </xsl:otherwise> + </xsl:choose> + </xsl:variable> + <xsl:variable name="dtracenamehack"> <!-- Ugly hack to deal with Session::assignMachine and similar. --> + <xsl:if test="name(..) = 'if'"> + <xsl:value-of select="concat('__', ../@target)"/> + </xsl:if> + </xsl:variable> + + <xsl:text> probe </xsl:text> + <!-- <xsl:value-of select="concat($dtracetopclass, '__', $dtracemethodname, $dtracenamehack, '__enter(struct ', $dtracetopclass, ' *a_pThis')"/> --> + <xsl:value-of select="concat($dtracetopclass, '__', $dtracemethodname, $dtracenamehack, '__enter(void *a_pThis')"/> + <xsl:for-each select="param[@dir='in']"> + <xsl:text>, </xsl:text> + <xsl:apply-templates select="@type" mode="dtraceparamdecl"> + <xsl:with-param name="dir" select="'@dir'"/> + </xsl:apply-templates> + </xsl:for-each> + <xsl:text>); + probe </xsl:text> + <!-- <xsl:value-of select="concat($dtracetopclass, '__', $dtracemethodname, '__return(struct ', $dtracetopclass, ' *a_pThis')"/> --> + <xsl:value-of select="concat($dtracetopclass, '__', $dtracemethodname, $dtracenamehack, '__return(void *a_pThis')"/> + <xsl:text>, uint32_t a_hrc, int32_t enmWhy</xsl:text> + <xsl:for-each select="param"> + <xsl:text>, </xsl:text> + <xsl:apply-templates select="@type" mode="dtraceparamdecl"> + <xsl:with-param name="dir" select="'@dir'"/> + </xsl:apply-templates> + </xsl:for-each> + <xsl:text>); +</xsl:text> + +</xsl:template> + + +<xsl:template name="emitIf"> + <xsl:param name="passmode"/> + <xsl:param name="target"/> + <xsl:param name="topclass"/> + <xsl:param name="emitmode"/> + <xsl:param name="dtracetopclass"/> + + <xsl:if test="($target = 'xpidl') or ($target = 'midl')"> + <xsl:choose> + <xsl:when test="$passmode='public'"> + <xsl:choose> + <xsl:when test="$emitmode='method'"> + <xsl:apply-templates select="method" mode="public"> + <xsl:with-param name="target" select="$target"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$emitmode='attribute'"> + <xsl:apply-templates select="attribute" mode="public"> + <xsl:with-param name="target" select="$target"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + </xsl:when> + <xsl:when test="$passmode='wrapped'"> + <xsl:choose> + <xsl:when test="$emitmode='method'"> + <xsl:apply-templates select="method" mode="wrapped"> + <xsl:with-param name="target" select="$target"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$emitmode='attribute'"> + <xsl:apply-templates select="attribute" mode="wrapped"> + <xsl:with-param name="target" select="$target"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + </xsl:when> + <xsl:when test="$passmode='code'"> + <xsl:choose> + <xsl:when test="$emitmode='method'"> + <xsl:apply-templates select="method" mode="code"> + <xsl:with-param name="target" select="$target"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$emitmode='attribute'"> + <xsl:apply-templates select="attribute" mode="code"> + <xsl:with-param name="target" select="$target"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + </xsl:when> + <xsl:when test="$passmode = 'dtrace-probes'"> + <xsl:choose> + <xsl:when test="$emitmode = 'method'"> + <xsl:apply-templates select="method" mode="dtrace-probes"> + <xsl:with-param name="target" select="$target"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$emitmode = 'attribute'"> + <xsl:apply-templates select="attribute" mode="dtrace-probes"> + <xsl:with-param name="target" select="$target"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + </xsl:if> +</xsl:template> + +<xsl:template match="if" mode="public"> + <xsl:param name="emitmode"/> + + <xsl:call-template name="emitIf"> + <xsl:with-param name="passmode" select="'public'"/> + <xsl:with-param name="target" select="@target"/> + <xsl:with-param name="emitmode" select="$emitmode"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="if" mode="wrapped"> + <xsl:param name="emitmode"/> + + <xsl:call-template name="emitIf"> + <xsl:with-param name="passmode" select="'wrapped'"/> + <xsl:with-param name="target" select="@target"/> + <xsl:with-param name="emitmode" select="$emitmode"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="if" mode="code"> + <xsl:param name="topclass"/> + <xsl:param name="emitmode"/> + <xsl:param name="dtracetopclass"/> + + <xsl:call-template name="emitIf"> + <xsl:with-param name="passmode" select="'code'"/> + <xsl:with-param name="target" select="@target"/> + <xsl:with-param name="emitmode" select="$emitmode"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:call-template> +</xsl:template> + +<xsl:template match="if" mode="dtrace-probes"> + <xsl:param name="topclass"/> + <xsl:param name="emitmode"/> + <xsl:param name="dtracetopclass"/> + + <xsl:call-template name="emitIf"> + <xsl:with-param name="passmode" select="'dtrace-probes'"/> + <xsl:with-param name="target" select="@target"/> + <xsl:with-param name="emitmode" select="$emitmode"/> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:call-template> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit all methods of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitMethods"> + <xsl:param name="topclass"/> + <xsl:param name="pmode"/> + <xsl:param name="dtracetopclass"/> + + <xsl:variable name="name" select="@name"/> + <!-- first recurse to emit all base interfaces --> + <xsl:variable name="extends" select="@extends"/> + <xsl:if test="$extends and not($extends='$unknown') and not($extends='$errorinfo')"> + <xsl:for-each select="key('G_keyInterfacesByName', $extends)"> + <xsl:call-template name="emitMethods"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + </xsl:call-template> + </xsl:for-each> + </xsl:if> + + <xsl:choose> + <xsl:when test="$pmode='code'"> + <xsl:text>// +</xsl:text> + <xsl:value-of select="concat('// ', $name, ' methods')"/> + <xsl:text> +// + +</xsl:text> + </xsl:when> + <xsl:when test="$pmode='dtrace-probes'"/> + <xsl:otherwise> + <xsl:value-of select="concat($G_sNewLine, ' /** @name ', translate(substring($pmode, 1, 1), $G_lowerCase, $G_upperCase), substring($pmode,2), ' ', $name, ' methods', $G_sNewLine)"/> + <xsl:text> * @{ */ +</xsl:text> + </xsl:otherwise> + </xsl:choose> + <xsl:choose> + <xsl:when test="$pmode='public'"> + <xsl:apply-templates select="./method | ./if" mode="public"> + <xsl:with-param name="emitmode" select="'method'"/> + </xsl:apply-templates> + <xsl:variable name="reservedMethods" select="@reservedMethods"/> + <xsl:if test="$reservedMethods > 0"> + <!-- tricky way to do a "for" loop without recursion --> + <xsl:for-each select="(//*)[position() <= $reservedMethods]"> + <xsl:text> STDMETHOD(InternalAndReservedMethod</xsl:text> + <xsl:value-of select="concat(position(), $name)"/> + <xsl:text>)();
</xsl:text> + </xsl:for-each> + </xsl:if> + </xsl:when> + <xsl:when test="$pmode='wrapped'"> + <xsl:apply-templates select="./method | ./if" mode="wrapped"> + <xsl:with-param name="emitmode" select="'method'"/> + </xsl:apply-templates> + </xsl:when> + <xsl:when test="$pmode='code'"> + <xsl:apply-templates select="./method | ./if" mode="code"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="emitmode" select="'method'"/> + </xsl:apply-templates> + <xsl:variable name="reservedMethods" select="@reservedMethods"/> + <xsl:if test="$reservedMethods > 0"> + <!-- tricky way to do a "for" loop without recursion --> + <xsl:for-each select="(//*)[position() <= $reservedMethods]"> + <xsl:value-of select="concat('STDMETHODIMP ', $topclass, 'Wrap::InternalAndReservedMethod', position(), $name, '()
')"/> + <xsl:text>{ + return E_NOTIMPL; +} + +</xsl:text> + </xsl:for-each> + </xsl:if> + </xsl:when> + <xsl:when test="$pmode='dtrace-probes'"> + <xsl:apply-templates select="./method | ./if" mode="dtrace-probes"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="emitmode" select="'method'"/> + </xsl:apply-templates> + </xsl:when> + <xsl:otherwise/> + </xsl:choose> + + <!-- close doxygen @name --> + <xsl:if test="($pmode != 'code') and ($pmode != 'dtrace-probes')" > + <xsl:text> /** @} */ +</xsl:text> + </xsl:if> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit all attributes and methods declarations of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitInterfaceDecls"> + <xsl:param name="pmode"/> + + <!-- attributes --> + <xsl:call-template name="emitAttributes"> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> + + <!-- methods --> + <xsl:call-template name="emitMethods"> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit auxiliary method declarations of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitAuxMethodDecls"> + <!-- currently nothing, maybe later some generic FinalConstruct/... helper declaration for ComObjPtr --> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit the header file of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitHeader"> + <xsl:param name="addinterfaces"/> + + <xsl:variable name="filename" select="concat(substring(@name, 2), 'Wrap.h')"/> + + <xsl:apply-templates select="." mode="startfile"> + <xsl:with-param name="file" select="$filename"/> + </xsl:apply-templates> + <xsl:call-template name="fileheader"> + <xsl:with-param name="name" select="$filename"/> + <xsl:with-param name="class" select="substring(@name, 2)"/> + <xsl:with-param name="type" select="'header'"/> + </xsl:call-template> + <xsl:apply-templates select="." mode="classheader"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:apply-templates> + + <!-- interface attributes/methods (public) --> + <xsl:call-template name="emitInterfaceDecls"> + <xsl:with-param name="pmode" select="'public'"/> + </xsl:call-template> + + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <!-- This is super tricky, as the for-each switches to the node set, + which means the normal document isn't available any more. We get + the data we need, uses a for-each to switch document and then a + key() to look up the interface by name. --> + <xsl:variable name="addifname"> + <xsl:value-of select="string(.)"/> + </xsl:variable> + <xsl:for-each select="$G_root"> + <xsl:for-each select="key('G_keyInterfacesByName', $addifname)"> + <xsl:call-template name="emitInterfaceDecls"> + <xsl:with-param name="pmode" select="'public'"/> + </xsl:call-template> + </xsl:for-each> + </xsl:for-each> + </xsl:for-each> + + <!-- auxiliary methods (public) --> + <xsl:call-template name="emitAuxMethodDecls"/> + + <!-- switch to private --> + <xsl:text> +private:</xsl:text> + + <!-- wrapped interface attributes/methods (private) --> + <xsl:call-template name="emitInterfaceDecls"> + <xsl:with-param name="pmode" select="'wrapped'"/> + </xsl:call-template> + + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <!-- This is super tricky, as the for-each switches to the node set, + which means the normal document isn't available any more. We get + the data we need, uses a for-each to switch document and then a + key() to look up the interface by name. --> + <xsl:variable name="addifname"> + <xsl:value-of select="string(.)"/> + </xsl:variable> + <xsl:for-each select="$G_root"> + <xsl:for-each select="key('G_keyInterfacesByName', $addifname)"> + <xsl:call-template name="emitInterfaceDecls"> + <xsl:with-param name="pmode" select="'wrapped'"/> + </xsl:call-template> + </xsl:for-each> + </xsl:for-each> + </xsl:for-each> + + <xsl:apply-templates select="." mode="classfooter"/> + <xsl:apply-templates select="." mode="endfile"> + <xsl:with-param name="file" select="$filename"/> + </xsl:apply-templates> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit all attributes and methods definitions (pmode=code) or probes (pmode=dtrace-probes) of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitInterfaceDefs"> + <xsl:param name="addinterfaces"/> + <xsl:param name="pmode" select="'code'"/> + + <xsl:variable name="topclass" select="substring(@name, 2)"/> + <xsl:variable name="dtracetopclass"> + <xsl:choose> + <xsl:when test="@dtracename"><xsl:value-of select="@dtracename"/></xsl:when> + <xsl:otherwise><xsl:value-of select="$topclass"/></xsl:otherwise> + </xsl:choose> + </xsl:variable> + + <xsl:if test="$pmode = 'code'"> + <xsl:value-of select="concat('DEFINE_EMPTY_CTOR_DTOR(', $topclass, 'Wrap)', $G_sNewLine, $G_sNewLine)"/> + </xsl:if> + + <!-- attributes --> + <xsl:call-template name="emitAttributes"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> + + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <!-- This is super tricky, as the for-each switches to the node set, + which means the normal document isn't available any more. We get + the data we need, uses a for-each to switch document and then a + key() to look up the interface by name. --> + <xsl:variable name="addifname"> + <xsl:value-of select="string(.)"/> + </xsl:variable> + <xsl:for-each select="$G_root"> + <xsl:for-each select="key('G_keyInterfacesByName', $addifname)"> + <xsl:call-template name="emitAttributes"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> + </xsl:for-each> + </xsl:for-each> + </xsl:for-each> + + <!-- methods --> + <xsl:call-template name="xsltprocNewlineOutputHack"/> + <xsl:call-template name="emitMethods"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> + + <xsl:for-each select="exsl:node-set($addinterfaces)/token"> + <!-- This is super tricky, as the for-each switches to the node set, + which means the normal document isn't available any more. We get + the data we need, uses a for-each to switch document and then a + key() to look up the interface by name. --> + <xsl:variable name="addifname"> + <xsl:value-of select="string(.)"/> + </xsl:variable> + <xsl:for-each select="$G_root"> + <xsl:for-each select="key('G_keyInterfacesByName', $addifname)"> + <xsl:call-template name="emitMethods"> + <xsl:with-param name="topclass" select="$topclass"/> + <xsl:with-param name="dtracetopclass" select="$dtracetopclass"/> + <xsl:with-param name="pmode" select="$pmode"/> + </xsl:call-template> + </xsl:for-each> + </xsl:for-each> + </xsl:for-each> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit auxiliary method declarations of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitAuxMethodDefs"> + <xsl:param name="pmode" select="'code'"/> + <!-- currently nothing, maybe later some generic FinalConstruct/... implementation --> +</xsl:template> + + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit the code file of the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitCode"> + <xsl:param name="addinterfaces"/> + + <xsl:variable name="filename" select="concat(substring(@name, 2), 'Wrap.cpp')"/> + + <xsl:apply-templates select="." mode="startfile"> + <xsl:with-param name="file" select="$filename"/> + </xsl:apply-templates> + <xsl:call-template name="fileheader"> + <xsl:with-param name="name" select="$filename"/> + <xsl:with-param name="class" select="substring(@name, 2)"/> + <xsl:with-param name="type" select="'code'"/> + </xsl:call-template> + <xsl:apply-templates select="." mode="codeheader"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:apply-templates> + + <!-- interface attributes/methods (public) --> + <xsl:call-template name="emitInterfaceDefs"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:call-template> + + <!-- auxiliary methods (public) --> + <xsl:call-template name="emitAuxMethodDefs"/> + + <xsl:apply-templates select="." mode="codefooter"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + </xsl:apply-templates> + <xsl:apply-templates select="." mode="endfile"> + <xsl:with-param name="file" select="$filename"/> + </xsl:apply-templates> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + emit the DTrace probes for the current interface + - - - - - - - - - - - - - - - - - - - - - - --> +<xsl:template name="emitDTraceProbes"> + <xsl:param name="addinterfaces"/> + + <!-- interface attributes/methods (public) --> + <xsl:call-template name="emitInterfaceDefs"> + <xsl:with-param name="addinterfaces" select="$addinterfaces"/> + <xsl:with-param name="pmode">dtrace-probes</xsl:with-param> + </xsl:call-template> + + <!-- auxiliary methods (public) --> + <xsl:call-template name="emitAuxMethodDefs"> + <xsl:with-param name="pmode">dtrace-probes</xsl:with-param> + </xsl:call-template> + +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + wildcard match, ignore everything which has no explicit match + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="*"/> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + ignore all if tags except those for XPIDL or MIDL target + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="if"> + <xsl:if test="(@target = 'xpidl') or (@target = 'midl')"> + <xsl:apply-templates/> + </xsl:if> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + interface match + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="interface"> + <xsl:if test="not(@internal='yes') and not(@autogen='VBoxEvent') and not(@supportsErrorInfo='no')"> + <xsl:call-template name="emitInterface"/> + </xsl:if> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + application match + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="application"> + <xsl:apply-templates/> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + library match + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="library"> + <xsl:apply-templates/> +</xsl:template> + +<!-- - - - - - - - - - - - - - - - - - - - - - - + root match + - - - - - - - - - - - - - - - - - - - - - - --> + +<xsl:template match="/idl"> + <xsl:choose> + <xsl:when test="$generating = 'headers'"> + <xsl:apply-templates/> + </xsl:when> + <xsl:when test="$generating = 'sources'"> + <xsl:apply-templates/> + </xsl:when> + <xsl:when test="$generating = 'dtrace-probes'"> + <xsl:apply-templates/> + </xsl:when> + <xsl:otherwise> + <xsl:message terminate="yes"> + Unknown string parameter value: generating='<xsl:value-of select="$generating"/>' + </xsl:message> + </xsl:otherwise> + </xsl:choose> +</xsl:template> + +</xsl:stylesheet> +<!-- vi: set tabstop=4 shiftwidth=4 expandtab: --> |