STOP command kills entire python instance
GoogleCodeExporter opened this issue · 10 comments
What steps will reproduce the problem?
1. Run a Fortran program wrapped using f2py in Python; any STOP command in
Fortran will kill the entire Python instance
What version of the product are you using? On what operating system?
Version 2
Please provide any additional information below.
I am trying to wrap a large, existing Fortran program
(http://www.itp.uzh.ch/~teyssier/ramses/RAMSES.html) with multiple files and
modules in f2py and call it from a Python package. The Fortran program makes
liberal use of STOP commands in both upper and lower case and is under active
development, so removing the STOP commands manually will be a pain. I would
like to keep the Python instance active after the Fortran program ends, rather
than having the Fortran program kill the Python script upon exit. Putting the
f2py call in a separate thread doesn't appear to help; a STOP command in a
spawned thread will kill the Python instance that spawned it. If you can
suggest work-arounds that don't involve calling the program from the command
line, let me know!
Original issue reported on code.google.com by samg...@googlemail.com
on 8 Nov 2013 at 3:46
Note that in Fortran STOP is a statement of the language that executes system
level function. As a result, it is impossible for f2py catch the results of
STOP execution.
I would write a script that walks through the source code of the Fortran
program and replace all STOP statements with something more appropriate, e.g. a
special function call that records the STOP statement argument and makes a
longjmp call to the wrapper function, all necessary modifications to the
generated wrapper function, eg defining longjmp points, etc. are supported by
f2py signature file specifications.
In principle, fparser could be used for that but it could be overkill as a
simple python script should be sufficient for replacing STOP statements with
something other.
Original comment by pearu.peterson
on 10 Nov 2013 at 10:49
Ah, I see, that makes sense. Thinking about it, it shouldn't be too hard to add
a simple script before compilation that replaces all the STOP commands. Thanks
for your advice - I'll let you know if I have any other problems with this. My
only other idea was to spawn a separate Python instance with the Fortran code
in it so that if it goes down the main thread stays alive, but I can't think of
a clean way to do this short of calling Python with os.system() or similar.
Original comment by samg...@googlemail.com
on 10 Nov 2013 at 2:30
I wrote some code to skim the Fortran and replace the STOP calls with something
that references longjmp (and isetjmp), and this works, but apparently gfortran
can't find longjmp/isetjmp ("undefined reference to `longjmp_' / collect2:
error: ld returned 1 exit status"). It's possible it's been removed:
http://gcc.gnu.org/ml/fortran/2010-11/msg00009.html - I'll keep poking, but it
seems quite difficult to find meaningful documentation on longjmp in gfortran
in any case.
This stackoverflow seems to propose a working C++ wrapper that calls longjmp in
C++, meaning you don't need to modify the Fortran at all:
http://stackoverflow.com/questions/19596375/intercepting-fortran-stop-from-c
(no idea how stable or portable this fix is, of course). I'd rather not go to
the trouble of wrapping Fortran in C++ and then wrapping that in Python, of
course, but if it works...
Original comment by samg...@googlemail.com
on 20 Nov 2013 at 1:42
1) I don't think that you need to deal with longjmp in Fortran. You can replace
Fortran STOP statement with a call to C function that will deal with the
longjmp stuff.
2) Finding the exit function as described in intercepting-fortran-stop-from-c
is equivalent to replacing STOP statement. You would not need to change Fortran
code at the expense of loosing portability (is portability relevant in your
case?). Otherwise, longjmp stuff has to be implemented anyway.
Original comment by pearu.peterson
on 20 Nov 2013 at 1:52
OK, it works, thanks! It's a bit convoluted but seems to work. For
reference/posterity, my solution was the following (I've changed around some of
the names and simplified my code a bit, but it should work as written):
1) Replace every "STOP" call with "call long_jump" using a Python script (I can
post this up if you like; I jumped through a few hoops to get rid of stuff like
comments/strings including the characters "stop" or functions called
"stop_stuff", etc.
2) Write these files:
---
clean_stop.c:
#include <stdio.h>
#include <setjmp.h>
static jmp_buf buf;
void run_fortran_();
void long_jump_()
{
longjmp(buf,1);
}
int set_jump_()
{
if(!setjmp(buf))
{
run_fortran_();
}
}
---
myfortran.f90:
program myfortran
! This is optional; allows you to run it as a command-line program too
call set_jump
end program myfortran
subroutine run
! This is what you call in Python
call set_jump
end subroutine run
subroutine run_fortran
! FORTAN CODE GOES HERE
end subroutine run_fortran
---
3) Compile both as object files in gfortran and gcc
4) Run "f2py -m myfortran -h myfortran.pyf myfortran.f90" (you can delete the
interface call to run_fortran if you like, might make things cleaner for the
users)
5) Run "f2py -lgfortran" -c myfortran.pyf *.o
6) In Python, call "import myfortran; myfortran.run()"
(Note: the output of this program is as files, I assume returning variables
should be a simple thing to hack into the solution above)
I suspect it might have been simpler to embed the C code in Python as the entry
point, but never mind.
Original comment by samg...@googlemail.com
on 20 Nov 2013 at 3:34
Thanks for your feedback!
After a thought, I found perhaps an easier solution. It is described in
https://code.google.com/p/f2py/wiki/FAQ2ed
Original comment by pearu.peterson
on 20 Nov 2013 at 11:04
Ah, cunning. I might try your solution #1 if I have to revisit the problem for
whatever reason; it seems a bit less invasive than the solution I gave above as
that way I don't need to mess about with C or modify the Makefile. Thanks again
for your help!
Original comment by samg...@googlemail.com
on 20 Nov 2013 at 11:10
Original comment by pearu.peterson
on 20 Nov 2013 at 11:38
- Changed state: Done
Note that I have changed the FAQ a little: actually there is no reason to
remove STOP statements, they can be left in provided that they precede a call
to Python function that raises an exception. In this way the Fortran sources
will be still usable by Fortran programs that need to provide the f2pystop
function that does nothing.
Original comment by pearu.peterson
on 21 Nov 2013 at 7:42
Sorry for dragging this back up again, but one thing I noticed was the module
Multiprocessing: http://docs.python.org/2/library/multiprocessing.html - I
haven't had a chance to confirm, but I believe this spawns multiple Python
instances, so in theory if the stop command is called it should only affect its
containing process (I found this as I didn't manage to get the !f2py threadsafe
command to work).
Original comment by samg...@googlemail.com
on 10 Dec 2013 at 10:22