542 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			542 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| /** \file
 | |
|  * \ingroup pythonintern
 | |
|  *
 | |
|  * This file defines a 'PyStructSequence' accessed via 'bpy.app', mostly
 | |
|  * exposing static applications variables such as version and buildinfo
 | |
|  * however some writable variables have been added such as 'debug' and 'tempdir'
 | |
|  */
 | |
| 
 | |
| #include <Python.h>
 | |
| 
 | |
| #include "bpy_app.h"
 | |
| 
 | |
| #include "bpy_app_alembic.h"
 | |
| #include "bpy_app_build_options.h"
 | |
| #include "bpy_app_ffmpeg.h"
 | |
| #include "bpy_app_ocio.h"
 | |
| #include "bpy_app_oiio.h"
 | |
| #include "bpy_app_opensubdiv.h"
 | |
| #include "bpy_app_openvdb.h"
 | |
| #include "bpy_app_sdl.h"
 | |
| #include "bpy_app_usd.h"
 | |
| 
 | |
| #include "bpy_app_translations.h"
 | |
| 
 | |
| #include "bpy_app_handlers.h"
 | |
| #include "bpy_driver.h"
 | |
| 
 | |
| /* modules */
 | |
| #include "bpy_app_icons.h"
 | |
| #include "bpy_app_timers.h"
 | |
| 
 | |
| #include "BLI_utildefines.h"
 | |
| 
 | |
| #include "BKE_appdir.h"
 | |
| #include "BKE_blender_version.h"
 | |
| #include "BKE_global.h"
 | |
| 
 | |
| #include "DNA_ID.h"
 | |
| 
 | |
| #include "UI_interface_icons.h"
 | |
| 
 | |
| /* for notifiers */
 | |
| #include "WM_api.h"
 | |
| #include "WM_types.h"
 | |
| 
 | |
| #include "../generic/py_capi_utils.h"
 | |
| #include "../generic/python_utildefines.h"
 | |
| 
 | |
| #ifdef BUILD_DATE
 | |
| extern char build_date[];
 | |
| extern char build_time[];
 | |
| extern ulong build_commit_timestamp;
 | |
| extern char build_commit_date[];
 | |
| extern char build_commit_time[];
 | |
| extern char build_hash[];
 | |
| extern char build_branch[];
 | |
| extern char build_platform[];
 | |
| extern char build_type[];
 | |
| extern char build_cflags[];
 | |
| extern char build_cxxflags[];
 | |
| extern char build_linkflags[];
 | |
| extern char build_system[];
 | |
| #endif
 | |
| 
 | |
| static PyTypeObject BlenderAppType;
 | |
| 
 | |
| static PyStructSequence_Field app_info_fields[] = {
 | |
|     {"version", "The Blender version as a tuple of 3 numbers. eg. (2, 83, 1)"},
 | |
|     {"version_string", "The Blender version formatted as a string"},
 | |
|     {"version_cycle", "The release status of this build alpha/beta/rc/release"},
 | |
|     {"version_char", "Deprecated, always an empty string"},
 | |
|     {"binary_path",
 | |
|      "The location of Blender's executable, useful for utilities that open new instances"},
 | |
|     {"background",
 | |
|      "Boolean, True when blender is running without a user interface (started with -b)"},
 | |
|     {"factory_startup", "Boolean, True when blender is running with --factory-startup)"},
 | |
| 
 | |
|     /* buildinfo */
 | |
|     {"build_date", "The date this blender instance was built"},
 | |
|     {"build_time", "The time this blender instance was built"},
 | |
|     {"build_commit_timestamp", "The unix timestamp of commit this blender instance was built"},
 | |
|     {"build_commit_date", "The date of commit this blender instance was built"},
 | |
|     {"build_commit_time", "The time of commit this blender instance was built"},
 | |
|     {"build_hash", "The commit hash this blender instance was built with"},
 | |
|     {"build_branch", "The branch this blender instance was built from"},
 | |
|     {"build_platform", "The platform this blender instance was built for"},
 | |
|     {"build_type", "The type of build (Release, Debug)"},
 | |
|     {"build_cflags", "C compiler flags"},
 | |
|     {"build_cxxflags", "C++ compiler flags"},
 | |
|     {"build_linkflags", "Binary linking flags"},
 | |
|     {"build_system", "Build system used"},
 | |
| 
 | |
|     /* submodules */
 | |
|     {"alembic", "Alembic library information backend"},
 | |
|     {"usd", "USD library information backend"},
 | |
|     {"ffmpeg", "FFmpeg library information backend"},
 | |
|     {"ocio", "OpenColorIO library information backend"},
 | |
|     {"oiio", "OpenImageIO library information backend"},
 | |
|     {"opensubdiv", "OpenSubdiv library information backend"},
 | |
|     {"openvdb", "OpenVDB library information backend"},
 | |
|     {"sdl", "SDL library information backend"},
 | |
|     {"build_options", "A set containing most important enabled optional build features"},
 | |
|     {"handlers", "Application handler callbacks"},
 | |
|     {"translations", "Application and addons internationalization API"},
 | |
| 
 | |
|     /* Modules (not struct sequence). */
 | |
|     {"icons", "Manage custom icons"},
 | |
|     {"timers", "Manage timers"},
 | |
|     {NULL},
 | |
| };
 | |
| 
 | |
| PyDoc_STRVAR(bpy_app_doc,
 | |
|              "This module contains application values that remain unchanged during runtime.\n"
 | |
|              "\n"
 | |
|              "Submodules:\n"
 | |
|              "\n"
 | |
|              ".. toctree::\n"
 | |
|              "   :maxdepth: 1\n"
 | |
|              "\n"
 | |
|              "   bpy.app.handlers.rst\n"
 | |
|              "   bpy.app.icons.rst\n"
 | |
|              "   bpy.app.timers.rst\n"
 | |
|              "   bpy.app.translations.rst\n");
 | |
| 
 | |
| static PyStructSequence_Desc app_info_desc = {
 | |
|     "bpy.app",       /* name */
 | |
|     bpy_app_doc,     /* doc */
 | |
|     app_info_fields, /* fields */
 | |
|     ARRAY_SIZE(app_info_fields) - 1,
 | |
| };
 | |
| 
 | |
| static PyObject *make_app_info(void)
 | |
| {
 | |
|   PyObject *app_info;
 | |
|   int pos = 0;
 | |
| 
 | |
|   app_info = PyStructSequence_New(&BlenderAppType);
 | |
|   if (app_info == NULL) {
 | |
|     return NULL;
 | |
|   }
 | |
| #define SetIntItem(flag) PyStructSequence_SET_ITEM(app_info, pos++, PyLong_FromLong(flag))
 | |
| #define SetStrItem(str) PyStructSequence_SET_ITEM(app_info, pos++, PyUnicode_FromString(str))
 | |
| #define SetBytesItem(str) PyStructSequence_SET_ITEM(app_info, pos++, PyBytes_FromString(str))
 | |
| #define SetObjItem(obj) PyStructSequence_SET_ITEM(app_info, pos++, obj)
 | |
| 
 | |
|   SetObjItem(
 | |
|       PyC_Tuple_Pack_I32(BLENDER_VERSION / 100, BLENDER_VERSION % 100, BLENDER_VERSION_PATCH));
 | |
|   SetStrItem(BKE_blender_version_string());
 | |
| 
 | |
|   SetStrItem(STRINGIFY(BLENDER_VERSION_CYCLE));
 | |
|   SetStrItem("");
 | |
|   SetStrItem(BKE_appdir_program_path());
 | |
|   SetObjItem(PyBool_FromLong(G.background));
 | |
|   SetObjItem(PyBool_FromLong(G.factory_startup));
 | |
| 
 | |
|   /* build info, use bytes since we can't assume _any_ encoding:
 | |
|    * see patch T30154 for issue */
 | |
| #ifdef BUILD_DATE
 | |
|   SetBytesItem(build_date);
 | |
|   SetBytesItem(build_time);
 | |
|   SetIntItem(build_commit_timestamp);
 | |
|   SetBytesItem(build_commit_date);
 | |
|   SetBytesItem(build_commit_time);
 | |
|   SetBytesItem(build_hash);
 | |
|   SetBytesItem(build_branch);
 | |
|   SetBytesItem(build_platform);
 | |
|   SetBytesItem(build_type);
 | |
|   SetBytesItem(build_cflags);
 | |
|   SetBytesItem(build_cxxflags);
 | |
|   SetBytesItem(build_linkflags);
 | |
|   SetBytesItem(build_system);
 | |
| #else
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetIntItem(0);
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
|   SetBytesItem("Unknown");
 | |
| #endif
 | |
| 
 | |
|   SetObjItem(BPY_app_alembic_struct());
 | |
|   SetObjItem(BPY_app_usd_struct());
 | |
|   SetObjItem(BPY_app_ffmpeg_struct());
 | |
|   SetObjItem(BPY_app_ocio_struct());
 | |
|   SetObjItem(BPY_app_oiio_struct());
 | |
|   SetObjItem(BPY_app_opensubdiv_struct());
 | |
|   SetObjItem(BPY_app_openvdb_struct());
 | |
|   SetObjItem(BPY_app_sdl_struct());
 | |
|   SetObjItem(BPY_app_build_options_struct());
 | |
|   SetObjItem(BPY_app_handlers_struct());
 | |
|   SetObjItem(BPY_app_translations_struct());
 | |
| 
 | |
|   /* modules */
 | |
|   SetObjItem(BPY_app_icons_module());
 | |
|   SetObjItem(BPY_app_timers_module());
 | |
| 
 | |
| #undef SetIntItem
 | |
| #undef SetStrItem
 | |
| #undef SetBytesItem
 | |
| #undef SetObjItem
 | |
| 
 | |
|   if (PyErr_Occurred()) {
 | |
|     Py_CLEAR(app_info);
 | |
|     return NULL;
 | |
|   }
 | |
|   return app_info;
 | |
| }
 | |
| 
 | |
| /* a few getsets because it makes sense for them to be in bpy.app even though
 | |
|  * they are not static */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     bpy_app_debug_doc,
 | |
|     "Boolean, for debug info (started with --debug / --debug_* matching this attribute name)");
 | |
| static PyObject *bpy_app_debug_get(PyObject *UNUSED(self), void *closure)
 | |
| {
 | |
|   const int flag = POINTER_AS_INT(closure);
 | |
|   return PyBool_FromLong(G.debug & flag);
 | |
| }
 | |
| 
 | |
| static int bpy_app_debug_set(PyObject *UNUSED(self), PyObject *value, void *closure)
 | |
| {
 | |
|   const int flag = POINTER_AS_INT(closure);
 | |
|   const int param = PyObject_IsTrue(value);
 | |
| 
 | |
|   if (param == -1) {
 | |
|     PyErr_SetString(PyExc_TypeError, "bpy.app.debug can only be True/False");
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   if (param) {
 | |
|     G.debug |= flag;
 | |
|   }
 | |
|   else {
 | |
|     G.debug &= ~flag;
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     bpy_app_global_flag_doc,
 | |
|     "Boolean, for application behavior (started with --enable-* matching this attribute name)");
 | |
| static PyObject *bpy_app_global_flag_get(PyObject *UNUSED(self), void *closure)
 | |
| {
 | |
|   const int flag = POINTER_AS_INT(closure);
 | |
|   return PyBool_FromLong(G.f & flag);
 | |
| }
 | |
| 
 | |
| static int bpy_app_global_flag_set(PyObject *UNUSED(self), PyObject *value, void *closure)
 | |
| {
 | |
|   const int flag = POINTER_AS_INT(closure);
 | |
|   const int param = PyObject_IsTrue(value);
 | |
| 
 | |
|   if (param == -1) {
 | |
|     PyErr_SetString(PyExc_TypeError, "bpy.app.use_* can only be True/False");
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   if (param) {
 | |
|     G.f |= flag;
 | |
|   }
 | |
|   else {
 | |
|     G.f &= ~flag;
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static int bpy_app_global_flag_set__only_disable(PyObject *UNUSED(self),
 | |
|                                                  PyObject *value,
 | |
|                                                  void *closure)
 | |
| {
 | |
|   const int param = PyObject_IsTrue(value);
 | |
|   if (param == 1) {
 | |
|     PyErr_SetString(PyExc_ValueError, "This bpy.app.use_* option can only be disabled");
 | |
|     return -1;
 | |
|   }
 | |
|   return bpy_app_global_flag_set(NULL, value, closure);
 | |
| }
 | |
| 
 | |
| #define BROKEN_BINARY_PATH_PYTHON_HACK
 | |
| 
 | |
| PyDoc_STRVAR(bpy_app_binary_path_python_doc,
 | |
|              "String, the path to the python executable (read-only)");
 | |
| static PyObject *bpy_app_binary_path_python_get(PyObject *self, void *UNUSED(closure))
 | |
| {
 | |
|   /* refcount is held in BlenderAppType.tp_dict */
 | |
|   static PyObject *ret = NULL;
 | |
| 
 | |
|   if (ret == NULL) {
 | |
|     /* only run once */
 | |
|     char fullpath[1024];
 | |
|     BKE_appdir_program_python_search(
 | |
|         fullpath, sizeof(fullpath), PY_MAJOR_VERSION, PY_MINOR_VERSION);
 | |
|     ret = PyC_UnicodeFromByte(fullpath);
 | |
| #ifdef BROKEN_BINARY_PATH_PYTHON_HACK
 | |
|     Py_INCREF(ret);
 | |
|     UNUSED_VARS(self);
 | |
| #else
 | |
|     PyDict_SetItem(
 | |
|         BlenderAppType.tp_dict,
 | |
|         /* XXX BAAAADDDDDD! self is not a PyDescr at all! it's bpy.app!!! */ PyDescr_NAME(self),
 | |
|         ret);
 | |
| #endif
 | |
|   }
 | |
|   else {
 | |
|     Py_INCREF(ret);
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(bpy_app_debug_value_doc,
 | |
|              "Short, number which can be set to non-zero values for testing purposes");
 | |
| static PyObject *bpy_app_debug_value_get(PyObject *UNUSED(self), void *UNUSED(closure))
 | |
| {
 | |
|   return PyLong_FromLong(G.debug_value);
 | |
| }
 | |
| 
 | |
| static int bpy_app_debug_value_set(PyObject *UNUSED(self), PyObject *value, void *UNUSED(closure))
 | |
| {
 | |
|   const short param = PyC_Long_AsI16(value);
 | |
| 
 | |
|   if (param == -1 && PyErr_Occurred()) {
 | |
|     PyC_Err_SetString_Prefix(PyExc_TypeError,
 | |
|                              "bpy.app.debug_value can only be set to a whole number");
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   G.debug_value = param;
 | |
| 
 | |
|   WM_main_add_notifier(NC_WINDOW, NULL);
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(bpy_app_tempdir_doc, "String, the temp directory used by blender (read-only)");
 | |
| static PyObject *bpy_app_tempdir_get(PyObject *UNUSED(self), void *UNUSED(closure))
 | |
| {
 | |
|   return PyC_UnicodeFromByte(BKE_tempdir_session());
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     bpy_app_driver_dict_doc,
 | |
|     "Dictionary for drivers namespace, editable in-place, reset on file load (read-only)");
 | |
| static PyObject *bpy_app_driver_dict_get(PyObject *UNUSED(self), void *UNUSED(closure))
 | |
| {
 | |
|   if (bpy_pydriver_Dict == NULL) {
 | |
|     if (bpy_pydriver_create_dict() != 0) {
 | |
|       PyErr_SetString(PyExc_RuntimeError, "bpy.app.driver_namespace failed to create dictionary");
 | |
|       return NULL;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return Py_INCREF_RET(bpy_pydriver_Dict);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(bpy_app_preview_render_size_doc,
 | |
|              "Reference size for icon/preview renders (read-only)");
 | |
| static PyObject *bpy_app_preview_render_size_get(PyObject *UNUSED(self), void *closure)
 | |
| {
 | |
|   return PyLong_FromLong((long)UI_icon_preview_to_render_size(POINTER_AS_INT(closure)));
 | |
| }
 | |
| 
 | |
| static PyObject *bpy_app_autoexec_fail_message_get(PyObject *UNUSED(self), void *UNUSED(closure))
 | |
| {
 | |
|   return PyC_UnicodeFromByte(G.autoexec_fail);
 | |
| }
 | |
| 
 | |
| static PyGetSetDef bpy_app_getsets[] = {
 | |
|     {"debug", bpy_app_debug_get, bpy_app_debug_set, bpy_app_debug_doc, (void *)G_DEBUG},
 | |
|     {"debug_ffmpeg",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_FFMPEG},
 | |
|     {"debug_freestyle",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_FREESTYLE},
 | |
|     {"debug_python",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_PYTHON},
 | |
|     {"debug_events",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_EVENTS},
 | |
|     {"debug_handlers",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_HANDLERS},
 | |
|     {"debug_wm", bpy_app_debug_get, bpy_app_debug_set, bpy_app_debug_doc, (void *)G_DEBUG_WM},
 | |
|     {"debug_depsgraph",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH},
 | |
|     {"debug_depsgraph_build",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH_BUILD},
 | |
|     {"debug_depsgraph_eval",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH_EVAL},
 | |
|     {"debug_depsgraph_tag",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH_TAG},
 | |
|     {"debug_depsgraph_time",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH_TIME},
 | |
|     {"debug_depsgraph_pretty",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_DEPSGRAPH_PRETTY},
 | |
|     {"debug_simdata",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_SIMDATA},
 | |
|     {"debug_gpumem",
 | |
|      bpy_app_debug_get,
 | |
|      bpy_app_debug_set,
 | |
|      bpy_app_debug_doc,
 | |
|      (void *)G_DEBUG_GPU_MEM},
 | |
|     {"debug_io", bpy_app_debug_get, bpy_app_debug_set, bpy_app_debug_doc, (void *)G_DEBUG_IO},
 | |
| 
 | |
|     {"use_event_simulate",
 | |
|      bpy_app_global_flag_get,
 | |
|      bpy_app_global_flag_set__only_disable,
 | |
|      bpy_app_global_flag_doc,
 | |
|      (void *)G_FLAG_EVENT_SIMULATE},
 | |
| 
 | |
|     {"use_userpref_skip_save_on_exit",
 | |
|      bpy_app_global_flag_get,
 | |
|      bpy_app_global_flag_set,
 | |
|      bpy_app_global_flag_doc,
 | |
|      (void *)G_FLAG_USERPREF_NO_SAVE_ON_EXIT},
 | |
| 
 | |
|     {"binary_path_python",
 | |
|      bpy_app_binary_path_python_get,
 | |
|      NULL,
 | |
|      bpy_app_binary_path_python_doc,
 | |
|      NULL},
 | |
| 
 | |
|     {"debug_value",
 | |
|      bpy_app_debug_value_get,
 | |
|      bpy_app_debug_value_set,
 | |
|      bpy_app_debug_value_doc,
 | |
|      NULL},
 | |
|     {"tempdir", bpy_app_tempdir_get, NULL, bpy_app_tempdir_doc, NULL},
 | |
|     {"driver_namespace", bpy_app_driver_dict_get, NULL, bpy_app_driver_dict_doc, NULL},
 | |
| 
 | |
|     {"render_icon_size",
 | |
|      bpy_app_preview_render_size_get,
 | |
|      NULL,
 | |
|      bpy_app_preview_render_size_doc,
 | |
|      (void *)ICON_SIZE_ICON},
 | |
|     {"render_preview_size",
 | |
|      bpy_app_preview_render_size_get,
 | |
|      NULL,
 | |
|      bpy_app_preview_render_size_doc,
 | |
|      (void *)ICON_SIZE_PREVIEW},
 | |
| 
 | |
|     /* security */
 | |
|     {"autoexec_fail", bpy_app_global_flag_get, NULL, NULL, (void *)G_FLAG_SCRIPT_AUTOEXEC_FAIL},
 | |
|     {"autoexec_fail_quiet",
 | |
|      bpy_app_global_flag_get,
 | |
|      NULL,
 | |
|      NULL,
 | |
|      (void *)G_FLAG_SCRIPT_AUTOEXEC_FAIL_QUIET},
 | |
|     {"autoexec_fail_message", bpy_app_autoexec_fail_message_get, NULL, NULL, NULL},
 | |
|     {NULL, NULL, NULL, NULL, NULL},
 | |
| };
 | |
| 
 | |
| static void py_struct_seq_getset_init(void)
 | |
| {
 | |
|   /* tricky dynamic members, not to py-spec! */
 | |
|   for (PyGetSetDef *getset = bpy_app_getsets; getset->name; getset++) {
 | |
|     PyObject *item = PyDescr_NewGetSet(&BlenderAppType, getset);
 | |
|     PyDict_SetItem(BlenderAppType.tp_dict, PyDescr_NAME(item), item);
 | |
|     Py_DECREF(item);
 | |
|   }
 | |
| }
 | |
| /* end dynamic bpy.app */
 | |
| 
 | |
| PyObject *BPY_app_struct(void)
 | |
| {
 | |
|   PyObject *ret;
 | |
| 
 | |
|   PyStructSequence_InitType(&BlenderAppType, &app_info_desc);
 | |
| 
 | |
|   ret = make_app_info();
 | |
| 
 | |
|   /* prevent user from creating new instances */
 | |
|   BlenderAppType.tp_init = NULL;
 | |
|   BlenderAppType.tp_new = NULL;
 | |
|   BlenderAppType.tp_hash = (hashfunc)
 | |
|       _Py_HashPointer; /* without this we can't do set(sys.modules) T29635. */
 | |
| 
 | |
|   /* kindof a hack ontop of PyStructSequence */
 | |
|   py_struct_seq_getset_init();
 | |
| 
 | |
|   return ret;
 | |
| }
 | 
