📄 jniutil.cpp
字号:
/** * @copyright * ==================================================================== * Copyright (c) 2003 CollabNet. All rights reserved. * * This software is licensed as described in the file COPYING, which * you should have received as part of this distribution. The terms * are also available at http://subversion.tigris.org/license-1.html. * If newer versions of this license are posted there, you may use a * newer version instead, at your option. * * This software consists of voluntary contributions made by many * individuals. For exact contribution history, see the revision * history and logs, available at http://subversion.tigris.org/. * ==================================================================== * @endcopyright * * @file JNIUtil.cpp * @brief Implementation of the class JNIUtil */#include "JNIUtil.h"#include <locale.h>#include <apr_strings.h>#include <apr_tables.h>#include <apr_general.h>#include <apr_lib.h>#include "svn_pools.h"#include "svn_wc.h"#include "svn_path.h"#include <apr_file_info.h>#include "svn_private_config.h"#ifdef WIN32/* FIXME: We're using an internal APR header here, which means we have to build Subversion with APR sources. This being Win32-only, that should be fine for now, but a better solution must be found in combination with issue #850. */extern "C" {#include <arch/win32/apr_arch_utf8.h>};#endif#include "SVNBase.h"#include "JNIMutex.h"#include "JNICriticalSection.h"#include "JNIThreadData.h"#include "JNIStringHolder.h"// static members of JNIUtil are allocated hereapr_pool_t *JNIUtil::g_pool = NULL;std::list<SVNBase*> JNIUtil::g_finalizedObjects;JNIMutex *JNIUtil::g_finalizedObjectsMutex = NULL;JNIMutex *JNIUtil::g_logMutex = NULL;JNIMutex *JNIUtil::g_globalPoolMutext = NULL;bool JNIUtil::g_initException;bool JNIUtil::g_inInit;JNIEnv *JNIUtil::g_initEnv;char JNIUtil::g_initFormatBuffer[formatBufferSize];int JNIUtil::g_logLevel = JNIUtil::noLog;std::ofstream JNIUtil::g_logStream;/** * initialize the environment for all requests * @param env the JNI environment for this request */bool JNIUtil::JNIInit(JNIEnv *env){ // clear all standing exceptions. env->ExceptionClear(); // remember the env paramater for the remainder of the request setEnv(env); // lock the list of finalized objects JNICriticalSection cs(*g_finalizedObjectsMutex) ; if(isExceptionThrown()) { return false; } // delete all finalized, but not yet deleted objects for(std::list<SVNBase*>::iterator it = g_finalizedObjects.begin(); it != g_finalizedObjects.end(); it++) { delete *it; } g_finalizedObjects.clear(); return true;}/** * initialize the environment for all requests * @param env the JNI environment for this request */bool JNIUtil::JNIGlobalInit(JNIEnv *env){ // this method has to be run only once during the run a // programm static bool run = false; if(run) // already run { return true; } run = true; // do not run this part more than one time. // this leaves a small time window when two threads create their first // SVNClient & SVNAdmin at the same time, but I do not see a better // option without APR already initialized if(g_inInit) { return false; } g_inInit = true; g_initEnv = env; apr_status_t status; /* C programs default to the "C" locale. But because svn is supposed to be i18n-aware, it should inherit the default locale of its environment. */ if (!setlocale(LC_ALL, "")) { if (stderr) { const char *env_vars[] = { "LC_ALL", "LC_CTYPE", "LANG", NULL }; const char **env_var = &env_vars[0], *env_val = NULL; while (*env_var) { env_val = getenv(*env_var); if (env_val && env_val[0]) break; ++env_var; } if (!*env_var) { /* Unlikely. Can setlocale fail if no env vars are set? */ --env_var; env_val = "not set"; } fprintf(stderr, "%s: error: cannot set LC_ALL locale\n" "%s: error: environment variable %s is %s\n" "%s: error: please check that your locale name is correct\n", "svnjavahl", "svnjavahl", *env_var, env_val, "svnjavahl"); } return FALSE; } /* Initialize the APR subsystem, and register an atexit() function to Uninitialize that subsystem at program exit. */ status = apr_initialize(); if (status) { if (stderr) { char buf[1024]; apr_strerror(status, buf, sizeof(buf) - 1); fprintf(stderr, "%s: error: cannot initialize APR: %s\n", "svnjavahl", buf); } return FALSE; } if (0 > atexit(apr_terminate)) { if (stderr) fprintf(stderr, "%s: error: atexit registration failed\n", "svnjavahl"); return FALSE; }#ifdef ENABLE_NLS#ifdef WIN32 { WCHAR ucs2_path[MAX_PATH]; char* utf8_path; const char* internal_path; apr_pool_t* pool; apr_status_t apr_err; unsigned int inwords, outbytes, outlength; apr_pool_create (&pool, 0); /* get dll name - our locale info will be in '../share/locale' */ inwords = sizeof (ucs2_path) / sizeof(ucs2_path[0]); HINSTANCE moduleHandle = GetModuleHandle("libsvnjavahl-1"); GetModuleFileNameW (moduleHandle, ucs2_path, inwords); inwords = lstrlenW (ucs2_path); outbytes = outlength = 3 * (inwords + 1); utf8_path = (char *)apr_palloc (pool, outlength); apr_err = apr_conv_ucs2_to_utf8 ((const apr_wchar_t *) ucs2_path, &inwords, utf8_path, &outbytes); if (!apr_err && (inwords > 0 || outbytes == 0)) apr_err = APR_INCOMPLETE; if (apr_err) { if (stderr) fprintf (stderr, "Can't convert module path to UTF-8"); return FALSE; } utf8_path[outlength - outbytes] = '\0'; internal_path = svn_path_internal_style (utf8_path, pool); /* get base path name */ internal_path = svn_path_dirname (internal_path, pool); internal_path = svn_path_join (internal_path, SVN_LOCALE_RELATIVE_PATH, pool); bindtextdomain (PACKAGE_NAME, internal_path); apr_pool_destroy (pool); }#else bindtextdomain(PACKAGE_NAME, SVN_LOCALE_DIR);#endif textdomain(PACKAGE_NAME);#endif /* Create our top-level pool. */ g_pool = svn_pool_create (NULL);#if defined(WIN32) || defined(__CYGWIN__) /* See http://svn.collab.net/repos/svn/trunk/notes/asp-dot-net-hack.txt */ /* ### This code really only needs to be invoked by consumers of ### the libsvn_wc library, which basically means SVNClient. */ if (getenv ("SVN_ASP_DOT_NET_HACK")) { svn_error_t *err = svn_wc_set_adm_dir("_svn", g_pool); if (err) { if (stderr) { fprintf(stderr, "%s: error: SVN_ASP_DOT_NET_HACK failed: %s\n", "svnjavahl", err->message); } svn_error_clear(err); return FALSE; } }#endif // build all mutexes g_finalizedObjectsMutex = new JNIMutex(g_pool); if(isExceptionThrown()) { return false; } g_logMutex = new JNIMutex(g_pool); if(isExceptionThrown()) { return false; } g_globalPoolMutext = new JNIMutex(g_pool); if(isExceptionThrown()) { return false; } // initialized the thread local storage if(!JNIThreadData::initThreadData()) { return false; } setEnv(env); if(isExceptionThrown()) { return false; } g_initEnv = NULL; g_inInit = false; return true;}/** * returns the global (not request specific) pool * @return global pool */apr_pool_t * JNIUtil::getPool(){ return g_pool;}/** * return the mutex securing the global pool * @return the mutex for the global pool */JNIMutex *JNIUtil::getGlobalPoolMutex(){ return g_globalPoolMutext;}void JNIUtil::raiseThrowable(const char *name, const char *message){ if (getLogLevel() >= errorLog) { JNICriticalSection cs(*g_logMutex); g_logStream << "Throwable raised <" << message << ">" << std::endl; } JNIEnv *env = getEnv(); jclass clazz = env->FindClass(name); if (isJavaExceptionThrown()) { return; } env->ThrowNew(clazz, message); setExceptionThrown(); env->DeleteLocalRef(clazz);}jstring JNIUtil::makeSVNErrorMessage(svn_error_t *err){ if(err == NULL) return NULL; std::string buffer; assembleErrorMessage(err, 0, APR_SUCCESS, buffer); jstring jmessage = makeJString(buffer.c_str()); return jmessage;}/** * process a svn error by wraping in into a ClientExpection * and throwing that * @param err the error to by handled */void JNIUtil::handleSVNError(svn_error_t *err){ JNIEnv *env = getEnv(); jclass clazz = env->FindClass(JAVA_PACKAGE"/ClientException"); if(getLogLevel() >= exceptionLog) { JNICriticalSection cs(*g_logMutex); g_logStream << "Error SVN exception thrown message:<"; g_logStream << err->message << "> file:<" << err->file <<"> apr-err:<"; g_logStream << err->apr_err << ">" << std::endl; } if(isJavaExceptionThrown()) { svn_error_clear(err); return; } std::string buffer; assembleErrorMessage(err, 0, APR_SUCCESS, buffer); jstring jmessage = makeJString(buffer.c_str()); if(isJavaExceptionThrown()) { svn_error_clear(err); return; } if(isJavaExceptionThrown()) { svn_error_clear(err); return; } jstring jfile = makeJString(err->file); if(isJavaExceptionThrown()) { svn_error_clear(err); return; } jmethodID mid = env->GetMethodID(clazz, "<init>", "(Ljava/lang/String;Ljava/lang/String;I)V"); if(isJavaExceptionThrown()) { svn_error_clear(err); return; } jobject error = env->NewObject(clazz, mid, jmessage, jfile, static_cast<jint>(err->apr_err)); svn_error_clear(err); if(isJavaExceptionThrown()) { return; } env->DeleteLocalRef(clazz); if(isJavaExceptionThrown()) { return; } env->DeleteLocalRef(jmessage); if(isJavaExceptionThrown()) { return; } env->DeleteLocalRef(jfile); if(isJavaExceptionThrown()) { return; } env->Throw(static_cast<jthrowable>(error));}void JNIUtil::putFinalizedClient(SVNBase *object){ enqueueForDeletion(object);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -