Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / bridges / source / jni_uno / jni_base.h
blob0d02261adb2a133a66c22429e4fbc106d071481e
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #pragma once
22 #include <sal/config.h>
24 #include <cassert>
26 #include <jvmaccess/unovirtualmachine.hxx>
27 #include <jvmaccess/virtualmachine.hxx>
29 #include <osl/diagnose.h>
31 #include <rtl/alloc.h>
32 #include <rtl/ustring.hxx>
33 #include <sal/log.hxx>
34 #include <utility>
36 #include <uno/environment.h>
37 #include <typelib/typedescription.h>
40 namespace jni_uno
43 class JNI_info;
45 struct BridgeRuntimeError
47 OUString m_message;
49 explicit BridgeRuntimeError( OUString message )
50 : m_message(std::move( message ))
55 class JNI_context
57 JNI_info const * m_jni_info;
58 JNIEnv * m_env;
59 jobject m_class_loader;
61 JNI_context( JNI_context const & ) = delete;
62 JNI_context& operator = ( JNI_context const &) = delete;
64 void java_exc_occurred() const;
65 public:
66 explicit JNI_context(
67 JNI_info const * jni_info, JNIEnv * env, jobject class_loader )
68 : m_jni_info( jni_info ),
69 m_env( env ),
70 m_class_loader( class_loader )
73 JNI_info const * get_info() const
74 { return m_jni_info; }
76 JNIEnv * operator -> () const
77 { return m_env; }
78 JNIEnv * get_jni_env() const
79 { return m_env; }
81 // does not handle exceptions, *classClass will be null if exception
82 // occurred:
83 void getClassForName(jclass * classClass, jmethodID * methodForName) const;
85 // if inException, does not handle exceptions, in which case returned value
86 // will be null if exception occurred:
87 jclass findClass(
88 char const * name, jclass classClass, jmethodID methodForName,
89 bool inException) const;
91 inline void ensure_no_exception() const; // throws BridgeRuntimeError
92 inline bool assert_no_exception() const; // asserts and clears exception
94 OUString get_stack_trace( jobject jo_exc = nullptr ) const;
97 inline void JNI_context::ensure_no_exception() const
99 if (m_env->ExceptionCheck())
101 java_exc_occurred();
105 inline bool JNI_context::assert_no_exception() const
107 if (m_env->ExceptionCheck())
109 SAL_WARN("bridges", "unexpected java exception occurred");
110 #if OSL_DEBUG_LEVEL > 0
111 m_env->ExceptionDescribe();
112 #endif
113 m_env->ExceptionClear();
114 return false;
116 return true;
120 class JNI_guarded_context
121 : private ::jvmaccess::VirtualMachine::AttachGuard,
122 public JNI_context
124 JNI_guarded_context( JNI_guarded_context const & ) = delete;
125 JNI_guarded_context& operator = ( JNI_guarded_context const &) = delete;
127 public:
128 explicit JNI_guarded_context(
129 JNI_info const * jni_info,
130 rtl::Reference<jvmaccess::UnoVirtualMachine> const & vm_access)
131 : AttachGuard( vm_access->getVirtualMachine() ),
132 JNI_context(
133 jni_info, AttachGuard::getEnvironment(),
134 static_cast< jobject >(vm_access->getClassLoader()) )
139 class JLocalAutoRef
141 JNI_context const & m_jni;
142 jobject m_jo;
144 public:
145 explicit JLocalAutoRef( JNI_context const & jni )
146 : m_jni( jni ),
147 m_jo( nullptr )
149 explicit JLocalAutoRef( JNI_context const & jni, jobject jo )
150 : m_jni( jni ),
151 m_jo( jo )
153 inline JLocalAutoRef( JLocalAutoRef & auto_ref );
154 inline ~JLocalAutoRef();
156 jobject get() const
157 { return m_jo; }
158 bool is() const
159 { return (nullptr != m_jo); }
160 inline jobject release();
161 inline void reset( jobject jo );
162 inline JLocalAutoRef & operator = ( JLocalAutoRef & auto_ref );
165 inline JLocalAutoRef::~JLocalAutoRef()
167 if (nullptr != m_jo)
168 m_jni->DeleteLocalRef( m_jo );
171 inline JLocalAutoRef::JLocalAutoRef( JLocalAutoRef & auto_ref )
172 : m_jni( auto_ref.m_jni ),
173 m_jo( auto_ref.m_jo )
175 auto_ref.m_jo = nullptr;
178 inline jobject JLocalAutoRef::release()
180 jobject jo = m_jo;
181 m_jo = nullptr;
182 return jo;
185 inline void JLocalAutoRef::reset( jobject jo )
187 if (jo != m_jo)
189 if (nullptr != m_jo)
190 m_jni->DeleteLocalRef( m_jo );
191 m_jo = jo;
195 inline JLocalAutoRef & JLocalAutoRef::operator = ( JLocalAutoRef & auto_ref )
197 assert( m_jni.get_jni_env() == auto_ref.m_jni.get_jni_env() );
198 reset( auto_ref.m_jo );
199 auto_ref.m_jo = nullptr;
200 return *this;
205 struct rtl_mem
207 static void * operator new ( size_t nSize )
208 { return std::malloc( nSize ); }
209 static void operator delete ( void * mem )
210 { std::free( mem ); }
211 static void * operator new ( size_t, void * mem )
212 { return mem; }
213 static void operator delete ( void *, void * )
216 static inline rtl_mem * allocate( std::size_t bytes );
219 inline rtl_mem * rtl_mem::allocate( std::size_t bytes )
221 void * p = std::malloc( bytes );
222 if (nullptr == p)
223 throw BridgeRuntimeError( "out of memory!" );
224 return static_cast<rtl_mem *>(p);
228 class TypeDescr
230 typelib_TypeDescription * m_td;
232 TypeDescr( TypeDescr const & ) = delete;
233 TypeDescr& operator = ( TypeDescr const & ) = delete;
235 public:
236 inline explicit TypeDescr( typelib_TypeDescriptionReference * td_ref );
237 ~TypeDescr()
238 { TYPELIB_DANGER_RELEASE( m_td ); }
240 typelib_TypeDescription * get() const
241 { return m_td; }
244 inline TypeDescr::TypeDescr( typelib_TypeDescriptionReference * td_ref )
245 : m_td( nullptr )
247 TYPELIB_DANGER_GET( &m_td, td_ref );
248 if (nullptr == m_td)
250 throw BridgeRuntimeError(
251 "cannot get comprehensive type description for " +
252 OUString::unacquired( &td_ref->pTypeName ) );
258 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */