4 * ***** BEGIN GPL LICENSE BLOCK *****
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software Foundation,
18 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
20 * Contributor(s): Campbell Barton
22 * ***** END GPL LICENSE BLOCK *****
25 /** \file blender/python/intern/bpy_app.c
26 * \ingroup pythonintern
33 #include "bpy_app_handlers.h"
34 #include "bpy_driver.h"
36 #include "BLI_path_util.h"
37 #include "BLI_utildefines.h"
40 #include "BKE_blender.h"
41 #include "BKE_global.h"
42 #include "structseq.h"
44 #include "../generic/py_capi_utils.h"
47 extern char build_date[];
48 extern char build_time[];
49 extern char build_rev[];
50 extern char build_platform[];
51 extern char build_type[];
52 extern char build_cflags[];
53 extern char build_cxxflags[];
54 extern char build_linkflags[];
55 extern char build_system[];
58 static PyTypeObject BlenderAppType;
60 static PyStructSequence_Field app_info_fields[]= {
61 {(char *)"version", (char *)"The Blender version as a tuple of 3 numbers. eg. (2, 50, 11)"},
62 {(char *)"version_string", (char *)"The Blender version formatted as a string"},
63 {(char *)"version_char", (char *)"The Blender version character (for minor releases)"},
64 {(char *)"version_cycle", (char *)"The release status of this build alpha/beta/rc/release"},
65 {(char *)"binary_path", (char *)"The location of blenders executable, useful for utilities that spawn new instances"},
66 {(char *)"background", (char *)"Boolean, True when blender is running without a user interface (started with -b)"},
69 {(char *)"build_date", (char *)"The date this blender instance was built"},
70 {(char *)"build_time", (char *)"The time this blender instance was built"},
71 {(char *)"build_revision", (char *)"The subversion revision this blender instance was built with"},
72 {(char *)"build_platform", (char *)"The platform this blender instance was built for"},
73 {(char *)"build_type", (char *)"The type of build (Release, Debug)"},
74 {(char *)"build_cflags", (char *)"C compiler flags"},
75 {(char *)"build_cxxflags", (char *)"C++ compiler flags"},
76 {(char *)"build_linkflags", (char *)"Binary linking flags"},
77 {(char *)"build_system", (char *)"Build system used"},
80 {(char *)"handlers", (char *)"Application handler callbacks"},
84 static PyStructSequence_Desc app_info_desc= {
85 (char *)"bpy.app", /* name */
86 (char *)"This module contains application values that remain unchanged during runtime.", /* doc */
87 app_info_fields, /* fields */
88 (sizeof(app_info_fields)/sizeof(PyStructSequence_Field)) - 1
91 #define DO_EXPAND(VAL) VAL ## 1
92 #define EXPAND(VAL) DO_EXPAND(VAL)
94 static PyObject *make_app_info(void)
96 extern char bprogname[]; /* argv[0] from creator.c */
101 app_info= PyStructSequence_New(&BlenderAppType);
102 if (app_info == NULL) {
106 #define SetIntItem(flag) \
107 PyStructSequence_SET_ITEM(app_info, pos++, PyLong_FromLong(flag))
108 #define SetStrItem(str) \
109 PyStructSequence_SET_ITEM(app_info, pos++, PyUnicode_FromString(str))
110 #define SetObjItem(obj) \
111 PyStructSequence_SET_ITEM(app_info, pos++, obj)
113 SetObjItem(Py_BuildValue("(iii)", BLENDER_VERSION/100, BLENDER_VERSION%100, BLENDER_SUBVERSION));
114 SetObjItem(PyUnicode_FromFormat("%d.%02d (sub %d)", BLENDER_VERSION/100, BLENDER_VERSION%100, BLENDER_SUBVERSION));
115 #if defined(BLENDER_VERSION_CHAR) && EXPAND(BLENDER_VERSION_CHAR) != 1
116 SetStrItem(STRINGIFY(BLENDER_VERSION_CHAR));
120 SetStrItem(STRINGIFY(BLENDER_VERSION_CYCLE));
121 SetStrItem(bprogname);
122 SetObjItem(PyBool_FromLong(G.background));
126 SetStrItem(build_date);
127 SetStrItem(build_time);
128 SetStrItem(build_rev);
129 SetStrItem(build_platform);
130 SetStrItem(build_type);
131 SetStrItem(build_cflags);
132 SetStrItem(build_cxxflags);
133 SetStrItem(build_linkflags);
134 SetStrItem(build_system);
136 SetStrItem("Unknown");
137 SetStrItem("Unknown");
138 SetStrItem("Unknown");
139 SetStrItem("Unknown");
140 SetStrItem("Unknown");
141 SetStrItem("Unknown");
142 SetStrItem("Unknown");
143 SetStrItem("Unknown");
144 SetStrItem("Unknown");
147 SetObjItem(BPY_app_handlers_struct());
153 if (PyErr_Occurred()) {
160 /* a few getsets because it makes sense for them to be in bpy.app even though
161 * they are not static */
162 static PyObject *bpy_app_debug_get(PyObject *UNUSED(self), void *UNUSED(closure))
164 return PyBool_FromLong(G.f & G_DEBUG);
167 static int bpy_app_debug_set(PyObject *UNUSED(self), PyObject *value, void *UNUSED(closure))
169 int param= PyObject_IsTrue(value);
172 PyErr_SetString(PyExc_TypeError, "bpy.app.debug can only be True/False");
176 if (param) G.f |= G_DEBUG;
177 else G.f &= ~G_DEBUG;
182 static PyObject *bpy_app_debug_value_get(PyObject *UNUSED(self), void *UNUSED(closure))
184 return PyLong_FromSsize_t(G.rt);
187 static int bpy_app_debug_value_set(PyObject *UNUSED(self), PyObject *value, void *UNUSED(closure))
189 int param= PyLong_AsSsize_t(value);
191 if (param == -1 && PyErr_Occurred()) {
192 PyErr_SetString(PyExc_TypeError, "bpy.app.debug_value can only be set to a whole number");
201 static PyObject *bpy_app_tempdir_get(PyObject *UNUSED(self), void *UNUSED(closure))
203 extern char btempdir[];
204 return PyC_UnicodeFromByte(btempdir);
207 static PyObject *bpy_app_driver_dict_get(PyObject *UNUSED(self), void *UNUSED(closure))
209 if (bpy_pydriver_Dict == NULL)
210 if (bpy_pydriver_create_dict() != 0) {
211 PyErr_SetString(PyExc_RuntimeError, "bpy.app.driver_namespace failed to create dictionary");
215 Py_INCREF(bpy_pydriver_Dict);
216 return bpy_pydriver_Dict;
220 static PyGetSetDef bpy_app_getsets[]= {
221 {(char *)"debug", bpy_app_debug_get, bpy_app_debug_set, (char *)"Boolean, set when blender is running in debug mode (started with -d)", NULL},
222 {(char *)"debug_value", bpy_app_debug_value_get, bpy_app_debug_value_set, (char *)"Int, number which can be set to non-zero values for testing purposes.", NULL},
223 {(char *)"tempdir", bpy_app_tempdir_get, NULL, (char *)"String, the temp directory used by blender (read-only)", NULL},
224 {(char *)"driver_namespace", bpy_app_driver_dict_get, NULL, (char *)"Dictionary for drivers namespace, editable in-place, reset on file load (read-only)", NULL},
225 {NULL, NULL, NULL, NULL, NULL}
228 static void py_struct_seq_getset_init(void)
230 /* tricky dynamic members, not to py-spec! */
233 for (getset= bpy_app_getsets; getset->name; getset++) {
234 PyDict_SetItemString(BlenderAppType.tp_dict, getset->name, PyDescr_NewGetSet(&BlenderAppType, getset));
237 /* end dynamic bpy.app */
240 PyObject *BPY_app_struct(void)
244 PyStructSequence_InitType(&BlenderAppType, &app_info_desc);
246 ret= make_app_info();
248 /* prevent user from creating new instances */
249 BlenderAppType.tp_init= NULL;
250 BlenderAppType.tp_new= NULL;
252 /* kindof a hack ontop of PyStructSequence */
253 py_struct_seq_getset_init();