summaryrefslogtreecommitdiffstats
path: root/kdecore/klibloader.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kdecore/klibloader.cpp')
-rw-r--r--kdecore/klibloader.cpp580
1 files changed, 0 insertions, 580 deletions
diff --git a/kdecore/klibloader.cpp b/kdecore/klibloader.cpp
deleted file mode 100644
index 290e23154..000000000
--- a/kdecore/klibloader.cpp
+++ /dev/null
@@ -1,580 +0,0 @@
-/* This file is part of the KDE libraries
- Copyright (C) 1999 Torben Weis <weis@kde.org>
- Copyright (C) 2000 Michael Matz <matz@kde.org>
-
- This library is free software; you can redistribute it and/or
- modify it under the terms of the GNU Library General Public
- License version 2 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
- Library General Public License for more details.
-
- You should have received a copy of the GNU Library General Public License
- along with this library; see the file COPYING.LIB. If not, write to
- the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
- Boston, MA 02110-1301, USA.
-*/
-#include "config.h"
-
-#include <config.h>
-#include <tqclipboard.h>
-#include <tqfile.h>
-#include <tqdir.h>
-#include <tqtimer.h>
-#include <tqobjectdict.h>
-
-#include "kapplication.h"
-#include "klibloader.h"
-#include "kstandarddirs.h"
-#include "kdebug.h"
-#include "klocale.h"
-
-#include "ltdl.h"
-
-template class TQAsciiDict<KLibrary>;
-
-#include <stdlib.h> //getenv
-
-
-#if HAVE_DLFCN_H
-# include <dlfcn.h>
-#endif
-
-#ifdef RTLD_GLOBAL
-# define LT_GLOBAL RTLD_GLOBAL
-#else
-# ifdef DL_GLOBAL
-# define LT_GLOBAL DL_GLOBAL
-# endif
-#endif /* !RTLD_GLOBAL */
-#ifndef LT_GLOBAL
-# define LT_GLOBAL 0
-#endif /* !LT_GLOBAL */
-
-
-class KLibLoaderPrivate
-{
-public:
- TQPtrList<KLibWrapPrivate> loaded_stack;
- TQPtrList<KLibWrapPrivate> pending_close;
- enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode;
-
- TQString errorMessage;
-};
-
-KLibLoader* KLibLoader::s_self = 0;
-
-// -------------------------------------------------------------------------
-
-KLibFactory::KLibFactory( TQObject* parent, const char* name )
- : TQObject( parent, name )
-{
-}
-
-KLibFactory::~KLibFactory()
-{
-// kdDebug(150) << "Deleting KLibFactory " << this << endl;
-}
-
-TQObject* KLibFactory::create( TQObject* parent, const char* name, const char* classname, const TQStringList &args )
-{
- TQObject* obj = createObject( parent, name, classname, args );
- if ( obj )
- emit objectCreated( obj );
- return obj;
-}
-
-
-TQObject* KLibFactory::createObject( TQObject*, const char*, const char*, const TQStringList &)
-{
- return 0;
-}
-
-
-// -----------------------------------------------
-
-KLibrary::KLibrary( const TQString& libname, const TQString& filename, void * handle )
-{
- /* Make sure, we have a KLibLoader */
- (void) KLibLoader::self();
- m_libname = libname;
- m_filename = filename;
- m_handle = handle;
- m_factory = 0;
- m_timer = 0;
-}
-
-KLibrary::~KLibrary()
-{
-// kdDebug(150) << "Deleting KLibrary " << this << " " << m_libname << endl;
- if ( m_timer && m_timer->isActive() )
- m_timer->stop();
-
- // If any object is remaining, delete
- if ( m_objs.count() > 0 )
- {
- TQPtrListIterator<TQObject> it( m_objs );
- for ( ; it.current() ; ++it )
- {
- kdDebug(150) << "Factory still has object " << it.current() << " " << it.current()->name () << " Library = " << m_libname << endl;
- disconnect( it.current(), TQT_SIGNAL( destroyed() ),
- this, TQT_SLOT( slotObjectDestroyed() ) );
- }
- m_objs.setAutoDelete(true);
- m_objs.clear();
- }
-
- if ( m_factory ) {
-// kdDebug(150) << " ... deleting the factory " << m_factory << endl;
- delete m_factory;
- m_factory = 0L;
- }
-}
-
-TQString KLibrary::name() const
-{
- return m_libname;
-}
-
-TQString KLibrary::fileName() const
-{
- return m_filename;
-}
-
-KLibFactory* KLibrary::factory()
-{
- if ( m_factory )
- return m_factory;
-
- TQCString symname;
- symname.sprintf("init_%s", name().latin1() );
-
- void* sym = symbol( symname );
- if ( !sym )
- {
- KLibLoader::self()->d->errorMessage = i18n( "The library %1 does not offer an %2 function." ).arg( name(), "init_" + name() );
- kdWarning(150) << KLibLoader::self()->d->errorMessage << endl;
- return 0;
- }
-
- typedef KLibFactory* (*t_func)();
- t_func func = (t_func)sym;
- m_factory = func();
-
- if( !m_factory )
- {
- KLibLoader::self()->d->errorMessage = i18n( "The library %1 does not offer a KDE compatible factory." ).arg( name() );
- kdWarning(150) << KLibLoader::self()->d->errorMessage << endl;
- return 0;
- }
-
- connect( m_factory, TQT_SIGNAL( objectCreated( TQObject * ) ),
- this, TQT_SLOT( slotObjectCreated( TQObject * ) ) );
-
- return m_factory;
-}
-
-void* KLibrary::symbol( const char* symname ) const
-{
- void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname );
- if ( !sym )
- {
- KLibLoader::self()->d->errorMessage = "KLibrary: " + TQString::fromLocal8Bit( lt_dlerror() );
- kdWarning(150) << KLibLoader::self()->d->errorMessage << endl;
- return 0;
- }
-
- return sym;
-}
-
-bool KLibrary::hasSymbol( const char* symname ) const
-{
- void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname );
- return (sym != 0L );
-}
-
-void KLibrary::unload() const
-{
- if (KLibLoader::s_self)
- KLibLoader::s_self->unloadLibrary(TQFile::encodeName(name()));
-}
-
-void KLibrary::slotObjectCreated( TQObject *obj )
-{
- if ( !obj )
- return;
-
- if ( m_timer && m_timer->isActive() )
- m_timer->stop();
-
- if ( m_objs.containsRef( obj ) )
- return; // we know this object already
-
- connect( obj, TQT_SIGNAL( destroyed() ),
- this, TQT_SLOT( slotObjectDestroyed() ) );
-
- m_objs.append( obj );
-}
-
-void KLibrary::slotObjectDestroyed()
-{
- m_objs.removeRef( TQT_TQOBJECT_CONST(sender()) );
-
- if ( m_objs.count() == 0 )
- {
-// kdDebug(150) << "KLibrary: shutdown timer for " << name() << " started!"
-// << endl;
-
- if ( !m_timer )
- {
- m_timer = new TQTimer( this, "klibrary_shutdown_timer" );
- connect( m_timer, TQT_SIGNAL( timeout() ),
- this, TQT_SLOT( slotTimeout() ) );
- }
-
- // as long as it's not stable make the timeout short, for debugging
- // pleasure (matz)
- //m_timer->start( 1000*60, true );
- m_timer->start( 1000*10, true );
- }
-}
-
-void KLibrary::slotTimeout()
-{
- if ( m_objs.count() != 0 )
- return;
-
- /* Don't go through KLibLoader::unloadLibrary(), because that uses the
- ref counter, but this timeout means to unconditionally close this library
- The destroyed() signal will take care to remove us from all lists.
- */
- delete this;
-}
-
-// -------------------------------------------------
-
-/* This helper class is needed, because KLibraries can go away without
- being unloaded. So we need some info about KLibraries even after its
- death. */
-class KLibWrapPrivate
-{
-public:
- KLibWrapPrivate(KLibrary *l, lt_dlhandle h);
-
- KLibrary *lib;
- enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode;
- int ref_count;
- lt_dlhandle handle;
- TQString name;
- TQString filename;
-};
-
-KLibWrapPrivate::KLibWrapPrivate(KLibrary *l, lt_dlhandle h)
- : lib(l), ref_count(1), handle(h), name(l->name()), filename(l->fileName())
-{
- unload_mode = UNKNOWN;
- if (lt_dlsym(handle, "__kde_do_not_unload") != 0) {
-// kdDebug(150) << "Will not unload " << name << endl;
- unload_mode = DONT_UNLOAD;
- } else if (lt_dlsym(handle, "__kde_do_unload") != 0) {
- unload_mode = UNLOAD;
- }
-}
-
-KLibLoader* KLibLoader::self()
-{
- if ( !s_self )
- s_self = new KLibLoader;
- return s_self;
-}
-
-void KLibLoader::cleanUp()
-{
- if ( !s_self )
- return;
-
- delete s_self;
- s_self = 0L;
-}
-
-KLibLoader::KLibLoader( TQObject* parent, const char* name )
- : TQObject( parent, name )
-{
- s_self = this;
- d = new KLibLoaderPrivate;
- lt_dlinit();
- d->unload_mode = KLibLoaderPrivate::UNKNOWN;
- if (getenv("KDE_NOUNLOAD") != 0)
- d->unload_mode = KLibLoaderPrivate::DONT_UNLOAD;
- else if (getenv("KDE_DOUNLOAD") != 0)
- d->unload_mode = KLibLoaderPrivate::UNLOAD;
- d->loaded_stack.setAutoDelete( true );
-}
-
-KLibLoader::~KLibLoader()
-{
-// kdDebug(150) << "Deleting KLibLoader " << this << " " << name() << endl;
-
- TQAsciiDictIterator<KLibWrapPrivate> it( m_libs );
- for (; it.current(); ++it )
- {
- kdDebug(150) << "The KLibLoader contains the library " << it.current()->name
- << " (" << it.current()->lib << ")" << endl;
- d->pending_close.append(it.current());
- }
-
- close_pending(0);
-
- delete d;
- d = 0L;
-}
-
-static inline TQCString makeLibName( const char* name )
-{
- TQCString libname(name);
- // only append ".la" if there is no extension
- // this allows to load non-libtool libraries as well
- // (mhk, 20000228)
- int pos = libname.findRev('/');
- if (pos < 0)
- pos = 0;
- if (libname.find('.', pos) < 0)
- libname += ".la";
- return libname;
-}
-
-//static
-TQString KLibLoader::findLibrary( const char * name, const KInstance * instance )
-{
- TQCString libname = makeLibName( name );
-
- // only look up the file if it is not an absolute filename
- // (mhk, 20000228)
- TQString libfile;
- if (!TQDir::isRelativePath(libname))
- libfile = TQFile::decodeName( libname );
- else
- {
- libfile = instance->dirs()->findResource( "module", libname );
- if ( libfile.isEmpty() )
- {
- libfile = instance->dirs()->findResource( "lib", libname );
-#ifndef NDEBUG
- if ( !libfile.isEmpty() && libname.left(3) == "lib" ) // don't warn for tdeinit modules
- kdDebug(150) << "library " << libname << " not found under 'module' but under 'lib'" << endl;
-#endif
- }
- }
- return libfile;
-}
-
-
-KLibrary* KLibLoader::globalLibrary( const char *name )
-{
-KLibrary *tmp;
-int olt_dlopen_flag = lt_dlopen_flag;
-
- lt_dlopen_flag |= LT_GLOBAL;
- kdDebug(150) << "Loading the next library global with flag "
- << lt_dlopen_flag
- << "." << endl;
- tmp = library(name);
- lt_dlopen_flag = olt_dlopen_flag;
-
-return tmp;
-}
-
-
-KLibrary* KLibLoader::library( const char *name )
-{
- if (!name)
- return 0;
-
- KLibWrapPrivate* wrap = m_libs[name];
- if (wrap) {
- /* Nothing to do to load the library. */
- wrap->ref_count++;
- return wrap->lib;
- }
-
- /* Test if this library was loaded at some time, but got
- unloaded meanwhile, whithout being dlclose()'ed. */
- TQPtrListIterator<KLibWrapPrivate> it(d->loaded_stack);
- for (; it.current(); ++it) {
- if (it.current()->name == name)
- wrap = it.current();
- }
-
- if (wrap) {
- d->pending_close.removeRef(wrap);
- if (!wrap->lib) {
- /* This lib only was in loaded_stack, but not in m_libs. */
- wrap->lib = new KLibrary( name, wrap->filename, wrap->handle );
- }
- wrap->ref_count++;
- } else {
- TQString libfile = findLibrary( name );
- if ( libfile.isEmpty() )
- {
- const TQCString libname = makeLibName( name );
-#ifndef NDEBUG
- kdDebug(150) << "library=" << name << ": No file named " << libname << " found in paths." << endl;
-#endif
- d->errorMessage = i18n("Library files for \"%1\" not found in paths.").arg(TQString(libname));
- return 0;
- }
-
- lt_dlhandle handle = lt_dlopen( TQFile::encodeName(libfile) );
- if ( !handle )
- {
- const char* errmsg = lt_dlerror();
- if(errmsg)
- d->errorMessage = TQString::fromLocal8Bit(errmsg);
- else
- d->errorMessage = TQString::null;
- return 0;
- }
- else
- d->errorMessage = TQString::null;
-
- KLibrary *lib = new KLibrary( name, libfile, handle );
- wrap = new KLibWrapPrivate(lib, handle);
- d->loaded_stack.prepend(wrap);
- }
- m_libs.insert( name, wrap );
-
- connect( wrap->lib, TQT_SIGNAL( destroyed() ),
- this, TQT_SLOT( slotLibraryDestroyed() ) );
-
- return wrap->lib;
-}
-
-TQString KLibLoader::lastErrorMessage() const
-{
- return d->errorMessage;
-}
-
-void KLibLoader::unloadLibrary( const char *libname )
-{
- KLibWrapPrivate *wrap = m_libs[ libname ];
- if (!wrap)
- return;
- if (--wrap->ref_count)
- return;
-
-// kdDebug(150) << "closing library " << libname << endl;
-
- m_libs.remove( libname );
-
- disconnect( wrap->lib, TQT_SIGNAL( destroyed() ),
- this, TQT_SLOT( slotLibraryDestroyed() ) );
- close_pending( wrap );
-}
-
-KLibFactory* KLibLoader::factory( const char* name )
-{
- KLibrary* lib = library( name );
- if ( !lib )
- return 0;
-
- return lib->factory();
-}
-
-void KLibLoader::slotLibraryDestroyed()
-{
- const KLibrary *lib = static_cast<const KLibrary *>( sender() );
-
- TQAsciiDictIterator<KLibWrapPrivate> it( m_libs );
- for (; it.current(); ++it )
- if ( it.current()->lib == lib )
- {
- KLibWrapPrivate *wrap = it.current();
- wrap->lib = 0; /* the KLibrary object is already away */
- m_libs.remove( it.currentKey() );
- close_pending( wrap );
- return;
- }
-}
-
-void KLibLoader::close_pending(KLibWrapPrivate *wrap)
-{
- if (wrap && !d->pending_close.containsRef( wrap ))
- d->pending_close.append( wrap );
-
- /* First delete all KLibrary objects in pending_close, but _don't_ unload
- the DSO behind it. */
- TQPtrListIterator<KLibWrapPrivate> it(d->pending_close);
- for (; it.current(); ++it) {
- wrap = it.current();
- if (wrap->lib) {
- disconnect( wrap->lib, TQT_SIGNAL( destroyed() ),
- this, TQT_SLOT( slotLibraryDestroyed() ) );
- KLibrary* to_delete = wrap->lib;
- wrap->lib = 0L; // unset first, because KLibrary dtor can cause
- delete to_delete; // recursive call to close_pending()
- }
- }
-
- if (d->unload_mode == KLibLoaderPrivate::DONT_UNLOAD) {
- d->pending_close.clear();
- return;
- }
-
- bool deleted_one = false;
- while ((wrap = d->loaded_stack.first())) {
- /* Let's first see, if we want to try to unload this lib.
- If the env. var KDE_DOUNLOAD is set, we try to unload every lib.
- If not, we look at the lib itself, and unload it only, if it exports
- the symbol __kde_do_unload. */
- if (d->unload_mode != KLibLoaderPrivate::UNLOAD
- && wrap->unload_mode != KLibWrapPrivate::UNLOAD)
- break;
-
- /* Now ensure, that the libs are only unloaded in the reverse direction
- they were loaded. */
- if (!d->pending_close.containsRef( wrap )) {
- if (!deleted_one)
- /* Only diagnose, if we really haven't deleted anything. */
-// kdDebug(150) << "try to dlclose " << wrap->name << ": not yet" << endl;
- break;
- }
-
-// kdDebug(150) << "try to dlclose " << wrap->name << ": yes, done." << endl;
-
- if ( !deleted_one ) {
- /* Only do the hack once in this loop.
- WABA: *HACK*
- We need to make sure to clear the clipboard before unloading a DSO
- because the DSO could have defined an object derived from QMimeSource
- and placed that on the clipboard. */
- /*kapp->tqclipboard()->clear();*/
-
- /* Well.. let's do something more subtle... convert the clipboard context
- to text. That should be safe as it only uses objects defined by Qt. */
- if( kapp->tqclipboard()->ownsSelection()) {
- kapp->tqclipboard()->setText(
- kapp->tqclipboard()->text( TQClipboard::Selection ), TQClipboard::Selection );
- }
- if( kapp->tqclipboard()->ownsClipboard()) {
- kapp->tqclipboard()->setText(
- kapp->tqclipboard()->text( TQClipboard::Clipboard ), TQClipboard::Clipboard );
- }
- }
-
- deleted_one = true;
- lt_dlclose(wrap->handle);
- d->pending_close.removeRef(wrap);
- /* loaded_stack is AutoDelete, so wrap is freed */
- d->loaded_stack.remove();
- }
-}
-
-void KLibLoader::virtual_hook( int, void* )
-{ /*BASE::virtual_hook( id, data );*/ }
-
-void KLibFactory::virtual_hook( int, void* )
-{ /*BASE::virtual_hook( id, data );*/ }
-
-#include "klibloader.moc"