summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sal/inc/rtl/bootstrap.h3
-rw-r--r--sal/rtl/source/bootstrap.cxx671
2 files changed, 429 insertions, 245 deletions
diff --git a/sal/inc/rtl/bootstrap.h b/sal/inc/rtl/bootstrap.h
index 6e3fc4aee708..98c3de55ef9e 100644
--- a/sal/inc/rtl/bootstrap.h
+++ b/sal/inc/rtl/bootstrap.h
@@ -124,8 +124,7 @@ extern "C" {
If it is never called, the filename is based on the name of the executable,
with the suffix ".ini" on Windows or "rc" on Unix.
- @param pFileUri URL of the ini file; must not be null, must not be the empty
- string
+ @param pNameUri URL of the inifile with path but WITHOUT suffix (.ini or rc)
*/
SAL_DLLPUBLIC void SAL_CALL rtl_bootstrap_setIniFileName( rtl_uString *pFileUri )
SAL_THROW_EXTERN_C();
diff --git a/sal/rtl/source/bootstrap.cxx b/sal/rtl/source/bootstrap.cxx
index e4558536597c..8083eff36ee8 100644
--- a/sal/rtl/source/bootstrap.cxx
+++ b/sal/rtl/source/bootstrap.cxx
@@ -27,7 +27,6 @@
************************************************************************/
-#include "boost/noncopyable.hpp"
#include "rtl/bootstrap.h"
#include "rtl/bootstrap.hxx"
#include <osl/diagnose.h>
@@ -47,10 +46,8 @@
#include <rtl/uri.hxx>
#include "rtl/allocator.hxx"
-#include <algorithm>
-#include <map>
-#include <memory>
-#include <utility>
+#include <boost/unordered_map.hpp>
+#include <list>
#ifdef IOS
#include <premac.h>
@@ -61,8 +58,7 @@
#define MY_STRING_(x) # x
#define MY_STRING(x) MY_STRING_(x)
-extern "C" oslProcessError SAL_CALL osl_bootstrap_getExecutableFile_Impl(
- rtl_uString ** ppFileURL) SAL_THROW_EXTERN_C();
+//----------------------------------------------------------------------------
using osl::DirectoryItem;
using osl::FileStatus;
@@ -127,95 +123,127 @@ rtl::OUString recursivelyExpandMacros(
return expandMacros(file, text, mode, &link);
}
-class ParameterMap: private boost::noncopyable {
-public:
- bool get(rtl::OUString const & key, rtl::OUString * value) const;
-
-protected:
- ParameterMap() {}
-
- ~ParameterMap() {}
-
- typedef std::map< rtl::OUString, rtl::OUString > Map;
-
- Map map_;
-};
-
-bool ParameterMap::get(rtl::OUString const & key, rtl::OUString * value) const {
- OSL_ASSERT(value != 0);
- Map::const_iterator i(map_.find(key));
- if (i == map_.end()) {
- return false;
- } else {
- *value = i->second;
- return true;
- }
}
-class ExplicitParameterMap: public ParameterMap {
-public:
- bool get(rtl::OUString const & key, rtl::OUString * value) const;
-
- void set(rtl::OUString const & key, rtl::OUString const & value);
+//----------------------------------------------------------------------------
-private:
- mutable osl::Mutex mutex_;
+struct rtl_bootstrap_NameValue
+{
+ OUString sName;
+ OUString sValue;
+
+ inline rtl_bootstrap_NameValue() SAL_THROW( () )
+ {}
+ inline rtl_bootstrap_NameValue(
+ OUString const & name, OUString const & value ) SAL_THROW( () )
+ : sName( name ),
+ sValue( value )
+ {}
};
-bool ExplicitParameterMap::get(rtl::OUString const & key, rtl::OUString * value)
- const
+typedef std::list<
+ rtl_bootstrap_NameValue,
+ rtl::Allocator< rtl_bootstrap_NameValue >
+> NameValueList;
+
+bool find(
+ NameValueList const & list, rtl::OUString const & key,
+ rtl::OUString * value)
{
- osl::MutexGuard g(mutex_);
- return ParameterMap::get(key, value);
+ OSL_ASSERT(value != NULL);
+ for (NameValueList::const_iterator i(list.begin()); i != list.end(); ++i) {
+ if (i->sName == key) {
+ *value = i->sValue;
+ return true;
+ }
+ }
+ return false;
}
-void ExplicitParameterMap::set(
- rtl::OUString const & key, rtl::OUString const & value)
-{
- osl::MutexGuard g(mutex_);
- map_[key] = value;
+namespace {
+ struct rtl_bootstrap_set_list :
+ public rtl::Static< NameValueList, rtl_bootstrap_set_list > {};
}
-struct ExplicitParameters:
- public rtl::Static< ExplicitParameterMap, ExplicitParameters >
-{};
+//----------------------------------------------------------------------------
-class CommandLineParameterMap: public ParameterMap {
-public:
- CommandLineParameterMap();
-};
+static sal_Bool getFromCommandLineArgs(
+ rtl::OUString const & key, rtl::OUString * value )
+{
+ OSL_ASSERT(value != NULL);
+ static NameValueList *pNameValueList = 0;
+ if( ! pNameValueList )
+ {
+ static NameValueList nameValueList;
-CommandLineParameterMap::CommandLineParameterMap() {
- sal_uInt32 n = osl_getCommandArgCount();
- for (sal_uInt32 i = 0; i != n; ++i) {
- rtl::OUString s;
- osl_getCommandArg(i, &s.pData);
- static char const PREFIX[] = "-env:";
- if (s.matchAsciiL(RTL_CONSTASCII_STRINGPARAM(PREFIX))) {
- sal_Int32 j = s.indexOf('=', RTL_CONSTASCII_LENGTH(PREFIX));
- if (j < 0) {
- map_.erase(s.copy(RTL_CONSTASCII_LENGTH(PREFIX)));
- } else {
- map_[
- s.copy(
- RTL_CONSTASCII_LENGTH(PREFIX),
- j - RTL_CONSTASCII_LENGTH(PREFIX))] =
- s.copy(j + 1);
+ sal_Int32 nArgCount = osl_getCommandArgCount();
+ for(sal_Int32 i = 0; i < nArgCount; ++ i)
+ {
+ rtl_uString *pArg = 0;
+ osl_getCommandArg( i, &pArg );
+ if( ('-' == pArg->buffer[0] || '/' == pArg->buffer[0] ) &&
+ 'e' == pArg->buffer[1] &&
+ 'n' == pArg->buffer[2] &&
+ 'v' == pArg->buffer[3] &&
+ ':' == pArg->buffer[4] )
+ {
+ sal_Int32 nIndex = rtl_ustr_indexOfChar( pArg->buffer, '=' );
+ if( nIndex >= 0 )
+ {
+
+ rtl_bootstrap_NameValue nameValue;
+ nameValue.sName = OUString( &(pArg->buffer[5]), nIndex - 5 );
+ nameValue.sValue = OUString( &(pArg->buffer[nIndex+1]) );
+ if( i == nArgCount-1 &&
+ nameValue.sValue.getLength() &&
+ nameValue.sValue[nameValue.sValue.getLength()-1] == 13 )
+ {
+ // avoid the 13 linefeed for the last argument,
+ // when the executable is started from a script,
+ // that was edited on windows
+ nameValue.sValue = nameValue.sValue.copy(0,nameValue.sValue.getLength()-1);
+ }
+ nameValueList.push_back( nameValue );
+ }
}
+ rtl_uString_release( pArg );
}
+ pNameValueList = &nameValueList;
}
+
+ sal_Bool found = sal_False;
+
+ for( NameValueList::iterator ii = pNameValueList->begin() ;
+ ii != pNameValueList->end() ;
+ ++ii )
+ {
+ if( (*ii).sName.equals(key) )
+ {
+ *value = (*ii).sValue;
+ found = sal_True;
+ break;
+ }
+ }
+
+ return found;
}
-struct CommandLineParameters:
- public rtl::Static< CommandLineParameterMap, CommandLineParameters >
-{};
+//----------------------------------------------------------------------------
+extern "C" oslProcessError SAL_CALL osl_bootstrap_getExecutableFile_Impl (
+ rtl_uString ** ppFileURL) SAL_THROW_EXTERN_C();
+
+inline void getExecutableFile_Impl (rtl_uString ** ppFileURL)
+{
+ osl_bootstrap_getExecutableFile_Impl (ppFileURL);
}
+//----------------------------------------------------------------------------
+
static void getExecutableDirectory_Impl (rtl_uString ** ppDirURL)
{
OUString fileName;
- osl_bootstrap_getExecutableFile_Impl (&(fileName.pData));
+ getExecutableFile_Impl (&(fileName.pData));
sal_Int32 nDirEnd = fileName.lastIndexOf('/');
OSL_ENSURE(nDirEnd >= 0, "Cannot locate executable directory");
@@ -223,6 +251,68 @@ static void getExecutableDirectory_Impl (rtl_uString ** ppDirURL)
rtl_uString_newFromStr_WithLength(ppDirURL,fileName.getStr(),nDirEnd);
}
+//----------------------------------------------------------------------------
+
+static OUString & getIniFileName_Impl()
+{
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+ static OUString *pStaticName = 0;
+ if( ! pStaticName )
+ {
+ OUString fileName;
+
+#if defined IOS
+ // On iOS hardcode the inifile as "rc" in the .app
+ // directory. Apps are self-contained anyway, there is no
+ // possibility to have several "applications" in the same
+ // installation location with different inifiles.
+ const char *inifile = [[@"vnd.sun.star.pathname:" stringByAppendingString: [[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent: @"rc"]] UTF8String];
+ uri = rtl::OUString(inifile, strlen(inifile), RTL_TEXTENCODING_UTF8);
+ resolvePathnameUrl(&uri);
+#elif defined ANDROID
+ // Apps are self-contained on Android, too, can as well hardcode
+ // it as "rc" in the "/assets" directory, i.e. inside the app's
+ // .apk (zip) archive as the /assets/rc file.
+ uri = rtl::OUString("vnd.sun.star.pathname:/assets/rc");
+ resolvePathnameUrl(&uri);
+#else
+ if(getFromCommandLineArgs(
+ OUString(RTL_CONSTASCII_USTRINGPARAM("INIFILENAME")), &fileName))
+ {
+ resolvePathnameUrl(&fileName);
+ }
+ else
+ {
+ getExecutableFile_Impl (&(fileName.pData));
+
+ // get rid of a potential executable extension
+ OUString progExt = ".bin";
+ if(fileName.getLength() > progExt.getLength()
+ && fileName.copy(fileName.getLength() - progExt.getLength()).equalsIgnoreAsciiCase(progExt))
+ fileName = fileName.copy(0, fileName.getLength() - progExt.getLength());
+
+ progExt = ".exe";
+ if(fileName.getLength() > progExt.getLength()
+ && fileName.copy(fileName.getLength() - progExt.getLength()).equalsIgnoreAsciiCase(progExt))
+ fileName = fileName.copy(0, fileName.getLength() - progExt.getLength());
+
+ // append config file suffix
+ fileName += OUString(RTL_CONSTASCII_USTRINGPARAM(SAL_CONFIGFILE("")));
+ }
+#endif
+
+ static OUString theFileName;
+ if(fileName.getLength())
+ theFileName = fileName;
+
+ pStaticName = &theFileName;
+ }
+
+ return *pStaticName;
+}
+
+//----------------------------------------------------------------------------
+
static inline bool path_exists( OUString const & path )
{
DirectoryItem dirItem;
@@ -241,11 +331,21 @@ inline void EnsureNoFinalSlash (rtl::OUString & url)
}
}
-struct Bootstrap_Impl: private ParameterMap
+struct Bootstrap_Impl
{
+ sal_Int32 _nRefCount;
+ Bootstrap_Impl * _base_ini;
+
+ NameValueList _nameValueList;
OUString _iniName;
explicit Bootstrap_Impl (OUString const & rIniName);
+ ~Bootstrap_Impl();
+
+ static void * operator new (std::size_t n) SAL_THROW(())
+ { return rtl_allocateMemory (sal_uInt32(n)); }
+ static void operator delete (void * p , std::size_t) SAL_THROW(())
+ { rtl_freeMemory (p); }
bool getValue(
rtl::OUString const & key, rtl_uString ** value,
@@ -266,8 +366,25 @@ struct Bootstrap_Impl: private ParameterMap
//----------------------------------------------------------------------------
Bootstrap_Impl::Bootstrap_Impl( OUString const & rIniName )
- : _iniName (rIniName)
+ : _nRefCount( 0 ),
+ _base_ini( 0 ),
+ _iniName (rIniName)
{
+ OUString base_ini( getIniFileName_Impl() );
+ // normalize path
+ FileStatus status( osl_FileStatus_Mask_FileURL );
+ DirectoryItem dirItem;
+ if (DirectoryItem::E_None == DirectoryItem::get( base_ini, dirItem ) &&
+ DirectoryItem::E_None == dirItem.getFileStatus( status ))
+ {
+ base_ini = status.getFileURL();
+ if (! rIniName.equals( base_ini ))
+ {
+ _base_ini = static_cast< Bootstrap_Impl * >(
+ rtl_bootstrap_args_open( base_ini.pData ) );
+ }
+ }
+
#if OSL_DEBUG_LEVEL > 1
OString sFile = OUStringToOString(_iniName, RTL_TEXTENCODING_ASCII_US);
OSL_TRACE("Bootstrap_Impl(): sFile=%s", sFile.getStr());
@@ -285,20 +402,21 @@ Bootstrap_Impl::Bootstrap_Impl( OUString const & rIniName )
sal_Int32 nIndex = line.indexOf('=');
if (nIndex >= 1)
{
- rtl::OUString sName = OStringToOUString(
+ struct rtl_bootstrap_NameValue nameValue;
+ nameValue.sName = OStringToOUString(
line.copy(0,nIndex).trim(), RTL_TEXTENCODING_ASCII_US );
- rtl::OUString sValue = OStringToOUString(
+ nameValue.sValue = OStringToOUString(
line.copy(nIndex+1).trim(), RTL_TEXTENCODING_UTF8 );
#if OSL_DEBUG_LEVEL > 1
- OString name_tmp = OUStringToOString(sName, RTL_TEXTENCODING_ASCII_US);
- OString value_tmp = OUStringToOString(sValue, RTL_TEXTENCODING_UTF8);
+ OString name_tmp = OUStringToOString(nameValue.sName, RTL_TEXTENCODING_ASCII_US);
+ OString value_tmp = OUStringToOString(nameValue.sValue, RTL_TEXTENCODING_UTF8);
OSL_TRACE(
"pushing: name=%s value=%s",
name_tmp.getStr(), value_tmp.getStr() );
#endif /* OSL_DEBUG_LEVEL > 1 */
- map_[sName] = sValue;
+ _nameValueList.push_back(nameValue);
}
}
osl_closeFile(handle);
@@ -312,150 +430,60 @@ Bootstrap_Impl::Bootstrap_Impl( OUString const & rIniName )
#endif /* OSL_DEBUG_LEVEL > 1 */
}
-namespace {
-
-class BootstrapMap: private boost::noncopyable {
-public:
- BootstrapMap();
-
- ~BootstrapMap();
-
- Bootstrap_Impl * getIni(rtl::OUString const & uri, bool alwaysCreate);
-
- void setBaseIniUri(rtl::OUString const & uri);
-
- Bootstrap_Impl * getBaseIni();
-
- Bootstrap_Impl * getFundamentalIni();
+//----------------------------------------------------------------------------
-private:
- typedef std::map< rtl::OUString, Bootstrap_Impl * > Map;
-
- osl::Mutex mutex_;
- Map map_;
- rtl::OUString baseIniUri_;
- Bootstrap_Impl * baseIni_;
- bool hasFundamentalIni_;
- Bootstrap_Impl * fundamentalIni_;
-};
+Bootstrap_Impl::~Bootstrap_Impl()
+{
+ if (_base_ini != 0)
+ rtl_bootstrap_args_close( _base_ini );
+}
-BootstrapMap::BootstrapMap():
- baseIni_(0), hasFundamentalIni_(false), fundamentalIni_(0)
-{}
+//----------------------------------------------------------------------------
-BootstrapMap::~BootstrapMap() {
- for (Map::iterator i(map_.begin()); i != map_.end(); ++i) {
- delete i->second;
- }
-}
+namespace {
-Bootstrap_Impl * BootstrapMap::getIni(
- rtl::OUString const & uri, bool alwaysCreate)
+Bootstrap_Impl * get_static_bootstrap_handle() SAL_THROW(())
{
- rtl::OUString normUri; // normalize URI if possible
- DirectoryItem dirItem;
- FileStatus status(osl_FileStatus_Mask_FileURL);
- if (DirectoryItem::get(uri, dirItem) == DirectoryItem::E_None &&
- dirItem.getFileStatus(status) == DirectoryItem::E_None)
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+ static Bootstrap_Impl * s_handle = 0;
+ if (s_handle == 0)
{
- normUri = status.getFileURL();
- } else if (alwaysCreate) {
- normUri = uri;
- } else {
- return 0;
- }
- osl::MutexGuard g(mutex_);
- Map::iterator i(map_.find(normUri));
- if (i == map_.end()) {
- std::auto_ptr< Bootstrap_Impl > b(new Bootstrap_Impl(normUri));
- std::pair< Map::iterator, bool > ins(
- map_.insert(Map::value_type(normUri, b.get())));
- b.release();
- OSL_ASSERT(ins.second);
- i = ins.first;
- }
- return i->second;
-}
-
-void BootstrapMap::setBaseIniUri(rtl::OUString const & uri) {
- OSL_ASSERT(!uri.isEmpty());
- osl::MutexGuard g(mutex_);
- OSL_ASSERT(baseIniUri_.isEmpty() && baseIni_ == 0);
- baseIniUri_ = uri;
-}
-
-Bootstrap_Impl * BootstrapMap::getBaseIni() {
- osl::MutexGuard g(mutex_);
- if (baseIni_ == 0) {
- rtl::OUString uri;
- if (baseIniUri_.isEmpty()) {
-#if defined IOS
- // On iOS hardcode the inifile as "rc" in the .app
- // directory. Apps are self-contained anyway, there is no
- // possibility to have several "applications" in the same
- // installation location with different inifiles.
- const char *inifile = [[@"vnd.sun.star.pathname:" stringByAppendingString: [[[NSBundle mainBundle] bundlePath] stringByAppendingPathComponent: @"rc"]] UTF8String];
- uri = rtl::OUString(inifile, strlen(inifile), RTL_TEXTENCODING_UTF8);
- resolvePathnameUrl(&uri);
-#elif defined ANDROID
- // Apps are self-contained on Android, too, can as well hardcode
- // it as "rc" in the "/assets" directory, i.e. inside the app's
- // .apk (zip) archive as the /assets/rc file.
- uri = rtl::OUString("vnd.sun.star.pathname:/assets/rc");
- resolvePathnameUrl(&uri);
-#else
- if (CommandLineParameters::get().get(
- rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("INIFILENAME")),
- &uri))
- {
- resolvePathnameUrl(&uri);
- } else {
- osl_bootstrap_getExecutableFile_Impl(&uri.pData);
- // Strip potentially two such extensions, to allow for
- // renaming of soffice.bin to soffice.bin.exe so that
- // Visual Studio agrees to start it, if you want to
- // debug it from the start.
- static char const BIN_EXT[] = ".bin";
- static char const EXE_EXT[] = ".exe";
- for (int i = 0; i < 2; i++) {
- if (uri.endsWithAsciiL(RTL_CONSTASCII_STRINGPARAM(BIN_EXT))) {
- uri = uri.copy(
- 0, uri.getLength() - RTL_CONSTASCII_LENGTH(BIN_EXT));
- } else if (uri.endsWithAsciiL(
- RTL_CONSTASCII_STRINGPARAM(EXE_EXT))) {
- uri = uri.copy(
- 0, uri.getLength() - RTL_CONSTASCII_LENGTH(EXE_EXT));
- }
- }
- uri += rtl::OUString(
- RTL_CONSTASCII_USTRINGPARAM(SAL_CONFIGFILE("")));
- }
-#endif
- } else {
- uri = baseIniUri_;
+ OUString iniName (getIniFileName_Impl());
+ s_handle = static_cast< Bootstrap_Impl * >(
+ rtl_bootstrap_args_open( iniName.pData ) );
+ if (s_handle == 0)
+ {
+ Bootstrap_Impl * that = new Bootstrap_Impl( iniName );
+ ++that->_nRefCount;
+ s_handle = that;
}
- baseIni_ = getIni(uri, true);
}
- return baseIni_;
+ return s_handle;
}
-Bootstrap_Impl * BootstrapMap::getFundamentalIni() {
- osl::MutexGuard g(mutex_);
- if (!hasFundamentalIni_) {
- rtl::OUString uri;
- fundamentalIni_ =
- (getBaseIni()->getValue(
- rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("URE_BOOTSTRAP")),
- &uri.pData, 0, LOOKUP_MODE_NORMAL, false, 0) &&
+struct FundamentalIniData {
+ rtlBootstrapHandle ini;
+
+ FundamentalIniData() {
+ OUString uri;
+ ini =
+ ((static_cast< Bootstrap_Impl * >(get_static_bootstrap_handle())->
+ getValue(
+ rtl::OUString(
+ RTL_CONSTASCII_USTRINGPARAM("URE_BOOTSTRAP")),
+ &uri.pData, 0, LOOKUP_MODE_NORMAL, false, 0)) &&
resolvePathnameUrl(&uri))
- ? getIni(uri, false) : 0;
- hasFundamentalIni_ = true;
+ ? rtl_bootstrap_args_open(uri.pData) : NULL;
}
- return fundamentalIni_;
-}
-struct BootstrapMapSingleton:
- public rtl::Static< BootstrapMap, BootstrapMapSingleton >
+ ~FundamentalIniData() { rtl_bootstrap_args_close(ini); }
+
+private:
+ FundamentalIniData(FundamentalIniData &); // not defined
+ void operator =(FundamentalIniData &); // not defined
+};
+
+struct FundamentalIni: public rtl::Static< FundamentalIniData, FundamentalIni >
{};
}
@@ -518,16 +546,18 @@ bool Bootstrap_Impl::getValue(
getExecutableDirectory_Impl(value);
return true;
}
- Bootstrap_Impl * b = BootstrapMapSingleton::get().getBaseIni();
- if (b != this && b->getDirectValue(key, value, mode, requestStack)) {
+ if (_base_ini != NULL &&
+ _base_ini->getDirectValue(key, value, mode, requestStack))
+ {
return true;
}
if (!override && getDirectValue(key, value, mode, requestStack)) {
return true;
}
if (mode == LOOKUP_MODE_NORMAL) {
- b = BootstrapMapSingleton::get().getFundamentalIni();
- if (b != 0 && b != this &&
+ FundamentalIniData const & d = FundamentalIni::get();
+ Bootstrap_Impl const * b = static_cast<Bootstrap_Impl const *>(d.ini);
+ if (b != NULL && b != this &&
b->getDirectValue(key, value, mode, requestStack))
{
return true;
@@ -546,7 +576,7 @@ bool Bootstrap_Impl::getDirectValue(
ExpandRequestLink const * requestStack) const
{
rtl::OUString v;
- if (get(key, &v)) {
+ if (find(_nameValueList, key, &v)) {
expandValue(value, v, mode, this, key, requestStack);
return true;
} else {
@@ -559,8 +589,12 @@ bool Bootstrap_Impl::getAmbienceValue(
ExpandRequestLink const * requestStack) const
{
rtl::OUString v;
- if (ExplicitParameters::get().get(key, &v) ||
- CommandLineParameters::get().get(key, &v) ||
+ bool f;
+ {
+ osl::MutexGuard g(osl::Mutex::getGlobalMutex());
+ f = find(rtl_bootstrap_set_list::get(), key, &v);
+ }
+ if (f || getFromCommandLineArgs(key, &v) ||
osl_getEnvironment(key.pData, &v.pData) == osl_Process_E_None)
{
expandValue(value, v, mode, NULL, key, requestStack);
@@ -586,20 +620,131 @@ void Bootstrap_Impl::expandValue(
requestFile, requestKey, requestStack)).pData);
}
+namespace {
+
+struct bootstrap_map {
+ typedef boost::unordered_map<
+ rtl::OUString, Bootstrap_Impl *,
+ rtl::OUStringHash, std::equal_to< rtl::OUString >,
+ rtl::Allocator< OUString > > t;
+
+ // get and release must only be called properly synchronized via some mutex
+ // (e.g., osl::Mutex::getGlobalMutex()):
+
+ static t * get() {
+ if (m_map == NULL) {
+ m_map = new t;
+ }
+ return m_map;
+ }
+
+ static void release() {
+ if (m_map != NULL && m_map->empty()) {
+ delete m_map;
+ m_map = NULL;
+ }
+ }
+
+private:
+ bootstrap_map(); // not defined
+
+ static t * m_map;
+};
+
+bootstrap_map::t * bootstrap_map::m_map = NULL;
+
+}
+
+//----------------------------------------------------------------------------
+
rtlBootstrapHandle SAL_CALL rtl_bootstrap_args_open (
rtl_uString * pIniName
) SAL_THROW_EXTERN_C()
{
- return BootstrapMapSingleton::get().getIni(rtl::OUString(pIniName), false);
+ OUString iniName( pIniName );
+
+ // normalize path
+ FileStatus status( osl_FileStatus_Mask_FileURL );
+ DirectoryItem dirItem;
+ if (DirectoryItem::E_None != DirectoryItem::get( iniName, dirItem ) ||
+ DirectoryItem::E_None != dirItem.getFileStatus( status ))
+ {
+ return 0;
+ }
+ iniName = status.getFileURL();
+
+ Bootstrap_Impl * that;
+ osl::ResettableMutexGuard guard( osl::Mutex::getGlobalMutex() );
+ bootstrap_map::t* p_bootstrap_map = bootstrap_map::get();
+ bootstrap_map::t::const_iterator iFind( p_bootstrap_map->find( iniName ) );
+ if (iFind == p_bootstrap_map->end())
+ {
+ bootstrap_map::release();
+ guard.clear();
+ that = new Bootstrap_Impl( iniName );
+ guard.reset();
+ p_bootstrap_map = bootstrap_map::get();
+ iFind = p_bootstrap_map->find( iniName );
+ if (iFind == p_bootstrap_map->end())
+ {
+ ++that->_nRefCount;
+ ::std::pair< bootstrap_map::t::iterator, bool > insertion(
+ p_bootstrap_map->insert(
+ bootstrap_map::t::value_type( iniName, that ) ) );
+ OSL_ASSERT( insertion.second );
+ }
+ else
+ {
+ Bootstrap_Impl * obsolete = that;
+ that = iFind->second;
+ ++that->_nRefCount;
+ guard.clear();
+ delete obsolete;
+ }
+ }
+ else
+ {
+ that = iFind->second;
+ ++that->_nRefCount;
+ }
+ return static_cast< rtlBootstrapHandle >( that );
}
//----------------------------------------------------------------------------
void SAL_CALL rtl_bootstrap_args_close (
- SAL_UNUSED_PARAMETER rtlBootstrapHandle
+ rtlBootstrapHandle handle
) SAL_THROW_EXTERN_C()
{
- // do nothing; the BootstrapMap::map_ just keeps growing for now
+ if (handle == 0)
+ return;
+ Bootstrap_Impl * that = static_cast< Bootstrap_Impl * >( handle );
+
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+ bootstrap_map::t* p_bootstrap_map = bootstrap_map::get();
+ OSL_ASSERT(
+ p_bootstrap_map->find( that->_iniName )->second == that );
+ --that->_nRefCount;
+ if (that->_nRefCount == 0)
+ {
+ ::std::size_t nLeaking = 8; // only hold up to 8 files statically
+
+#if OSL_DEBUG_LEVEL == 1 // nonpro
+ nLeaking = 0;
+#elif OSL_DEBUG_LEVEL > 1 // debug
+ nLeaking = 1;
+#endif /* OSL_DEBUG_LEVEL */
+
+ if (p_bootstrap_map->size() > nLeaking)
+ {
+ ::std::size_t erased = p_bootstrap_map->erase( that->_iniName );
+ if (erased != 1) {
+ OSL_ASSERT( false );
+ }
+ delete that;
+ }
+ bootstrap_map::release();
+ }
}
//----------------------------------------------------------------------------
@@ -611,11 +756,18 @@ sal_Bool SAL_CALL rtl_bootstrap_get_from_handle(
rtl_uString * pDefault
) SAL_THROW_EXTERN_C()
{
- return
- (handle == 0
- ? BootstrapMapSingleton::get().getBaseIni()
- : static_cast< Bootstrap_Impl * >(handle))->
- getValue(pName, ppValue, pDefault, LOOKUP_MODE_NORMAL, false, 0);
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+
+ sal_Bool found = sal_False;
+ if(ppValue && pName)
+ {
+ if (handle == 0)
+ handle = get_static_bootstrap_handle();
+ found = static_cast< Bootstrap_Impl * >( handle )->getValue(
+ pName, ppValue, pDefault, LOOKUP_MODE_NORMAL, false, NULL );
+ }
+
+ return found;
}
//----------------------------------------------------------------------------
@@ -625,12 +777,19 @@ void SAL_CALL rtl_bootstrap_get_iniName_from_handle (
rtl_uString ** ppIniName
) SAL_THROW_EXTERN_C()
{
- rtl_uString_assign(
- ppIniName,
- ((handle == 0
- ? BootstrapMapSingleton::get().getBaseIni()
- : static_cast<Bootstrap_Impl*>(handle))->
- _iniName.pData));
+ if(ppIniName)
+ {
+ if(handle)
+ {
+ Bootstrap_Impl * pImpl = static_cast<Bootstrap_Impl*>(handle);
+ rtl_uString_assign(ppIniName, pImpl->_iniName.pData);
+ }
+ else
+ {
+ const OUString & iniName = getIniFileName_Impl();
+ rtl_uString_assign(ppIniName, iniName.pData);
+ }
+ }
}
//----------------------------------------------------------------------------
@@ -639,7 +798,9 @@ void SAL_CALL rtl_bootstrap_setIniFileName (
rtl_uString * pName
) SAL_THROW_EXTERN_C()
{
- BootstrapMapSingleton::get().setBaseIniUri(rtl::OUString(pName));
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+ OUString & file = getIniFileName_Impl();
+ file = pName;
}
//----------------------------------------------------------------------------
@@ -660,7 +821,32 @@ void SAL_CALL rtl_bootstrap_set (
rtl_uString * pValue
) SAL_THROW_EXTERN_C()
{
- ExplicitParameters::get().set(rtl::OUString(pName), rtl::OUString(pValue));
+ const OUString name( pName );
+ const OUString value( pValue );
+
+ osl::MutexGuard guard( osl::Mutex::getGlobalMutex() );
+
+ NameValueList& r_rtl_bootstrap_set_list = rtl_bootstrap_set_list::get();
+ NameValueList::iterator iPos( r_rtl_bootstrap_set_list.begin() );
+ NameValueList::iterator iEnd( r_rtl_bootstrap_set_list.end() );
+ for ( ; iPos != iEnd; ++iPos )
+ {
+ if (iPos->sName.equals( name ))
+ {
+ iPos->sValue = value;
+ return;
+ }
+ }
+
+#if OSL_DEBUG_LEVEL > 1
+ OString cstr_name( OUStringToOString( name, RTL_TEXTENCODING_ASCII_US ) );
+ OString cstr_value( OUStringToOString( value, RTL_TEXTENCODING_ASCII_US ) );
+ OSL_TRACE(
+ "bootstrap.cxx: explicitly setting: name=%s value=%s\n",
+ cstr_name.getStr(), cstr_value.getStr() );
+#endif /* OSL_DEBUG_LEVEL > 1 */
+
+ r_rtl_bootstrap_set_list.push_back( rtl_bootstrap_NameValue( name, value ) );
}
//----------------------------------------------------------------------------
@@ -670,14 +856,13 @@ void SAL_CALL rtl_bootstrap_expandMacros_from_handle (
rtl_uString ** macro
) SAL_THROW_EXTERN_C()
{
- rtl::OUString expanded(
- expandMacros(
- (handle == 0
- ? BootstrapMapSingleton::get().getBaseIni()
- : static_cast< Bootstrap_Impl * >(handle)),
- *reinterpret_cast< OUString const * >(macro), LOOKUP_MODE_NORMAL,
- 0));
- rtl_uString_assign(macro, expanded.pData);
+ if (handle == NULL) {
+ handle = get_static_bootstrap_handle();
+ }
+ OUString expanded( expandMacros( static_cast< Bootstrap_Impl * >( handle ),
+ * reinterpret_cast< OUString const * >( macro ),
+ LOOKUP_MODE_NORMAL, NULL ) );
+ rtl_uString_assign( macro, expanded.pData );
}
//----------------------------------------------------------------------------
@@ -743,7 +928,7 @@ rtl::OUString lookup(
rtl::OUString const & key, ExpandRequestLink const * requestStack)
{
rtl::OUString v;
- (file == 0 ? BootstrapMapSingleton::get().getBaseIni() : file)->getValue(
+ (file == NULL ? get_static_bootstrap_handle() : file)->getValue(
key, &v.pData, NULL, mode, override, requestStack);
return v;
}