528 lines
14 KiB
C++
528 lines
14 KiB
C++
/*
|
|
* Execute Python scripts
|
|
*
|
|
* $Id$
|
|
*
|
|
* ***** 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.
|
|
*
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
* All rights reserved.
|
|
*
|
|
* The Original Code is: all of this file.
|
|
*
|
|
* Contributor(s): none yet.
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*/
|
|
|
|
/** \file gameengine/GameLogic/SCA_PythonController.cpp
|
|
* \ingroup gamelogic
|
|
*/
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include "SCA_PythonController.h"
|
|
#include "SCA_LogicManager.h"
|
|
#include "SCA_ISensor.h"
|
|
#include "SCA_IActuator.h"
|
|
#include "PyObjectPlus.h"
|
|
|
|
#ifdef WITH_PYTHON
|
|
#include "compile.h"
|
|
#include "eval.h"
|
|
#endif // WITH_PYTHON
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
// initialize static member variables
|
|
SCA_PythonController* SCA_PythonController::m_sCurrentController = NULL;
|
|
|
|
|
|
SCA_PythonController::SCA_PythonController(SCA_IObject* gameobj, int mode)
|
|
: SCA_IController(gameobj),
|
|
#ifdef WITH_PYTHON
|
|
m_bytecode(NULL),
|
|
m_function(NULL),
|
|
#endif
|
|
m_function_argc(0),
|
|
m_bModified(true),
|
|
m_debug(false),
|
|
m_mode(mode)
|
|
#ifdef WITH_PYTHON
|
|
, m_pythondictionary(NULL)
|
|
#endif
|
|
|
|
{
|
|
|
|
}
|
|
|
|
/*
|
|
//debugging
|
|
CValue* SCA_PythonController::AddRef()
|
|
{
|
|
//printf("AddRef refcount = %i\n",GetRefCount());
|
|
return CValue::AddRef();
|
|
}
|
|
int SCA_PythonController::Release()
|
|
{
|
|
//printf("Release refcount = %i\n",GetRefCount());
|
|
return CValue::Release();
|
|
}
|
|
*/
|
|
|
|
|
|
|
|
SCA_PythonController::~SCA_PythonController()
|
|
{
|
|
|
|
#ifdef WITH_PYTHON
|
|
//printf("released python byte script\n");
|
|
|
|
Py_XDECREF(m_bytecode);
|
|
Py_XDECREF(m_function);
|
|
|
|
if (m_pythondictionary) {
|
|
// break any circular references in the dictionary
|
|
PyDict_Clear(m_pythondictionary);
|
|
Py_DECREF(m_pythondictionary);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
CValue* SCA_PythonController::GetReplica()
|
|
{
|
|
SCA_PythonController* replica = new SCA_PythonController(*this);
|
|
|
|
#ifdef WITH_PYTHON
|
|
/* why is this needed at all??? - m_bytecode is NULL'd below so this doesnt make sense
|
|
* but removing it crashes blender (with YoFrankie). so leave in for now - Campbell */
|
|
Py_XINCREF(replica->m_bytecode);
|
|
|
|
Py_XINCREF(replica->m_function); // this is ok since its not set to NULL
|
|
replica->m_bModified = replica->m_bytecode == NULL;
|
|
|
|
// The replica->m_pythondictionary is stolen - replace with a copy.
|
|
if (m_pythondictionary)
|
|
replica->m_pythondictionary = PyDict_Copy(m_pythondictionary);
|
|
|
|
/*
|
|
// The other option is to incref the replica->m_pythondictionary -
|
|
// the replica objects can then share data.
|
|
if (m_pythondictionary)
|
|
Py_INCREF(replica->m_pythondictionary);
|
|
*/
|
|
#endif
|
|
|
|
// this will copy properties and so on...
|
|
replica->ProcessReplica();
|
|
|
|
return replica;
|
|
}
|
|
|
|
|
|
|
|
void SCA_PythonController::SetScriptText(const STR_String& text)
|
|
{
|
|
m_scriptText = text;
|
|
m_bModified = true;
|
|
}
|
|
|
|
|
|
|
|
void SCA_PythonController::SetScriptName(const STR_String& name)
|
|
{
|
|
m_scriptName = name;
|
|
}
|
|
|
|
|
|
#ifdef WITH_PYTHON
|
|
void SCA_PythonController::SetNamespace(PyObject* pythondictionary)
|
|
{
|
|
if (m_pythondictionary)
|
|
{
|
|
PyDict_Clear(m_pythondictionary);
|
|
Py_DECREF(m_pythondictionary);
|
|
}
|
|
m_pythondictionary = PyDict_Copy(pythondictionary); /* new reference */
|
|
|
|
/* Without __file__ set the sys.argv[0] is used for the filename
|
|
* which ends up with lines from the blender binary being printed in the console */
|
|
PyDict_SetItemString(m_pythondictionary, "__file__", PyUnicode_FromString(m_scriptName.Ptr()));
|
|
|
|
}
|
|
#endif
|
|
|
|
int SCA_PythonController::IsTriggered(class SCA_ISensor* sensor)
|
|
{
|
|
if (std::find(m_triggeredSensors.begin(), m_triggeredSensors.end(), sensor) !=
|
|
m_triggeredSensors.end())
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
#ifdef WITH_PYTHON
|
|
|
|
/* warning, self is not the SCA_PythonController, its a PyObjectPlus_Proxy */
|
|
PyObject* SCA_PythonController::sPyGetCurrentController(PyObject *self)
|
|
{
|
|
if(m_sCurrentController==NULL)
|
|
{
|
|
PyErr_SetString(PyExc_SystemError, "bge.logic.getCurrentController(), this function is being run outside the python controllers context, or blenders internal state is corrupt.");
|
|
return NULL;
|
|
}
|
|
return m_sCurrentController->GetProxy();
|
|
}
|
|
|
|
SCA_IActuator* SCA_PythonController::LinkedActuatorFromPy(PyObject *value)
|
|
{
|
|
// for safety, todo: only allow for registered actuators (pointertable)
|
|
// we don't want to crash gameengine/blender by python scripts
|
|
std::vector<SCA_IActuator*> lacts = m_sCurrentController->GetLinkedActuators();
|
|
std::vector<SCA_IActuator*>::iterator it;
|
|
|
|
if (PyUnicode_Check(value)) {
|
|
/* get the actuator from the name */
|
|
char *name= _PyUnicode_AsString(value);
|
|
for(it = lacts.begin(); it!= lacts.end(); ++it) {
|
|
if( name == (*it)->GetName() ) {
|
|
return *it;
|
|
}
|
|
}
|
|
}
|
|
else if (PyObject_TypeCheck(value, &SCA_IActuator::Type)) {
|
|
PyObjectPlus *value_plus= BGE_PROXY_REF(value);
|
|
for(it = lacts.begin(); it!= lacts.end(); ++it) {
|
|
if( static_cast<SCA_IActuator*>(value_plus) == (*it) ) {
|
|
return *it;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* set the exception */
|
|
PyObject *value_str = PyObject_Repr(value); /* new ref */
|
|
PyErr_Format(PyExc_ValueError, "'%s' not in this python controllers actuator list", _PyUnicode_AsString(value_str));
|
|
Py_DECREF(value_str);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
const char* SCA_PythonController::sPyGetCurrentController__doc__ = "getCurrentController()";
|
|
|
|
PyTypeObject SCA_PythonController::Type = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
"SCA_PythonController",
|
|
sizeof(PyObjectPlus_Proxy),
|
|
0,
|
|
py_base_dealloc,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
py_base_repr,
|
|
0,0,0,0,0,0,0,0,0,
|
|
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,
|
|
0,0,0,0,0,0,0,
|
|
Methods,
|
|
0,
|
|
0,
|
|
&SCA_IController::Type,
|
|
0,0,0,0,0,0,
|
|
py_base_new
|
|
};
|
|
|
|
PyMethodDef SCA_PythonController::Methods[] = {
|
|
{"activate", (PyCFunction) SCA_PythonController::sPyActivate, METH_O},
|
|
{"deactivate", (PyCFunction) SCA_PythonController::sPyDeActivate, METH_O},
|
|
{NULL,NULL} //Sentinel
|
|
};
|
|
|
|
PyAttributeDef SCA_PythonController::Attributes[] = {
|
|
KX_PYATTRIBUTE_RW_FUNCTION("script", SCA_PythonController, pyattr_get_script, pyattr_set_script),
|
|
KX_PYATTRIBUTE_INT_RO("mode", SCA_PythonController, m_mode),
|
|
{ NULL } //Sentinel
|
|
};
|
|
|
|
void SCA_PythonController::ErrorPrint(const char *error_msg)
|
|
{
|
|
printf("%s - object '%s', controller '%s':\n", error_msg, GetParent()->GetName().Ptr(), GetName().Ptr());
|
|
PyErr_Print();
|
|
|
|
/* Added in 2.48a, the last_traceback can reference Objects for example, increasing
|
|
* their user count. Not to mention holding references to wrapped data.
|
|
* This is especially bad when the PyObject for the wrapped data is free'd, after blender
|
|
* has already dealocated the pointer */
|
|
PySys_SetObject( (char *)"last_traceback", NULL);
|
|
PyErr_Clear(); /* just to be sure */
|
|
}
|
|
|
|
bool SCA_PythonController::Compile()
|
|
{
|
|
//printf("py script modified '%s'\n", m_scriptName.Ptr());
|
|
m_bModified= false;
|
|
|
|
// if a script already exists, decref it before replace the pointer to a new script
|
|
if (m_bytecode) {
|
|
Py_DECREF(m_bytecode);
|
|
m_bytecode=NULL;
|
|
}
|
|
|
|
// recompile the scripttext into bytecode
|
|
m_bytecode = Py_CompileString(m_scriptText.Ptr(), m_scriptName.Ptr(), Py_file_input);
|
|
|
|
if (m_bytecode) {
|
|
return true;
|
|
} else {
|
|
ErrorPrint("Python error compiling script");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool SCA_PythonController::Import()
|
|
{
|
|
//printf("py module modified '%s'\n", m_scriptName.Ptr());
|
|
m_bModified= false;
|
|
|
|
/* incase we re-import */
|
|
Py_XDECREF(m_function);
|
|
m_function= NULL;
|
|
|
|
STR_String mod_path_str= m_scriptText; /* just for storage, use C style string access */
|
|
char *mod_path= mod_path_str.Ptr();
|
|
char *function_string;
|
|
|
|
function_string= strrchr(mod_path, '.');
|
|
|
|
if(function_string == NULL) {
|
|
printf("Python module name formatting error in object '%s', controller '%s':\n\texpected 'SomeModule.Func', got '%s'\n", GetParent()->GetName().Ptr(), GetName().Ptr(), m_scriptText.Ptr());
|
|
return false;
|
|
}
|
|
|
|
*function_string= '\0';
|
|
function_string++;
|
|
|
|
// Import the module and print an error if it's not found
|
|
PyObject *mod = PyImport_ImportModule(mod_path);
|
|
|
|
if (mod == NULL) {
|
|
ErrorPrint("Python module can't be imported");
|
|
return false;
|
|
}
|
|
|
|
if(m_debug)
|
|
mod = PyImport_ReloadModule(mod);
|
|
|
|
if (mod == NULL) {
|
|
ErrorPrint("Python module can't be reloaded");
|
|
return false;
|
|
}
|
|
|
|
// Get the function object
|
|
m_function = PyObject_GetAttrString(mod, function_string);
|
|
|
|
// DECREF the module as we don't need it anymore
|
|
Py_DECREF(mod);
|
|
|
|
if(m_function==NULL) {
|
|
if(PyErr_Occurred())
|
|
ErrorPrint("Python controller found the module but could not access the function");
|
|
else
|
|
printf("Python module error in object '%s', controller '%s':\n '%s' module found but function missing\n", GetParent()->GetName().Ptr(), GetName().Ptr(), m_scriptText.Ptr());
|
|
return false;
|
|
}
|
|
|
|
if(!PyCallable_Check(m_function)) {
|
|
Py_DECREF(m_function);
|
|
m_function = NULL;
|
|
printf("Python module function error in object '%s', controller '%s':\n '%s' not callable\n", GetParent()->GetName().Ptr(), GetName().Ptr(), m_scriptText.Ptr());
|
|
return false;
|
|
}
|
|
|
|
m_function_argc = 0; /* rare cases this could be a function that isnt defined in python, assume zero args */
|
|
if (PyFunction_Check(m_function)) {
|
|
m_function_argc= ((PyCodeObject *)PyFunction_GET_CODE(m_function))->co_argcount;
|
|
}
|
|
|
|
if(m_function_argc > 1) {
|
|
Py_DECREF(m_function);
|
|
m_function = NULL;
|
|
printf("Python module function in object '%s', controller '%s':\n '%s' takes %d args, should be zero or 1 controller arg\n", GetParent()->GetName().Ptr(), GetName().Ptr(), m_scriptText.Ptr(), m_function_argc);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void SCA_PythonController::Trigger(SCA_LogicManager* logicmgr)
|
|
{
|
|
m_sCurrentController = this;
|
|
m_sCurrentLogicManager = logicmgr;
|
|
|
|
PyObject *excdict= NULL;
|
|
PyObject* resultobj= NULL;
|
|
|
|
switch(m_mode) {
|
|
case SCA_PYEXEC_SCRIPT:
|
|
{
|
|
if (m_bModified)
|
|
if (Compile()==false) // sets m_bModified to false
|
|
return;
|
|
if (!m_bytecode)
|
|
return;
|
|
|
|
/*
|
|
* This part here with excdict is a temporary patch
|
|
* to avoid python/gameengine crashes when python
|
|
* inadvertently holds references to game objects
|
|
* in global variables.
|
|
*
|
|
* The idea is always make a fresh dictionary, and
|
|
* destroy it right after it is used to make sure
|
|
* python won't hold any gameobject references.
|
|
*
|
|
* Note that the PyDict_Clear _is_ necessary before
|
|
* the Py_DECREF() because it is possible for the
|
|
* variables inside the dictionary to hold references
|
|
* to the dictionary (ie. generate a cycle), so we
|
|
* break it by hand, then DECREF (which in this case
|
|
* should always ensure excdict is cleared).
|
|
*/
|
|
|
|
excdict= PyDict_Copy(m_pythondictionary);
|
|
|
|
resultobj = PyEval_EvalCode((PyObject *)m_bytecode, excdict, excdict);
|
|
|
|
/* PyRun_SimpleString(m_scriptText.Ptr()); */
|
|
break;
|
|
}
|
|
case SCA_PYEXEC_MODULE:
|
|
{
|
|
if (m_bModified || m_debug)
|
|
if (Import()==false) // sets m_bModified to false
|
|
return;
|
|
if (!m_function)
|
|
return;
|
|
|
|
PyObject *args= NULL;
|
|
|
|
if(m_function_argc==1) {
|
|
args = PyTuple_New(1);
|
|
PyTuple_SET_ITEM(args, 0, GetProxy());
|
|
}
|
|
|
|
resultobj = PyObject_CallObject(m_function, args);
|
|
Py_XDECREF(args);
|
|
break;
|
|
}
|
|
|
|
} /* end switch */
|
|
|
|
|
|
|
|
/* Free the return value and print the error */
|
|
if (resultobj)
|
|
Py_DECREF(resultobj);
|
|
else
|
|
ErrorPrint("Python script error");
|
|
|
|
if(excdict) /* Only for SCA_PYEXEC_SCRIPT types */
|
|
{
|
|
/* clear after PyErrPrint - seems it can be using
|
|
* something in this dictionary and crash? */
|
|
// This doesn't appear to be needed anymore
|
|
//PyDict_Clear(excdict);
|
|
Py_DECREF(excdict);
|
|
}
|
|
|
|
m_triggeredSensors.clear();
|
|
m_sCurrentController = NULL;
|
|
}
|
|
|
|
PyObject* SCA_PythonController::PyActivate(PyObject *value)
|
|
{
|
|
if(m_sCurrentController != this) {
|
|
PyErr_SetString(PyExc_SystemError, "Cannot add an actuator from a non-active controller");
|
|
return NULL;
|
|
}
|
|
|
|
SCA_IActuator* actu = LinkedActuatorFromPy(value);
|
|
if(actu==NULL)
|
|
return NULL;
|
|
|
|
m_sCurrentLogicManager->AddActiveActuator((SCA_IActuator*)actu, true);
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyObject* SCA_PythonController::PyDeActivate(PyObject *value)
|
|
{
|
|
if(m_sCurrentController != this) {
|
|
PyErr_SetString(PyExc_SystemError, "Cannot add an actuator from a non-active controller");
|
|
return NULL;
|
|
}
|
|
|
|
SCA_IActuator* actu = LinkedActuatorFromPy(value);
|
|
if(actu==NULL)
|
|
return NULL;
|
|
|
|
m_sCurrentLogicManager->AddActiveActuator((SCA_IActuator*)actu, false);
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyObject* SCA_PythonController::pyattr_get_script(void *self_v, const KX_PYATTRIBUTE_DEF *attrdef)
|
|
{
|
|
//SCA_PythonController* self= static_cast<SCA_PythonController*>(static_cast<SCA_IController*>(static_cast<SCA_ILogicBrick*>(static_cast<CValue*>(static_cast<PyObjectPlus*>(self_v)))));
|
|
// static_cast<void *>(dynamic_cast<Derived *>(obj)) - static_cast<void *>(obj)
|
|
|
|
SCA_PythonController* self= static_cast<SCA_PythonController*>(self_v);
|
|
return PyUnicode_FromString(self->m_scriptText);
|
|
}
|
|
|
|
|
|
|
|
int SCA_PythonController::pyattr_set_script(void *self_v, const KX_PYATTRIBUTE_DEF *attrdef, PyObject *value)
|
|
{
|
|
SCA_PythonController* self= static_cast<SCA_PythonController*>(self_v);
|
|
|
|
char *scriptArg = _PyUnicode_AsString(value);
|
|
|
|
if (scriptArg==NULL) {
|
|
PyErr_SetString(PyExc_TypeError, "controller.script = string: Python Controller, expected a string script text");
|
|
return PY_SET_ATTR_FAIL;
|
|
}
|
|
|
|
/* set scripttext sets m_bModified to true,
|
|
so next time the script is needed, a reparse into byte code is done */
|
|
self->SetScriptText(scriptArg);
|
|
|
|
return PY_SET_ATTR_SUCCESS;
|
|
}
|
|
|
|
#else // WITH_PYTHON
|
|
|
|
void SCA_PythonController::Trigger(SCA_LogicManager* logicmgr)
|
|
{
|
|
/* intentionally blank */
|
|
}
|
|
|
|
#endif // WITH_PYTHON
|
|
|
|
/* eof */
|