Jean Brouwers added the comment: Attached is an updated dlibtest.c file. It prints a message in the con- /destructor functions and if that fails it calls _exit(9).
Compile and run it as before and check the exit status. If the latter is 9 or 011, a printf error occurred indicating e.g. that stdout was closed or something similar. This version can also be used with gdb, either by pre-loading the dlibtest.so library within gdb or before invoking gdb. To preload the library within gdb (on Linux) use gdb .../python (gdb) set environment LD_PRELOAD ./dlibtest.so (gdb) run ..... or to preload before gdb use setenv LD_PRELOAD ./dlibtest.so gdb .../python (gdb) run ..... Lastly, my previous observations about this issue were clearly a "trompe d'oeil", especially my statement that PyImport_Cleanup never returned. The missing print statements *after* the PyImport_Cleanup call are simply due to printf errors, and nothing else ;-) Added file: http://bugs.python.org/file8625/dlibtest.c __________________________________ Tracker <[EMAIL PROTECTED]> <http://bugs.python.org/issue1329> __________________________________
#include "stdio.h" #include "string.h" #include "unistd.h" #ifndef __GNUC__ # error requires the GNU compiler #endif extern const char* __progname; /* call printf() only if inside python binary */ static void _printf (const char* fmt, const char* arg1, const char* arg2) { if (!strcmp("python", __progname)) { if (printf(fmt, arg1, arg2) < 0) { _exit(9); /* stdout closed? */ } } } #if 0 /* pick this ... */ static void __attribute__((constructor)) /* called on main() */ _ctor (void) { _printf("*** %s called in %s ...\n", "ctor", __progname); } static void __attribute__((destructor)) /* called on exit() */ _dtor (void) { _printf("*** %s called in %s ...\n", "dtor", __progname); } #else /* ... or this case */ static void /* called on Python exit */ _dtor (void) { _printf("*** %s called in %s ...\n", "dtor", __progname); } /* the weak attribute prevents unresolved symbol errors */ extern int Py_AtExit (void(*func)(void)) __attribute__((weak)); static void __attribute__((constructor)) /* called on main() */ _ctor (void) { _printf("*** %s called in %s ...\n", "ctor", __progname); if (Py_AtExit) { _printf("*** %s %s\n", "Py_AtExit", (Py_AtExit(_dtor) < 0 ? "failed!" : "OK")); } } #endif /* ===================================================== Build this file into a shared library, then pre-load that library with the Python binary as follows. On Linux, compile as gcc -o dlibtest.os -c -m32 -Wall -Werror -fPIC dlibtest.c gcc -o dlibtest.so -m32 -ldl -shared dlibtest.os and then run env LD_PRELOAD=./dlibtest.so .../python On MacOS X, compile as gcc -o dlibtest.os -c -Wall -Werror -march=i686 -fPIC dlibtest.c gcc -o dlibtest.dylib -undefined dynamic_lookup -lpthread -dynamiclib dlibtest.os and run env DYLD_INSERT_LIBRARIES=./dlibtest.dylib .../python.exe To preload the library within gdb (on Linux) use gdb .../python (gdb) set environment LD_PRELOAD ./dlibtest.so (gdb) run ..... or use setenv LD_PRELOAD ./dlibtest.so gdb .../python (gdb) run ..... After Ctrl-D two messages should have been printed, but 3.0a1 prints only the first one. An exist status is 9 or 011 indicates that a printf error occurred. Here is a log from Linux with my 3.0a1 and 2.51 builds and with 2.3.4 included with the Linux distro: $ gcc -o dlibtest.os -c -m32 -Wall -Werror -fPIC dlibtest.c $ gcc -o dlibtest.so -m32 -ldl -shared dlibtest.os $ env LD_PRELOAD=./dlibtest.so ~/Python-3.0a1/python *** ctor called ... Python 3.0a1 (py3k, Oct 26 2007, 09:45:17) [GCC 3.4.6 20060404 (Red Hat 3.4.6-8)] on linux2 Type "help", "copyright", "credits" or "license" for more information. >>> $ env LD_PRELOAD=./dlibtest.so ~/Python-2.5.1/python *** ctor called ... Python 2.5.1 (r251:54863, Oct 22 2007, 16:19:11) [GCC 3.4.6 20060404 (Red Hat 3.4.6-8)] on linux2 Type "help", "copyright", "credits" or "license" for more information. >>> *** dtor called ... $ env LD_PRELOAD=./dlibtest.so /usr/bin/python *** ctor called ... Python 2.3.4 (#1, May 2 2007, 19:26:00) [GCC 3.4.6 20060404 (Red Hat 3.4.6-8)] on linux2 Type "help", "copyright", "credits" or "license" for more information. >>> *** dtor called ... Similarly on MacOS X with Python 2.3.5 distributed by Apple: % env DYLD_INSERT_LIBRARIES=./dlibtest.dylib ~/Python-3.0a1/python.exe *** ctor called ... Python 3.0a1 (py3k, Oct 25 2007, 14:55:57) [GCC 4.0.1 (Apple Computer, Inc. build 5367)] on darwin Type "help", "copyright", "credits" or "license" for more information. >>> ^D % env DYLD_INSERT_LIBRARIES=./dlibtest.dylib ~/Python-2.5.1/python.exe *** ctor called ... Python 2.5.1 (r251:54863, Oct 22 2007, 16:18:08) [GCC 4.0.1 (Apple Computer, Inc. build 5341)] on darwin Type "help", "copyright", "credits" or "license" for more information. >>> ^D *** dtor called ... % env DYLD_INSERT_LIBRARIES=./dlibtest.dylib /usr/bin/python *** ctor called ... Python 2.3.5 (#1, Jan 13 2006, 20:13:11) [GCC 4.0.1 (Apple Computer, Inc. build 5250)] on darwin Type "help", "copyright", "credits" or "license" for more information. >>> ^D *** dtor called ... /Jean Brouwers <[EMAIL PROTECTED]> PS) For more details on the con/destructor attributes, see the GNU gcc documentation at <http://gcc.gnu.org/onlinedocs/gcc-3.2/gcc/Function-Attributes.html> See also Apple's documentation DynamicLibrary.pdf which contains some excellent examples plus special MacOS X features. E.g. c/dtor must be ZPRIVATE and a static ctor can have main-like argc, argv and envp arguments! ===================================================== */
_______________________________________________ Python-bugs-list mailing list Unsubscribe: http://mail.python.org/mailman/options/python-bugs-list/archive%40mail-archive.com