| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program 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 General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Campbell Barton | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file blender/python/intern/bpy_interface_atexit.c
 | 
					
						
							|  |  |  |  *  \ingroup pythonintern | 
					
						
							| 
									
										
										
										
											2011-11-05 08:21:12 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This file inserts an exit callback into pythons 'atexit' module. | 
					
						
							|  |  |  |  * Without this sys.exit() can crash because blender is not properly closing | 
					
						
							|  |  |  |  * resources. | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <Python.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "bpy_util.h"
 | 
					
						
							| 
									
										
										
										
											2012-09-15 01:52:28 +00:00
										 |  |  | #include "bpy.h"  /* own include */
 | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "WM_api.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject *bpy_atexit(PyObject *UNUSED(self), PyObject *UNUSED(args), PyObject *UNUSED(kw)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* close down enough of blender at least not to crash */ | 
					
						
							| 
									
										
										
										
											2011-12-26 12:26:11 +00:00
										 |  |  | 	struct bContext *C = BPy_GetContext(); | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	WM_exit_ext(C, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-26 12:26:11 +00:00
										 |  |  | static PyMethodDef meth_bpy_atexit = {"bpy_atexit", (PyCFunction)bpy_atexit, METH_NOARGS, NULL}; | 
					
						
							|  |  |  | static PyObject *func_bpy_atregister = NULL; /* borrowed referebce, atexit holds */ | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | static void atexit_func_call(const char *func_name, PyObject *atexit_func_arg) | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* note - no error checking, if any of these fail we'll get a crash
 | 
					
						
							|  |  |  | 	 * this is intended, but if its problematic it could be changed | 
					
						
							|  |  |  | 	 * - campbell */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-26 12:26:11 +00:00
										 |  |  | 	PyObject *atexit_mod = PyImport_ImportModuleLevel((char *)"atexit", NULL, NULL, NULL, 0); | 
					
						
							|  |  |  | 	PyObject *atexit_func = PyObject_GetAttrString(atexit_mod, func_name); | 
					
						
							|  |  |  | 	PyObject *args = PyTuple_New(1); | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 	PyObject *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | 	PyTuple_SET_ITEM(args, 0, atexit_func_arg); | 
					
						
							| 
									
										
										
										
											2012-03-18 07:38:51 +00:00
										 |  |  | 	Py_INCREF(atexit_func_arg); /* only incref so we don't dec'ref along with 'args' */ | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-26 12:26:11 +00:00
										 |  |  | 	ret = PyObject_CallObject(atexit_func, args); | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Py_DECREF(atexit_mod); | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | 	Py_DECREF(atexit_func); | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 	Py_DECREF(args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-13 01:29:08 +00:00
										 |  |  | 	if (ret) { | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 		Py_DECREF(ret); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { /* should never happen */ | 
					
						
							|  |  |  | 		PyErr_Print(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BPY_atexit_register(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* atexit module owns this new function reference */ | 
					
						
							| 
									
										
										
										
											2011-09-20 17:07:33 +00:00
										 |  |  | 	BLI_assert(func_bpy_atregister == NULL); | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-26 12:26:11 +00:00
										 |  |  | 	func_bpy_atregister = (PyObject *)PyCFunction_New(&meth_bpy_atexit, NULL); | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | 	atexit_func_call("register", func_bpy_atregister); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BPY_atexit_unregister(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-20 17:07:33 +00:00
										 |  |  | 	BLI_assert(func_bpy_atregister != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-20 15:17:24 +00:00
										 |  |  | 	atexit_func_call("unregister", func_bpy_atregister); | 
					
						
							| 
									
										
										
										
											2012-03-09 00:41:09 +00:00
										 |  |  | 	func_bpy_atregister = NULL; /* don't really need to set but just in case */ | 
					
						
							| 
									
										
										
										
											2011-09-20 12:22:19 +00:00
										 |  |  | } |