Files
libreoffice/stoc/source/javaloader/javaloader.cxx

444 lines
18 KiB
C++
Raw Normal View History

2000-10-26 11:28:22 +00:00
/*************************************************************************
2000-09-18 14:29:57 +00:00
*
* $RCSfile: javaloader.cxx,v $
*
2002-10-29 08:25:58 +00:00
* $Revision: 1.15 $
2000-09-18 14:29:57 +00:00
*
2002-10-29 08:25:58 +00:00
* last change: $Author: dbo $ $Date: 2002-10-29 09:25:58 $
2000-09-18 14:29:57 +00:00
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (the "License"); You may not use this file
* except in compliance with the License. You may obtain a copy of the
* License at http://www.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#include <cstdarg>
2000-09-18 14:29:57 +00:00
#include <osl/diagnose.h>
#include <osl/process.h>
2000-09-18 14:29:57 +00:00
#include <rtl/ustring>
#include <rtl/process.h>
#include <rtl/ustrbuf.hxx>
2000-09-18 14:29:57 +00:00
#include <uno/environment.h>
#include <uno/mapping.hxx>
#include <cppuhelper/servicefactory.hxx>
#ifdef LINUX
#undef minor
#undef major
#endif
#include <com/sun/star/java/XJavaVM.hpp>
#include <com/sun/star/java/XJavaThreadRegister_11.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#include "jni.h"
#include <cppuhelper/factory.hxx>
#include <cppuhelper/implementationentry.hxx>
2000-09-18 14:29:57 +00:00
#include <cppuhelper/implbase2.hxx>
#include <com/sun/star/loader/XImplementationLoader.hpp>
#include <com/sun/star/lang/IllegalArgumentException.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#include <com/sun/star/lang/XServiceInfo.hpp>
#include <com/sun/star/lang/XInitialization.hpp>
#include <com/sun/star/registry/XRegistryKey.hpp>
using namespace ::com::sun::star::java;
using namespace ::com::sun::star::lang;
using namespace ::com::sun::star::loader;
using namespace ::com::sun::star::uno;
using namespace ::com::sun::star::registry;
using namespace ::cppu;
using namespace ::rtl;
using namespace ::osl;
namespace stoc_javaloader {
static Sequence< OUString > loader_getSupportedServiceNames()
{
static Sequence < OUString > *pNames = 0;
if( ! pNames )
{
MutexGuard guard( Mutex::getGlobalMutex() );
if( !pNames )
{
static Sequence< OUString > seqNames(2);
seqNames.getArray()[0] = OUString(
RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.loader.Java") );
seqNames.getArray()[1] = OUString(
RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.loader.Java2") );
pNames = &seqNames;
}
}
return *pNames;
}
static OUString loader_getImplementationName()
{
static OUString *pImplName = 0;
if( ! pImplName )
{
MutexGuard guard( Mutex::getGlobalMutex() );
if( ! pImplName )
{
static OUString implName(
RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.stoc.JavaComponentLoader" ) );
pImplName = &implName;
}
}
return *pImplName;
}
2000-09-18 14:29:57 +00:00
class JavaComponentLoader : public WeakImplHelper2<XImplementationLoader, XServiceInfo> {
Reference<XMultiComponentFactory> _xSMgr;
Reference<XComponentContext> _xCtx;
2000-09-18 14:29:57 +00:00
Reference<XImplementationLoader> _javaLoader;
public:
JavaComponentLoader(const Reference<XComponentContext> & xCtx) throw(RuntimeException);
virtual ~JavaComponentLoader() throw();
2000-09-18 14:29:57 +00:00
public:
// XServiceInfo
virtual OUString SAL_CALL getImplementationName() throw(RuntimeException);
virtual sal_Bool SAL_CALL supportsService(const OUString& ServiceName) throw(RuntimeException);
virtual Sequence<OUString> SAL_CALL getSupportedServiceNames() throw(RuntimeException);
2000-09-18 14:29:57 +00:00
// XImplementationLoader
virtual Reference<XInterface> SAL_CALL activate(const OUString& implementationName, const OUString& implementationLoaderUrl, const OUString& locationUrl, const Reference<XRegistryKey>& xKey) throw(CannotActivateFactoryException, RuntimeException);
virtual sal_Bool SAL_CALL writeRegistryInfo(const Reference<XRegistryKey>& xKey, const OUString& implementationLoaderUrl, const OUString& locationUrl) throw(CannotRegisterImplementationException, RuntimeException);
};
JavaComponentLoader::JavaComponentLoader(const Reference<XComponentContext> & xCtx) throw(RuntimeException)
: _xSMgr(xCtx->getServiceManager())
, _xCtx( xCtx )
2000-09-18 14:29:57 +00:00
{
sal_Int32 size = 0;
uno_Environment ** ppJava_environments = NULL;
uno_Environment * pUno_environment = NULL;
typelib_InterfaceTypeDescription * pType_XImplementationLoader = 0;
JNIEnv * pJNIEnv = NULL;
2000-09-18 14:29:57 +00:00
Reference<XJavaThreadRegister_11> xJavaThreadRegister_11;
Reference<XInterface> javaVM;
JavaVM * pJavaVM = NULL;
2000-09-18 14:29:57 +00:00
try {
// get a java vm, where we can create a loader
javaVM = _xSMgr->createInstanceWithContext(
OUString(RTL_CONSTASCII_USTRINGPARAM("com.sun.star.java.JavaVirtualMachine")),
xCtx );
2002-09-17 12:43:40 +00:00
if(!javaVM.is())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - could not create com.sun.star.java.JavaVirtualMachine")), Reference<XInterface>());
Reference<XJavaVM> javaVM_xJavaVM(javaVM, UNO_QUERY);
2002-09-17 12:43:40 +00:00
if(!javaVM_xJavaVM.is())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - could not get XJavaVM interface from JavaVirtualMachine service")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
Sequence<sal_Int8> processID(16);
rtl_getGlobalProcessId(reinterpret_cast<sal_uInt8 *>(processID.getArray()));
2000-09-18 14:29:57 +00:00
2000-10-25 10:15:49 +00:00
if(javaVM_xJavaVM->getJavaVM(processID).hasValue())
pJavaVM = *(JavaVM **)(javaVM_xJavaVM->getJavaVM(processID).getValue());
2002-09-17 12:43:40 +00:00
if(!pJavaVM)
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - JavaVirtualMachine service could not provide a VM")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
xJavaThreadRegister_11 = Reference<XJavaThreadRegister_11>(javaVM, UNO_QUERY);
if(xJavaThreadRegister_11.is())
xJavaThreadRegister_11->registerThread();
2000-09-18 14:29:57 +00:00
pJavaVM->AttachCurrentThread((void **)&pJNIEnv, (void *)NULL);
2000-09-18 14:29:57 +00:00
// instantiate the java JavaLoader
2002-09-17 12:43:40 +00:00
jclass jcJavaLoader = pJNIEnv->FindClass("com/sun/star/comp/loader/JavaLoader");
if(pJNIEnv->ExceptionOccurred())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - could not find class com/sun/star/comp/loader/JavaLoader")), Reference<XInterface>());
jmethodID jmJavaLoader_init = pJNIEnv->GetMethodID(jcJavaLoader, "<init>", "()V");
if(pJNIEnv->ExceptionOccurred())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - instantiation of com.sun.star.comp.loader.JavaLoader failed")), Reference<XInterface>());
jobject joJavaLoader = pJNIEnv->NewObject(jcJavaLoader, jmJavaLoader_init);
if(pJNIEnv->ExceptionOccurred())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - instantiation of com.sun.star.comp.loader.JavaLoader failed")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
// map the java JavaLoader to this environment
OUString sJava(RTL_CONSTASCII_USTRINGPARAM("java"));
2000-09-18 14:29:57 +00:00
uno_getRegisteredEnvironments(&ppJava_environments, &size, (uno_memAlloc)malloc, sJava.pData);
2002-09-17 12:43:40 +00:00
if(!size)
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - no Java environment available")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
// why is there no convinient contructor?
OUString sCppu_current_lb_name(RTL_CONSTASCII_USTRINGPARAM(CPPU_CURRENT_LANGUAGE_BINDING_NAME));
uno_getEnvironment(&pUno_environment, sCppu_current_lb_name.pData, NULL);
2002-09-17 12:43:40 +00:00
if(!pUno_environment)
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - no C++ environment available")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
Mapping java_curr(ppJava_environments[0], pUno_environment);
2002-09-17 12:43:40 +00:00
if(!java_curr.is())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - no mapping from java to C++ ")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
// release all java environments
for(sal_Int32 i = 0; i < size; ++ i)
ppJava_environments[i]->release(ppJava_environments[i]);
size = 0;
free(ppJava_environments);
2000-09-18 14:29:57 +00:00
// release uno environment
pUno_environment->release(pUno_environment);
pUno_environment = NULL;
2000-09-18 14:29:57 +00:00
getCppuType((Reference<XImplementationLoader> *) 0).getDescription((typelib_TypeDescription **) & pType_XImplementationLoader);
2002-09-17 12:43:40 +00:00
if(!pType_XImplementationLoader)
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - no type information for XImplementationLoader")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
2002-09-17 12:43:40 +00:00
_javaLoader = Reference<XImplementationLoader>(reinterpret_cast<XImplementationLoader *>(
java_curr.mapInterface(joJavaLoader, pType_XImplementationLoader)));
2002-10-29 08:25:58 +00:00
pJNIEnv->DeleteLocalRef( joJavaLoader );
2002-09-17 12:43:40 +00:00
if(!_javaLoader.is())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - mapping of java XImplementationLoader to c++ failed")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
typelib_typedescription_release(reinterpret_cast<typelib_TypeDescription *>(pType_XImplementationLoader));
pType_XImplementationLoader = NULL;
2000-09-18 14:29:57 +00:00
// detach from java vm
if(xJavaThreadRegister_11.is()) {
xJavaThreadRegister_11->revokeThread();
2000-09-18 14:29:57 +00:00
if(!xJavaThreadRegister_11->isThreadAttached())
pJavaVM->DetachCurrentThread();
}
else
pJavaVM->DetachCurrentThread();
2000-09-18 14:29:57 +00:00
pJNIEnv = NULL;
2000-09-18 14:29:57 +00:00
// set the service manager at the javaloader
Reference<XInitialization> javaLoader_XInitialization(_javaLoader, UNO_QUERY);
2002-09-17 12:43:40 +00:00
if(!javaLoader_XInitialization.is())
throw RuntimeException(OUString(RTL_CONSTASCII_USTRINGPARAM(
"javaloader error - initialization of java javaloader failed, no XInitialization")), Reference<XInterface>());
2000-09-18 14:29:57 +00:00
Any any;
any <<= _xSMgr;
2000-09-18 14:29:57 +00:00
javaLoader_XInitialization->initialize(Sequence<Any>(&any, 1));
}
catch(RuntimeException & runtimeException) {
// release all java environments
for(sal_Int32 i = 0; i < size; ++ i)
ppJava_environments[i]->release(ppJava_environments[i]);
2000-09-18 14:29:57 +00:00
if(pUno_environment)
pUno_environment->release(pUno_environment);
2000-09-18 14:29:57 +00:00
if(pType_XImplementationLoader)
typelib_typedescription_release(reinterpret_cast<typelib_TypeDescription *>(pType_XImplementationLoader));
2000-09-18 14:29:57 +00:00
// detach from java vm
if(pJNIEnv)
if(xJavaThreadRegister_11.is()) {
xJavaThreadRegister_11->revokeThread();
2000-09-18 14:29:57 +00:00
if(!xJavaThreadRegister_11->isThreadAttached())
pJavaVM->DetachCurrentThread();
}
else
pJavaVM->DetachCurrentThread();
2000-09-18 14:29:57 +00:00
throw;
2000-09-18 14:29:57 +00:00
}
OSL_TRACE("javaloader.cxx: mapped javaloader - 0x%x", _javaLoader.get());
}
2000-09-18 14:29:57 +00:00
JavaComponentLoader::~JavaComponentLoader() throw() {
2000-09-18 14:29:57 +00:00
}
// XServiceInfo
OUString SAL_CALL JavaComponentLoader::getImplementationName() throw(::com::sun::star::uno::RuntimeException)
{
return loader_getImplementationName();
2000-09-18 14:29:57 +00:00
}
sal_Bool SAL_CALL JavaComponentLoader::supportsService(const OUString & ServiceName) throw(::com::sun::star::uno::RuntimeException) {
sal_Bool bSupport = sal_False;
Sequence<OUString> aSNL = getSupportedServiceNames();
const OUString * pArray = aSNL.getArray();
for(sal_Int32 i = 0; i < aSNL.getLength() && !bSupport; ++ i)
bSupport = pArray[i] == ServiceName;
return bSupport;
}
Sequence<OUString> SAL_CALL JavaComponentLoader::getSupportedServiceNames() throw(::com::sun::star::uno::RuntimeException)
{
return loader_getSupportedServiceNames();
2000-09-18 14:29:57 +00:00
}
// XImplementationLoader
sal_Bool SAL_CALL JavaComponentLoader::writeRegistryInfo(const Reference<XRegistryKey> & xKey, const OUString & blabla, const OUString & rLibName)
throw(CannotRegisterImplementationException, RuntimeException)
{
return _javaLoader->writeRegistryInfo(xKey, blabla, rLibName);
2000-09-18 14:29:57 +00:00
}
Reference<XInterface> SAL_CALL JavaComponentLoader::activate(const OUString & rImplName,
const OUString & blabla,
const OUString & rLibName,
const Reference<XRegistryKey> & xKey)
throw(CannotActivateFactoryException, RuntimeException)
{
return _javaLoader->activate(rImplName, blabla, rLibName, xKey);
2000-09-18 14:29:57 +00:00
}
static Mutex & getInitMutex()
{
static Mutex * pMutex = 0;
if( ! pMutex )
{
MutexGuard guard( Mutex::getGlobalMutex() );
if( ! pMutex )
{
static Mutex mutex;
pMutex = &mutex;
}
}
return *pMutex;
2000-09-18 14:29:57 +00:00
}
Reference<XInterface> SAL_CALL JavaComponentLoader_CreateInstance(const Reference<XComponentContext> & xCtx) throw(Exception)
{
Reference<XInterface> xRet;
2000-09-18 14:29:57 +00:00
try {
MutexGuard guard( getInitMutex() );
// The javaloader is never destroyed and there can be only one!
// Note that the first context wins ....
static Reference< XInterface > *pStaticRef = 0;
if( pStaticRef )
{
xRet = *pStaticRef;
2000-09-18 14:29:57 +00:00
}
else
{
xRet = *new JavaComponentLoader(xCtx);
pStaticRef = new Reference< XInterface > ( xRet );
2000-09-18 14:29:57 +00:00
}
}
catch(RuntimeException & runtimeException) {
OString message = OUStringToOString(runtimeException.Message, RTL_TEXTENCODING_ASCII_US);
osl_trace("javaloader - could not init javaloader cause of %s", message.getStr());
}
2000-09-18 14:29:57 +00:00
return xRet;
2000-09-18 14:29:57 +00:00
}
}
2000-09-18 14:29:57 +00:00
using namespace stoc_javaloader;
2000-09-18 14:29:57 +00:00
static struct ImplementationEntry g_entries[] =
{
{
JavaComponentLoader_CreateInstance, loader_getImplementationName,
loader_getSupportedServiceNames, createSingleComponentFactory,
0 , 0
},
{ 0, 0, 0, 0, 0, 0 }
};
2000-09-18 14:29:57 +00:00
extern "C"
{
// NOTE: component_canUnload is not exported, as the library cannot be unloaded.
2000-09-18 14:29:57 +00:00
//==================================================================================================
void SAL_CALL component_getImplementationEnvironment(
const sal_Char ** ppEnvTypeName, uno_Environment ** ppEnv )
{
*ppEnvTypeName = CPPU_CURRENT_LANGUAGE_BINDING_NAME;
}
//==================================================================================================
sal_Bool SAL_CALL component_writeInfo(
void * pServiceManager, void * pRegistryKey )
{
return component_writeInfoHelper( pServiceManager, pRegistryKey, g_entries );
}
//==================================================================================================
void * SAL_CALL component_getFactory(
const sal_Char * pImplName, void * pServiceManager, void * pRegistryKey )
{
return component_getFactoryHelper( pImplName, pServiceManager, pRegistryKey , g_entries );
}
2000-09-18 14:29:57 +00:00
}