-rw-r--r-- | microkde/kdecore/klibloader.cpp | 24 |
1 files changed, 19 insertions, 5 deletions
diff --git a/microkde/kdecore/klibloader.cpp b/microkde/kdecore/klibloader.cpp index 1410308..130cc7c 100644 --- a/microkde/kdecore/klibloader.cpp +++ b/microkde/kdecore/klibloader.cpp | |||
@@ -1,626 +1,640 @@ | |||
1 | /* This file is part of the KDE libraries | 1 | /* This file is part of the KDE libraries |
2 | Copyright (C) 1999 Torben Weis <weis@kde.org> | 2 | Copyright (C) 1999 Torben Weis <weis@kde.org> |
3 | Copyright (C) 2000 Michael Matz <matz@kde.org> | 3 | Copyright (C) 2000 Michael Matz <matz@kde.org> |
4 | 4 | ||
5 | This library is free software; you can redistribute it and/or | 5 | This library is free software; you can redistribute it and/or |
6 | modify it under the terms of the GNU Library General Public | 6 | modify it under the terms of the GNU Library General Public |
7 | License version 2 as published by the Free Software Foundation. | 7 | License version 2 as published by the Free Software Foundation. |
8 | 8 | ||
9 | This library is distributed in the hope that it will be useful, | 9 | This library is distributed in the hope that it will be useful, |
10 | but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
12 | Library General Public License for more details. | 12 | Library General Public License for more details. |
13 | 13 | ||
14 | You should have received a copy of the GNU Library General Public License | 14 | You should have received a copy of the GNU Library General Public License |
15 | along with this library; see the file COPYING.LIB. If not, write to | 15 | along with this library; see the file COPYING.LIB. If not, write to |
16 | the Free Software Foundation, Inc., 59 Temple Place - Suite 330, | 16 | the Free Software Foundation, Inc., 59 Temple Place - Suite 330, |
17 | Boston, MA 02111-1307, USA. | 17 | Boston, MA 02111-1307, USA. |
18 | */ | 18 | */ |
19 | //US #include <config.h> | 19 | //US #include <config.h> |
20 | #include <qclipboard.h> | 20 | #include <qclipboard.h> |
21 | #include <qfile.h> | 21 | #include <qfile.h> |
22 | #include <qtimer.h> | 22 | #include <qtimer.h> |
23 | #include <qobjectdict.h> | 23 | #include <qobjectdict.h> |
24 | #include <qwidgetlist.h> | 24 | #include <qwidgetlist.h> |
25 | #include <qwidget.h> | 25 | #include <qwidget.h> |
26 | 26 | ||
27 | #include "kapplication.h" | 27 | #include "kapplication.h" |
28 | #include "klibloader.h" | 28 | #include "klibloader.h" |
29 | #include "kstandarddirs.h" | 29 | #include "kstandarddirs.h" |
30 | #include "kdebug.h" | 30 | #include "kdebug.h" |
31 | #include "klocale.h" | 31 | #include "klocale.h" |
32 | 32 | ||
33 | //US #include "ltdl.h" | 33 | //US #include "ltdl.h" |
34 | 34 | ||
35 | //US do everything through qlibrary | 35 | //US do everything through qlibrary |
36 | #ifndef DESKTOP_VERSION | 36 | #ifndef DESKTOP_VERSION |
37 | #include <qpe/qpeapplication.h> | 37 | #include <qpe/qpeapplication.h> |
38 | #include <qtopia/qlibrary.h> | 38 | #include <qtopia/qlibrary.h> |
39 | #endif | 39 | #endif |
40 | 40 | ||
41 | /*US | 41 | /*US |
42 | #ifdef Q_WS_X11 | 42 | #ifdef Q_WS_X11 |
43 | #include <X11/Xlib.h> | 43 | #include <X11/Xlib.h> |
44 | #include <X11/Xatom.h> | 44 | #include <X11/Xatom.h> |
45 | #endif | 45 | #endif |
46 | */ | 46 | */ |
47 | template class QAsciiDict<KLibrary>; | 47 | template class QAsciiDict<KLibrary>; |
48 | 48 | ||
49 | #include <stdlib.h> //getenv | 49 | #include <stdlib.h> //getenv |
50 | 50 | ||
51 | /*US | 51 | /*US |
52 | #if HAVE_DLFCN_H | 52 | #if HAVE_DLFCN_H |
53 | # include <dlfcn.h> | 53 | # include <dlfcn.h> |
54 | #endif | 54 | #endif |
55 | 55 | ||
56 | #ifdef RTLD_GLOBAL | 56 | #ifdef RTLD_GLOBAL |
57 | # define LT_GLOBAL RTLD_GLOBAL | 57 | # define LT_GLOBAL RTLD_GLOBAL |
58 | #else | 58 | #else |
59 | # ifdef DL_GLOBAL | 59 | # ifdef DL_GLOBAL |
60 | # define LT_GLOBAL DL_GLOBAL | 60 | # define LT_GLOBAL DL_GLOBAL |
61 | # endif | 61 | # endif |
62 | #endif | 62 | #endif |
63 | #ifndef LT_GLOBAL | 63 | #ifndef LT_GLOBAL |
64 | # define LT_GLOBAL 0 | 64 | # define LT_GLOBAL 0 |
65 | #endif | 65 | #endif |
66 | */ | 66 | */ |
67 | 67 | ||
68 | /*US | 68 | /*US |
69 | extern "C" { | 69 | extern "C" { |
70 | extern int lt_dlopen_flag; | 70 | extern int lt_dlopen_flag; |
71 | } | 71 | } |
72 | */ | 72 | */ |
73 | 73 | ||
74 | KLibFactory::KLibFactory( QObject* parent, const char* name ) | 74 | KLibFactory::KLibFactory( QObject* parent, const char* name ) |
75 | : QObject( parent, name ) | 75 | : QObject( parent, name ) |
76 | { | 76 | { |
77 | } | 77 | } |
78 | 78 | ||
79 | KLibFactory::~KLibFactory() | 79 | KLibFactory::~KLibFactory() |
80 | { | 80 | { |
81 | // kdDebug(150) << "Deleting KLibFactory " << this << endl; | 81 | // kdDebug(150) << "Deleting KLibFactory " << this << endl; |
82 | } | 82 | } |
83 | 83 | ||
84 | QObject* KLibFactory::create( QObject* parent, const char* name, const char* classname, const QStringList &args ) | 84 | QObject* KLibFactory::create( QObject* parent, const char* name, const char* classname, const QStringList &args ) |
85 | { | 85 | { |
86 | QObject* obj = createObject( parent, name, classname, args ); | 86 | QObject* obj = createObject( parent, name, classname, args ); |
87 | if ( obj ) | 87 | if ( obj ) |
88 | emit objectCreated( obj ); | 88 | emit objectCreated( obj ); |
89 | return obj; | 89 | return obj; |
90 | } | 90 | } |
91 | 91 | ||
92 | 92 | ||
93 | QObject* KLibFactory::createObject( QObject*, const char*, const char*, const QStringList &) | 93 | QObject* KLibFactory::createObject( QObject*, const char*, const char*, const QStringList &) |
94 | { | 94 | { |
95 | return 0; | 95 | return 0; |
96 | } | 96 | } |
97 | 97 | ||
98 | 98 | ||
99 | // ----------------------------------------------- | 99 | // ----------------------------------------------- |
100 | 100 | ||
101 | //US KLibrary::KLibrary( const QString& libname, const QString& filename, void * handle ) | 101 | //US KLibrary::KLibrary( const QString& libname, const QString& filename, void * handle ) |
102 | KLibrary::KLibrary( const QString& libname, const QString& filename, QLibrary* handle ) | 102 | KLibrary::KLibrary( const QString& libname, const QString& filename, QLibrary* handle ) |
103 | { | 103 | { |
104 | /* Make sure, we have a KLibLoader */ | 104 | /* Make sure, we have a KLibLoader */ |
105 | (void) KLibLoader::self(); | 105 | (void) KLibLoader::self(); |
106 | m_libname = libname; | 106 | m_libname = libname; |
107 | m_filename = filename; | 107 | m_filename = filename; |
108 | m_handle = handle; | 108 | m_handle = handle; |
109 | m_factory = 0; | 109 | m_factory = 0; |
110 | m_timer = 0; | 110 | m_timer = 0; |
111 | } | 111 | } |
112 | 112 | ||
113 | KLibrary::~KLibrary() | 113 | KLibrary::~KLibrary() |
114 | { | 114 | { |
115 | // kdDebug(150) << "Deleting KLibrary " << this << " " << m_libname << endl; | 115 | // kdDebug(150) << "Deleting KLibrary " << this << " " << m_libname << endl; |
116 | if ( m_timer && m_timer->isActive() ) | 116 | if ( m_timer && m_timer->isActive() ) |
117 | m_timer->stop(); | 117 | m_timer->stop(); |
118 | 118 | ||
119 | // If any object is remaining, delete | 119 | // If any object is remaining, delete |
120 | if ( m_objs.count() > 0 ) | 120 | if ( m_objs.count() > 0 ) |
121 | { | 121 | { |
122 | QPtrListIterator<QObject> it( m_objs ); | 122 | QPtrListIterator<QObject> it( m_objs ); |
123 | for ( ; it.current() ; ++it ) | 123 | for ( ; it.current() ; ++it ) |
124 | { | 124 | { |
125 | kdDebug(150) << "Factory still has object " << it.current() << " " << it.current()->name () << " Library = " << m_libname << endl; | 125 | kdDebug(150) << "Factory still has object " << it.current() << " " << it.current()->name () << " Library = " << m_libname << endl; |
126 | disconnect( it.current(), SIGNAL( destroyed() ), | 126 | disconnect( it.current(), SIGNAL( destroyed() ), |
127 | this, SLOT( slotObjectDestroyed() ) ); | 127 | this, SLOT( slotObjectDestroyed() ) ); |
128 | } | 128 | } |
129 | m_objs.setAutoDelete(true); | 129 | m_objs.setAutoDelete(true); |
130 | m_objs.clear(); | 130 | m_objs.clear(); |
131 | } | 131 | } |
132 | 132 | ||
133 | if ( m_factory ) { | 133 | if ( m_factory ) { |
134 | //kdDebug(150) << " ... deleting the factory " << m_factory << endl; | 134 | //kdDebug(150) << " ... deleting the factory " << m_factory << endl; |
135 | delete m_factory; | 135 | delete m_factory; |
136 | } | 136 | } |
137 | } | 137 | } |
138 | 138 | ||
139 | QString KLibrary::name() const | 139 | QString KLibrary::name() const |
140 | { | 140 | { |
141 | return m_libname; | 141 | return m_libname; |
142 | } | 142 | } |
143 | 143 | ||
144 | QString KLibrary::fileName() const | 144 | QString KLibrary::fileName() const |
145 | { | 145 | { |
146 | return m_filename; | 146 | return m_filename; |
147 | } | 147 | } |
148 | 148 | ||
149 | KLibFactory* KLibrary::factory() | 149 | KLibFactory* KLibrary::factory() |
150 | { | 150 | { |
151 | if ( m_factory ) | 151 | if ( m_factory ) |
152 | return m_factory; | 152 | return m_factory; |
153 | 153 | ||
154 | QCString symname; | 154 | QCString symname; |
155 | symname.sprintf("init_%s", name().latin1() ); | 155 | symname.sprintf("init_%s", name().latin1() ); |
156 | 156 | ||
157 | void* sym = symbol( symname ); | 157 | void* sym = symbol( symname ); |
158 | if ( !sym ) | 158 | if ( !sym ) |
159 | { | 159 | { |
160 | kdWarning(150) << "KLibrary: The library " << name() << " does not offer an init_" << name() << " function" << endl; | 160 | qDebug("KLibrary: The library %s does not offer an %s function", name().latin1(), symname.data()); |
161 | kdWarning(150) << "KLibrary: The library " << name().latin1() << " does not offer an init_" << name().latin1() << " function" << endl; | ||
161 | return 0; | 162 | return 0; |
162 | } | 163 | } |
163 | 164 | ||
164 | typedef KLibFactory* (*t_func)(); | 165 | typedef KLibFactory* (*t_func)(); |
165 | t_func func = (t_func)sym; | 166 | t_func func = (t_func)sym; |
166 | m_factory = func(); | 167 | m_factory = func(); |
167 | 168 | ||
168 | if( !m_factory ) | 169 | if( !m_factory ) |
169 | { | 170 | { |
170 | kdWarning(150) << "KLibrary: The library " << name() << " does not offer a KDE compatible factory" << endl; | 171 | kdWarning(150) << "KLibrary: The library " << name() << " does not offer a KDE compatible factory" << endl; |
171 | return 0; | 172 | return 0; |
172 | } | 173 | } |
173 | 174 | ||
174 | connect( m_factory, SIGNAL( objectCreated( QObject * ) ), | 175 | connect( m_factory, SIGNAL( objectCreated( QObject * ) ), |
175 | this, SLOT( slotObjectCreated( QObject * ) ) ); | 176 | this, SLOT( slotObjectCreated( QObject * ) ) ); |
176 | 177 | ||
177 | return m_factory; | 178 | return m_factory; |
178 | } | 179 | } |
179 | 180 | ||
180 | void* KLibrary::symbol( const char* symname ) const | 181 | void* KLibrary::symbol( const char* symname ) const |
181 | { | 182 | { |
182 | //US void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname ); | 183 | //US void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname ); |
183 | void* sym = m_handle->resolve( symname ); | 184 | void* sym = m_handle->resolve( symname ); |
184 | if ( !sym ) | 185 | if ( !sym ) |
185 | { | 186 | { |
186 | //US kdWarning(150) << "KLibrary: " << lt_dlerror() << endl; | 187 | //US kdWarning(150) << "KLibrary: " << lt_dlerror() << endl; |
187 | kdWarning(150) << "KLibrary: " << m_libname << ", symbol:" << symname << " not found " << endl; | 188 | kdWarning(150) << "KLibrary: " << m_libname << ", symbol:" << symname << " not found " << endl; |
188 | return 0; | 189 | return 0; |
189 | } | 190 | } |
190 | 191 | ||
191 | return sym; | 192 | return sym; |
192 | } | 193 | } |
193 | 194 | ||
194 | bool KLibrary::hasSymbol( const char* symname ) const | 195 | bool KLibrary::hasSymbol( const char* symname ) const |
195 | { | 196 | { |
196 | //US void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname ); | 197 | //US void* sym = lt_dlsym( (lt_dlhandle) m_handle, symname ); |
197 | void* sym = m_handle->resolve( symname ); | 198 | void* sym = m_handle->resolve( symname ); |
198 | return (sym != 0L ); | 199 | return (sym != 0L ); |
199 | } | 200 | } |
200 | 201 | ||
201 | void KLibrary::unload() const | 202 | void KLibrary::unload() const |
202 | { | 203 | { |
203 | if (KLibLoader::s_self) | 204 | if (KLibLoader::s_self) |
204 | KLibLoader::s_self->unloadLibrary(QFile::encodeName(name())); | 205 | KLibLoader::s_self->unloadLibrary(QFile::encodeName(name())); |
205 | } | 206 | } |
206 | 207 | ||
207 | void KLibrary::slotObjectCreated( QObject *obj ) | 208 | void KLibrary::slotObjectCreated( QObject *obj ) |
208 | { | 209 | { |
209 | if ( !obj ) | 210 | if ( !obj ) |
210 | return; | 211 | return; |
211 | 212 | ||
212 | if ( m_timer && m_timer->isActive() ) | 213 | if ( m_timer && m_timer->isActive() ) |
213 | m_timer->stop(); | 214 | m_timer->stop(); |
214 | 215 | ||
215 | if ( m_objs.containsRef( obj ) ) | 216 | if ( m_objs.containsRef( obj ) ) |
216 | return; // we know this object already | 217 | return; // we know this object already |
217 | 218 | ||
218 | connect( obj, SIGNAL( destroyed() ), | 219 | connect( obj, SIGNAL( destroyed() ), |
219 | this, SLOT( slotObjectDestroyed() ) ); | 220 | this, SLOT( slotObjectDestroyed() ) ); |
220 | 221 | ||
221 | m_objs.append( obj ); | 222 | m_objs.append( obj ); |
222 | } | 223 | } |
223 | 224 | ||
224 | void KLibrary::slotObjectDestroyed() | 225 | void KLibrary::slotObjectDestroyed() |
225 | { | 226 | { |
226 | m_objs.removeRef( sender() ); | 227 | m_objs.removeRef( sender() ); |
227 | 228 | ||
228 | if ( m_objs.count() == 0 ) | 229 | if ( m_objs.count() == 0 ) |
229 | { | 230 | { |
230 | // kdDebug(150) << "KLibrary: shutdown timer for " << name() << " started!" | 231 | // kdDebug(150) << "KLibrary: shutdown timer for " << name() << " started!" |
231 | // << endl; | 232 | // << endl; |
232 | 233 | ||
233 | if ( !m_timer ) | 234 | if ( !m_timer ) |
234 | { | 235 | { |
235 | m_timer = new QTimer( this, "klibrary_shutdown_timer" ); | 236 | m_timer = new QTimer( this, "klibrary_shutdown_timer" ); |
236 | connect( m_timer, SIGNAL( timeout() ), | 237 | connect( m_timer, SIGNAL( timeout() ), |
237 | this, SLOT( slotTimeout() ) ); | 238 | this, SLOT( slotTimeout() ) ); |
238 | } | 239 | } |
239 | 240 | ||
240 | // as long as it's not stable make the timeout short, for debugging | 241 | // as long as it's not stable make the timeout short, for debugging |
241 | // pleasure (matz) | 242 | // pleasure (matz) |
242 | //m_timer->start( 1000*60, true ); | 243 | //m_timer->start( 1000*60, true ); |
243 | m_timer->start( 1000*10, true ); | 244 | m_timer->start( 1000*10, true ); |
244 | } | 245 | } |
245 | } | 246 | } |
246 | 247 | ||
247 | void KLibrary::slotTimeout() | 248 | void KLibrary::slotTimeout() |
248 | { | 249 | { |
249 | if ( m_objs.count() != 0 ) | 250 | if ( m_objs.count() != 0 ) |
250 | return; | 251 | return; |
251 | 252 | ||
252 | /* Don't go through KLibLoader::unloadLibrary(), because that uses the | 253 | /* Don't go through KLibLoader::unloadLibrary(), because that uses the |
253 | ref counter, but this timeout means to unconditionally close this library | 254 | ref counter, but this timeout means to unconditionally close this library |
254 | The destroyed() signal will take care to remove us from all lists. | 255 | The destroyed() signal will take care to remove us from all lists. |
255 | */ | 256 | */ |
256 | delete this; | 257 | delete this; |
257 | } | 258 | } |
258 | 259 | ||
259 | // ------------------------------------------------- | 260 | // ------------------------------------------------- |
260 | 261 | ||
261 | /* This helper class is needed, because KLibraries can go away without | 262 | /* This helper class is needed, because KLibraries can go away without |
262 | being unloaded. So we need some info about KLibraries even after its | 263 | being unloaded. So we need some info about KLibraries even after its |
263 | death. */ | 264 | death. */ |
264 | class KLibWrapPrivate | 265 | class KLibWrapPrivate |
265 | { | 266 | { |
266 | public: | 267 | public: |
267 | //US KLibWrapPrivate(KLibrary *l, lt_dlhandle h); | 268 | //US KLibWrapPrivate(KLibrary *l, lt_dlhandle h); |
268 | KLibWrapPrivate(KLibrary *l, QLibrary* h); | 269 | KLibWrapPrivate(KLibrary *l, QLibrary* h); |
269 | 270 | ||
270 | KLibrary *lib; | 271 | KLibrary *lib; |
271 | enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode; | 272 | enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode; |
272 | int ref_count; | 273 | int ref_count; |
273 | //US lt_dlhandle handle; | 274 | //US lt_dlhandle handle; |
274 | QLibrary *handle; | 275 | QLibrary *handle; |
275 | QString name; | 276 | QString name; |
276 | QString filename; | 277 | QString filename; |
277 | }; | 278 | }; |
278 | 279 | ||
279 | //US KLibWrapPrivate::KLibWrapPrivate(KLibrary *l, lt_dlhandle h) | 280 | //US KLibWrapPrivate::KLibWrapPrivate(KLibrary *l, lt_dlhandle h) |
280 | KLibWrapPrivate::KLibWrapPrivate(KLibrary *l, QLibrary* h) | 281 | KLibWrapPrivate::KLibWrapPrivate(KLibrary *l, QLibrary* h) |
281 | : lib(l), ref_count(1), handle(h), name(l->name()), filename(l->fileName()) | 282 | : lib(l), ref_count(1), handle(h), name(l->name()), filename(l->fileName()) |
282 | { | 283 | { |
283 | unload_mode = UNKNOWN; | 284 | unload_mode = UNKNOWN; |
284 | /*US | 285 | /*US |
285 | if (lt_dlsym(handle, "__kde_do_not_unload") != 0) { | 286 | if (lt_dlsym(handle, "__kde_do_not_unload") != 0) { |
286 | // kdDebug(150) << "Will not unload " << name << endl; | 287 | // kdDebug(150) << "Will not unload " << name << endl; |
287 | unload_mode = DONT_UNLOAD; | 288 | unload_mode = DONT_UNLOAD; |
288 | } else if (lt_dlsym(handle, "__kde_do_unload") != 0) { | 289 | } else if (lt_dlsym(handle, "__kde_do_unload") != 0) { |
289 | unload_mode = UNLOAD; | 290 | unload_mode = UNLOAD; |
290 | } | 291 | } |
291 | */ | 292 | */ |
292 | //US use instead: | 293 | //US use instead: |
293 | if (h->resolve("__kde_do_not_unload") != 0) { | 294 | if (h->resolve("__kde_do_not_unload") != 0) { |
294 | // kdDebug(150) << "Will not unload " << name << endl; | 295 | // kdDebug(150) << "Will not unload " << name << endl; |
295 | unload_mode = DONT_UNLOAD; | 296 | unload_mode = DONT_UNLOAD; |
296 | } else if (h->resolve("__kde_do_unload") != 0) { | 297 | } else if (h->resolve("__kde_do_unload") != 0) { |
297 | unload_mode = UNLOAD; | 298 | unload_mode = UNLOAD; |
298 | } | 299 | } |
299 | } | 300 | } |
300 | 301 | ||
301 | class KLibLoaderPrivate | 302 | class KLibLoaderPrivate |
302 | { | 303 | { |
303 | public: | 304 | public: |
304 | QPtrList<KLibWrapPrivate> loaded_stack; | 305 | QPtrList<KLibWrapPrivate> loaded_stack; |
305 | QPtrList<KLibWrapPrivate> pending_close; | 306 | QPtrList<KLibWrapPrivate> pending_close; |
306 | enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode; | 307 | enum {UNKNOWN, UNLOAD, DONT_UNLOAD} unload_mode; |
307 | 308 | ||
308 | QString errorMessage; | 309 | QString errorMessage; |
309 | }; | 310 | }; |
310 | 311 | ||
311 | KLibLoader* KLibLoader::s_self = 0; | 312 | KLibLoader* KLibLoader::s_self = 0; |
312 | 313 | ||
313 | KLibLoader* KLibLoader::self() | 314 | KLibLoader* KLibLoader::self() |
314 | { | 315 | { |
315 | if ( !s_self ) | 316 | if ( !s_self ) |
316 | s_self = new KLibLoader; | 317 | s_self = new KLibLoader; |
317 | return s_self; | 318 | return s_self; |
318 | } | 319 | } |
319 | 320 | ||
320 | void KLibLoader::cleanUp() | 321 | void KLibLoader::cleanUp() |
321 | { | 322 | { |
322 | if ( !s_self ) | 323 | if ( !s_self ) |
323 | return; | 324 | return; |
324 | 325 | ||
325 | delete s_self; | 326 | delete s_self; |
326 | s_self = 0; | 327 | s_self = 0; |
327 | } | 328 | } |
328 | 329 | ||
329 | KLibLoader::KLibLoader( QObject* parent, const char* name ) | 330 | KLibLoader::KLibLoader( QObject* parent, const char* name ) |
330 | : QObject( parent, name ) | 331 | : QObject( parent, name ) |
331 | { | 332 | { |
332 | s_self = this; | 333 | s_self = this; |
333 | d = new KLibLoaderPrivate; | 334 | d = new KLibLoaderPrivate; |
334 | //US lt_dlinit(); | 335 | //US lt_dlinit(); |
335 | d->unload_mode = KLibLoaderPrivate::UNKNOWN; | 336 | d->unload_mode = KLibLoaderPrivate::UNKNOWN; |
336 | if (getenv("KDE_NOUNLOAD") != 0) | 337 | if (getenv("KDE_NOUNLOAD") != 0) |
337 | d->unload_mode = KLibLoaderPrivate::DONT_UNLOAD; | 338 | d->unload_mode = KLibLoaderPrivate::DONT_UNLOAD; |
338 | else if (getenv("KDE_DOUNLOAD") != 0) | 339 | else if (getenv("KDE_DOUNLOAD") != 0) |
339 | d->unload_mode = KLibLoaderPrivate::UNLOAD; | 340 | d->unload_mode = KLibLoaderPrivate::UNLOAD; |
340 | d->loaded_stack.setAutoDelete( true ); | 341 | d->loaded_stack.setAutoDelete( true ); |
341 | } | 342 | } |
342 | 343 | ||
343 | KLibLoader::~KLibLoader() | 344 | KLibLoader::~KLibLoader() |
344 | { | 345 | { |
345 | // kdDebug(150) << "Deleting KLibLoader " << this << " " << name() << endl; | 346 | // kdDebug(150) << "Deleting KLibLoader " << this << " " << name() << endl; |
346 | 347 | ||
347 | QAsciiDictIterator<KLibWrapPrivate> it( m_libs ); | 348 | QAsciiDictIterator<KLibWrapPrivate> it( m_libs ); |
348 | for (; it.current(); ++it ) | 349 | for (; it.current(); ++it ) |
349 | { | 350 | { |
350 | kdDebug(150) << "The KLibLoader contains the library " << it.current()->name | 351 | kdDebug(150) << "The KLibLoader contains the library " << it.current()->name |
351 | << " (" << it.current()->lib << ")" << endl; | 352 | << " (" << it.current()->lib << ")" << endl; |
352 | d->pending_close.append(it.current()); | 353 | d->pending_close.append(it.current()); |
353 | } | 354 | } |
354 | 355 | ||
355 | close_pending(0); | 356 | close_pending(0); |
356 | 357 | ||
357 | delete d; | 358 | delete d; |
358 | } | 359 | } |
359 | 360 | ||
360 | //static | 361 | //static |
361 | QString KLibLoader::findLibrary( const char * name/*US , const KInstance * instance*/ ) | 362 | QString KLibLoader::findLibrary( const char * name/*US , const KInstance * instance*/ ) |
362 | { | 363 | { |
363 | QCString libname( name ); | 364 | QCString libname( name ); |
364 | 365 | ||
365 | // only append ".la" if there is no extension | 366 | // only append ".la" if there is no extension |
366 | // this allows to load non-libtool libraries as well | 367 | // this allows to load non-libtool libraries as well |
367 | // (mhk, 20000228) | 368 | // (mhk, 20000228) |
368 | int pos = libname.findRev('/'); | 369 | int pos = libname.findRev('/'); |
369 | if (pos < 0) | 370 | if (pos < 0) |
370 | pos = 0; | 371 | pos = 0; |
371 | if (libname.find('.', pos) < 0) | 372 | /*US |
373 | if (libname.find('.', pos) < 0) { | ||
372 | libname += ".la"; | 374 | libname += ".la"; |
375 | } | ||
376 | */ | ||
377 | //US in the microedition we work only with shared libraries. | ||
378 | if (libname.find('.', pos) < 0) { | ||
379 | libname += ".so"; | ||
380 | } | ||
373 | 381 | ||
374 | // only look up the file if it is not an absolute filename | 382 | // only look up the file if it is not an absolute filename |
375 | // (mhk, 20000228) | 383 | // (mhk, 20000228) |
376 | QString libfile; | 384 | QString libfile; |
377 | if (libname[0] == '/') | 385 | if (libname[0] == '/') |
378 | libfile = libname; | 386 | libfile = libname; |
379 | else | 387 | else |
380 | { | 388 | { |
389 | //US at this point the libname must exist as real filesname. No expansions will be made later | ||
390 | // in findResources. Because of that we prepend the lib prefix here to the name | ||
391 | //US I add also the "lib" prefix. I do not how could this could have worked before without it? | ||
392 | libname.insert(pos, "lib"); | ||
393 | |||
394 | |||
381 | //US libfile = instance->dirs()->findResource( "module", libname ); | 395 | //US libfile = instance->dirs()->findResource( "module", libname ); |
382 | libfile = KGlobal::dirs()->findResource( "module", libname ); | 396 | libfile = KGlobal::dirs()->findResource( "module", libname ); |
383 | if ( libfile.isEmpty() ) | 397 | if ( libfile.isEmpty() ) |
384 | { | 398 | { |
385 | //US libfile = instance->dirs()->findResource( "lib", libname ); | 399 | //US libfile = instance->dirs()->findResource( "lib", libname ); |
386 | libfile = KGlobal::dirs()->findResource( "lib", libname ); | 400 | libfile = KGlobal::dirs()->findResource( "lib", libname ); |
387 | #ifndef NDEBUG | 401 | #ifndef NDEBUG |
388 | if ( !libfile.isEmpty() && libname.left(3) == "lib" ) // don't warn for kdeinit modules | 402 | if ( !libfile.isEmpty() && libname.left(3) == "lib" ) // don't warn for kdeinit modules |
389 | kdDebug(150) << "library " << libname << " not found under 'module' but under 'lib'" << endl; | 403 | kdDebug(150) << "library " << libname << " not found under 'module' but under 'lib'" << endl; |
390 | #endif | 404 | #endif |
391 | } | 405 | } |
392 | if ( libfile.isEmpty() ) | 406 | if ( libfile.isEmpty() ) |
393 | { | 407 | { |
394 | #ifndef NDEBUG | 408 | #ifndef NDEBUG |
395 | kdDebug(150) << "library=" << libname << ": No file names " << libname.data() << " found in paths." << endl; | 409 | kdDebug(150) << "library=" << libname << ": No file names " << libname.data() << " found in paths." << endl; |
396 | #endif | 410 | #endif |
397 | self()->d->errorMessage = i18n("Library files for \"%1\" not found in paths").arg(libname); | 411 | self()->d->errorMessage = i18n("Library files for \"%1\" not found in paths").arg(libname); |
412 | |||
413 | qDebug("KLibLoader::library could not find library: %s", libname.data()); | ||
414 | |||
398 | } | 415 | } |
399 | else | 416 | else |
400 | self()->d->errorMessage = QString::null; | 417 | self()->d->errorMessage = QString::null; |
401 | } | 418 | } |
402 | return libfile; | 419 | return libfile; |
403 | } | 420 | } |
404 | 421 | ||
405 | 422 | ||
406 | KLibrary* KLibLoader::globalLibrary( const char *name ) | 423 | KLibrary* KLibLoader::globalLibrary( const char *name ) |
407 | { | 424 | { |
408 | KLibrary *tmp; | 425 | KLibrary *tmp; |
409 | /*US | 426 | /*US |
410 | int olt_dlopen_flag = lt_dlopen_flag; | 427 | int olt_dlopen_flag = lt_dlopen_flag; |
411 | 428 | ||
412 | lt_dlopen_flag |= LT_GLOBAL; | 429 | lt_dlopen_flag |= LT_GLOBAL; |
413 | kdDebug(150) << "Loading the next library global with flag " | 430 | kdDebug(150) << "Loading the next library global with flag " |
414 | << lt_dlopen_flag | 431 | << lt_dlopen_flag |
415 | << "." << endl; | 432 | << "." << endl; |
416 | */ | 433 | */ |
417 | tmp = library(name); | 434 | tmp = library(name); |
418 | /*US | 435 | /*US |
419 | lt_dlopen_flag = olt_dlopen_flag; | 436 | lt_dlopen_flag = olt_dlopen_flag; |
420 | */ | 437 | */ |
421 | return tmp; | 438 | return tmp; |
422 | } | 439 | } |
423 | 440 | ||
424 | 441 | ||
425 | KLibrary* KLibLoader::library( const char *name ) | 442 | KLibrary* KLibLoader::library( const char *name ) |
426 | { | 443 | { |
427 | if (!name) | 444 | if (!name) |
428 | return 0; | 445 | return 0; |
429 | 446 | ||
430 | KLibWrapPrivate* wrap = m_libs[name]; | 447 | KLibWrapPrivate* wrap = m_libs[name]; |
431 | if (wrap) { | 448 | if (wrap) { |
432 | /* Nothing to do to load the library. */ | 449 | /* Nothing to do to load the library. */ |
433 | wrap->ref_count++; | 450 | wrap->ref_count++; |
434 | return wrap->lib; | 451 | return wrap->lib; |
435 | } | 452 | } |
436 | 453 | ||
437 | /* Test if this library was loaded at some time, but got | 454 | /* Test if this library was loaded at some time, but got |
438 | unloaded meanwhile, whithout being dlclose()'ed. */ | 455 | unloaded meanwhile, whithout being dlclose()'ed. */ |
439 | QPtrListIterator<KLibWrapPrivate> it(d->loaded_stack); | 456 | QPtrListIterator<KLibWrapPrivate> it(d->loaded_stack); |
440 | for (; it.current(); ++it) { | 457 | for (; it.current(); ++it) { |
441 | if (it.current()->name == name) | 458 | if (it.current()->name == name) |
442 | wrap = it.current(); | 459 | wrap = it.current(); |
443 | } | 460 | } |
444 | 461 | ||
445 | if (wrap) { | 462 | if (wrap) { |
446 | d->pending_close.removeRef(wrap); | 463 | d->pending_close.removeRef(wrap); |
447 | if (!wrap->lib) { | 464 | if (!wrap->lib) { |
448 | /* This lib only was in loaded_stack, but not in m_libs. */ | 465 | /* This lib only was in loaded_stack, but not in m_libs. */ |
449 | wrap->lib = new KLibrary( name, wrap->filename, wrap->handle ); | 466 | wrap->lib = new KLibrary( name, wrap->filename, wrap->handle ); |
450 | } | 467 | } |
451 | wrap->ref_count++; | 468 | wrap->ref_count++; |
452 | } else { | 469 | } else { |
453 | QString libfile = findLibrary( name ); | 470 | QString libfile = findLibrary( name ); |
454 | if ( libfile.isEmpty() ) | 471 | if ( libfile.isEmpty() ) |
455 | return 0; | 472 | return 0; |
456 | 473 | ||
457 | const QString & qpeDir = QPEApplication::qpeDir(); | ||
458 | libfile = qpeDir + libfile; | ||
459 | //US QLibrary *lib = new QLibrary( qpeDir + "/plugins/korganizer/libopiekabc.so", QLibrary::Immediately ); | ||
460 | QLibrary *qlib = new QLibrary( libfile.latin1(), QLibrary::Immediately ); | 474 | QLibrary *qlib = new QLibrary( libfile.latin1(), QLibrary::Immediately ); |
461 | 475 | ||
462 | //US lt_dlhandle handle = lt_dlopen( libfile.latin1() ); | 476 | //US lt_dlhandle handle = lt_dlopen( libfile.latin1() ); |
463 | //US if ( !handle ) | 477 | //US if ( !handle ) |
464 | if ( !qlib ) | 478 | if ( !qlib ) |
465 | { | 479 | { |
466 | //US const char* errmsg = lt_dlerror(); | 480 | //US const char* errmsg = lt_dlerror(); |
467 | char* errmsg; | 481 | char* errmsg; |
468 | sprintf(errmsg, "KLibLoader::library could not load library: %s", libfile.latin1()); | 482 | sprintf(errmsg, "KLibLoader::library could not load library: %s", libfile.latin1()); |
469 | qDebug(errmsg); | 483 | qDebug(errmsg); |
470 | 484 | ||
471 | if(errmsg) | 485 | if(errmsg) |
472 | d->errorMessage = QString::fromLatin1(errmsg); | 486 | d->errorMessage = QString::fromLatin1(errmsg); |
473 | else | 487 | else |
474 | d->errorMessage = QString::null; | 488 | d->errorMessage = QString::null; |
475 | kdWarning(150) << "library=" << name << ": file=" << libfile << ": " << d->errorMessage << endl; | 489 | kdWarning(150) << "library=" << name << ": file=" << libfile << ": " << d->errorMessage << endl; |
476 | return 0; | 490 | return 0; |
477 | } | 491 | } |
478 | else | 492 | else |
479 | d->errorMessage = QString::null; | 493 | d->errorMessage = QString::null; |
480 | 494 | ||
481 | KLibrary *lib = new KLibrary( name, libfile, qlib ); | 495 | KLibrary *lib = new KLibrary( name, libfile, qlib ); |
482 | wrap = new KLibWrapPrivate(lib, qlib); | 496 | wrap = new KLibWrapPrivate(lib, qlib); |
483 | d->loaded_stack.prepend(wrap); | 497 | d->loaded_stack.prepend(wrap); |
484 | } | 498 | } |
485 | m_libs.insert( name, wrap ); | 499 | m_libs.insert( name, wrap ); |
486 | 500 | ||
487 | connect( wrap->lib, SIGNAL( destroyed() ), | 501 | connect( wrap->lib, SIGNAL( destroyed() ), |
488 | this, SLOT( slotLibraryDestroyed() ) ); | 502 | this, SLOT( slotLibraryDestroyed() ) ); |
489 | 503 | ||
490 | return wrap->lib; | 504 | return wrap->lib; |
491 | } | 505 | } |
492 | 506 | ||
493 | QString KLibLoader::lastErrorMessage() const | 507 | QString KLibLoader::lastErrorMessage() const |
494 | { | 508 | { |
495 | return d->errorMessage; | 509 | return d->errorMessage; |
496 | } | 510 | } |
497 | 511 | ||
498 | void KLibLoader::unloadLibrary( const char *libname ) | 512 | void KLibLoader::unloadLibrary( const char *libname ) |
499 | { | 513 | { |
500 | KLibWrapPrivate *wrap = m_libs[ libname ]; | 514 | KLibWrapPrivate *wrap = m_libs[ libname ]; |
501 | if (!wrap) | 515 | if (!wrap) |
502 | return; | 516 | return; |
503 | if (--wrap->ref_count) | 517 | if (--wrap->ref_count) |
504 | return; | 518 | return; |
505 | 519 | ||
506 | // kdDebug(150) << "closing library " << libname << endl; | 520 | // kdDebug(150) << "closing library " << libname << endl; |
507 | 521 | ||
508 | m_libs.remove( libname ); | 522 | m_libs.remove( libname ); |
509 | 523 | ||
510 | disconnect( wrap->lib, SIGNAL( destroyed() ), | 524 | disconnect( wrap->lib, SIGNAL( destroyed() ), |
511 | this, SLOT( slotLibraryDestroyed() ) ); | 525 | this, SLOT( slotLibraryDestroyed() ) ); |
512 | close_pending( wrap ); | 526 | close_pending( wrap ); |
513 | } | 527 | } |
514 | 528 | ||
515 | KLibFactory* KLibLoader::factory( const char* name ) | 529 | KLibFactory* KLibLoader::factory( const char* name ) |
516 | { | 530 | { |
517 | KLibrary* lib = library( name ); | 531 | KLibrary* lib = library( name ); |
518 | if ( !lib ) | 532 | if ( !lib ) |
519 | return 0; | 533 | return 0; |
520 | 534 | ||
521 | return lib->factory(); | 535 | return lib->factory(); |
522 | } | 536 | } |
523 | 537 | ||
524 | void KLibLoader::slotLibraryDestroyed() | 538 | void KLibLoader::slotLibraryDestroyed() |
525 | { | 539 | { |
526 | const KLibrary *lib = static_cast<const KLibrary *>( sender() ); | 540 | const KLibrary *lib = static_cast<const KLibrary *>( sender() ); |
527 | 541 | ||
528 | QAsciiDictIterator<KLibWrapPrivate> it( m_libs ); | 542 | QAsciiDictIterator<KLibWrapPrivate> it( m_libs ); |
529 | for (; it.current(); ++it ) | 543 | for (; it.current(); ++it ) |
530 | if ( it.current()->lib == lib ) | 544 | if ( it.current()->lib == lib ) |
531 | { | 545 | { |
532 | KLibWrapPrivate *wrap = it.current(); | 546 | KLibWrapPrivate *wrap = it.current(); |
533 | wrap->lib = 0; /* the KLibrary object is already away */ | 547 | wrap->lib = 0; /* the KLibrary object is already away */ |
534 | m_libs.remove( it.currentKey() ); | 548 | m_libs.remove( it.currentKey() ); |
535 | close_pending( wrap ); | 549 | close_pending( wrap ); |
536 | return; | 550 | return; |
537 | } | 551 | } |
538 | } | 552 | } |
539 | 553 | ||
540 | void KLibLoader::close_pending(KLibWrapPrivate *wrap) | 554 | void KLibLoader::close_pending(KLibWrapPrivate *wrap) |
541 | { | 555 | { |
542 | if (wrap && !d->pending_close.containsRef( wrap )) | 556 | if (wrap && !d->pending_close.containsRef( wrap )) |
543 | d->pending_close.append( wrap ); | 557 | d->pending_close.append( wrap ); |
544 | 558 | ||
545 | /* First delete all KLibrary objects in pending_close, but _don't_ unload | 559 | /* First delete all KLibrary objects in pending_close, but _don't_ unload |
546 | the DSO behind it. */ | 560 | the DSO behind it. */ |
547 | QPtrListIterator<KLibWrapPrivate> it(d->pending_close); | 561 | QPtrListIterator<KLibWrapPrivate> it(d->pending_close); |
548 | for (; it.current(); ++it) { | 562 | for (; it.current(); ++it) { |
549 | wrap = it.current(); | 563 | wrap = it.current(); |
550 | if (wrap->lib) { | 564 | if (wrap->lib) { |
551 | disconnect( wrap->lib, SIGNAL( destroyed() ), | 565 | disconnect( wrap->lib, SIGNAL( destroyed() ), |
552 | this, SLOT( slotLibraryDestroyed() ) ); | 566 | this, SLOT( slotLibraryDestroyed() ) ); |
553 | delete wrap->lib; | 567 | delete wrap->lib; |
554 | wrap->lib = 0; | 568 | wrap->lib = 0; |
555 | } | 569 | } |
556 | } | 570 | } |
557 | 571 | ||
558 | if (d->unload_mode == KLibLoaderPrivate::DONT_UNLOAD) return; | 572 | if (d->unload_mode == KLibLoaderPrivate::DONT_UNLOAD) return; |
559 | 573 | ||
560 | bool deleted_one = false; | 574 | bool deleted_one = false; |
561 | while ((wrap = d->loaded_stack.first())) { | 575 | while ((wrap = d->loaded_stack.first())) { |
562 | /* Let's first see, if we want to try to unload this lib. | 576 | /* Let's first see, if we want to try to unload this lib. |
563 | If the env. var KDE_DOUNLOAD is set, we try to unload every lib. | 577 | If the env. var KDE_DOUNLOAD is set, we try to unload every lib. |
564 | If not, we look at the lib itself, and unload it only, if it exports | 578 | If not, we look at the lib itself, and unload it only, if it exports |
565 | the symbol __kde_do_unload. */ | 579 | the symbol __kde_do_unload. */ |
566 | if (d->unload_mode != KLibLoaderPrivate::UNLOAD | 580 | if (d->unload_mode != KLibLoaderPrivate::UNLOAD |
567 | && wrap->unload_mode != KLibWrapPrivate::UNLOAD) | 581 | && wrap->unload_mode != KLibWrapPrivate::UNLOAD) |
568 | break; | 582 | break; |
569 | 583 | ||
570 | /* Now ensure, that the libs are only unloaded in the reverse direction | 584 | /* Now ensure, that the libs are only unloaded in the reverse direction |
571 | they were loaded. */ | 585 | they were loaded. */ |
572 | if (!d->pending_close.containsRef( wrap )) { | 586 | if (!d->pending_close.containsRef( wrap )) { |
573 | if (!deleted_one) | 587 | if (!deleted_one) |
574 | /* Only diagnose, if we really haven't deleted anything. */ | 588 | /* Only diagnose, if we really haven't deleted anything. */ |
575 | // kdDebug(150) << "try to dlclose " << wrap->name << ": not yet" << endl; | 589 | // kdDebug(150) << "try to dlclose " << wrap->name << ": not yet" << endl; |
576 | break; | 590 | break; |
577 | } | 591 | } |
578 | 592 | ||
579 | // kdDebug(150) << "try to dlclose " << wrap->name << ": yes, done." << endl; | 593 | // kdDebug(150) << "try to dlclose " << wrap->name << ": yes, done." << endl; |
580 | 594 | ||
581 | #ifndef Q_WS_QWS | 595 | #ifndef Q_WS_QWS |
582 | if ( !deleted_one ) { | 596 | if ( !deleted_one ) { |
583 | /* Only do the hack once in this loop. | 597 | /* Only do the hack once in this loop. |
584 | WABA: *HACK* | 598 | WABA: *HACK* |
585 | We need to make sure to clear the clipboard before unloading a DSO | 599 | We need to make sure to clear the clipboard before unloading a DSO |
586 | because the DSO could have defined an object derived from QMimeSource | 600 | because the DSO could have defined an object derived from QMimeSource |
587 | and placed that on the clipboard. */ | 601 | and placed that on the clipboard. */ |
588 | /*kapp->clipboard()->clear();*/ | 602 | /*kapp->clipboard()->clear();*/ |
589 | 603 | ||
590 | /* Well.. let's do something more subtle... convert the clipboard context | 604 | /* Well.. let's do something more subtle... convert the clipboard context |
591 | to text. That should be safe as it only uses objects defined by Qt. */ | 605 | to text. That should be safe as it only uses objects defined by Qt. */ |
592 | 606 | ||
593 | QWidgetList *widgetlist = QApplication::topLevelWidgets(); | 607 | QWidgetList *widgetlist = QApplication::topLevelWidgets(); |
594 | QWidget *co = widgetlist->first(); | 608 | QWidget *co = widgetlist->first(); |
595 | while (co) { | 609 | while (co) { |
596 | if (qstrcmp(co->name(), "internal clipboard owner") == 0) { | 610 | if (qstrcmp(co->name(), "internal clipboard owner") == 0) { |
597 | if (XGetSelectionOwner(co->x11Display(), XA_PRIMARY) == co->winId()) | 611 | if (XGetSelectionOwner(co->x11Display(), XA_PRIMARY) == co->winId()) |
598 | kapp->clipboard()->setText(kapp->clipboard()->text()); | 612 | kapp->clipboard()->setText(kapp->clipboard()->text()); |
599 | 613 | ||
600 | break; | 614 | break; |
601 | } | 615 | } |
602 | co = widgetlist->next(); | 616 | co = widgetlist->next(); |
603 | } | 617 | } |
604 | delete widgetlist; | 618 | delete widgetlist; |
605 | } | 619 | } |
606 | #else | 620 | #else |
607 | // FIXME(E): Implement in Qt Embedded | 621 | // FIXME(E): Implement in Qt Embedded |
608 | #endif | 622 | #endif |
609 | 623 | ||
610 | deleted_one = true; | 624 | deleted_one = true; |
611 | //US lt_dlclose(wrap->handle); | 625 | //US lt_dlclose(wrap->handle); |
612 | wrap->handle->unload(); | 626 | wrap->handle->unload(); |
613 | 627 | ||
614 | d->pending_close.removeRef(wrap); | 628 | d->pending_close.removeRef(wrap); |
615 | /* loaded_stack is AutoDelete, so wrap is freed */ | 629 | /* loaded_stack is AutoDelete, so wrap is freed */ |
616 | d->loaded_stack.remove(); | 630 | d->loaded_stack.remove(); |
617 | } | 631 | } |
618 | } | 632 | } |
619 | 633 | ||
620 | void KLibLoader::virtual_hook( int, void* ) | 634 | void KLibLoader::virtual_hook( int, void* ) |
621 | { /*BASE::virtual_hook( id, data );*/ } | 635 | { /*BASE::virtual_hook( id, data );*/ } |
622 | 636 | ||
623 | void KLibFactory::virtual_hook( int, void* ) | 637 | void KLibFactory::virtual_hook( int, void* ) |
624 | { /*BASE::virtual_hook( id, data );*/ } | 638 | { /*BASE::virtual_hook( id, data );*/ } |
625 | 639 | ||
626 | //US #include "klibloader.moc" | 640 | //US #include "klibloader.moc" |