+ unify DLL export defines to namespace names
git-svn-id: https://free-cad.svn.sourceforge.net/svnroot/free-cad/trunk@5000 e8eeb9e2-ec13-0410-a4a9-efa5cf37419d
This commit is contained in:
572
src/Gui/PythonDebugger.cpp
Normal file
572
src/Gui/PythonDebugger.cpp
Normal file
@@ -0,0 +1,572 @@
|
||||
/***************************************************************************
|
||||
* Copyright (c) 2010 Werner Mayer <wmayer[at]users.sourceforge.net> *
|
||||
* *
|
||||
* This file is part of the FreeCAD CAx development system. *
|
||||
* *
|
||||
* This library is free software; you can redistribute it and/or *
|
||||
* modify it under the terms of the GNU Library General Public *
|
||||
* License as published by the Free Software Foundation; either *
|
||||
* version 2 of the License, or (at your option) any later version. *
|
||||
* *
|
||||
* This library 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 Library General Public License for more details. *
|
||||
* *
|
||||
* You should have received a copy of the GNU Library General Public *
|
||||
* License along with this library; see the file COPYING.LIB. If not, *
|
||||
* write to the Free Software Foundation, Inc., 59 Temple Place, *
|
||||
* Suite 330, Boston, MA 02111-1307, USA *
|
||||
* *
|
||||
***************************************************************************/
|
||||
|
||||
|
||||
#include "PreCompiled.h"
|
||||
#ifndef _PreComp_
|
||||
# include <QEventLoop>
|
||||
# include <QCoreApplication>
|
||||
# include <QFileInfo>
|
||||
# include <QTimer>
|
||||
#endif
|
||||
|
||||
#include "PythonDebugger.h"
|
||||
#include "MainWindow.h"
|
||||
#include "EditorView.h"
|
||||
#include "PythonEditor.h"
|
||||
#include "BitmapFactory.h"
|
||||
#include <Base/Interpreter.h>
|
||||
#include <Base/Console.h>
|
||||
|
||||
using namespace Gui;
|
||||
|
||||
Breakpoint::Breakpoint()
|
||||
{
|
||||
}
|
||||
|
||||
Breakpoint::Breakpoint(const Breakpoint& rBp)
|
||||
{
|
||||
setFilename(rBp.filename());
|
||||
for (std::set<int>::const_iterator it = rBp._linenums.begin(); it != rBp._linenums.end(); ++it)
|
||||
_linenums.insert(*it);
|
||||
}
|
||||
|
||||
Breakpoint& Breakpoint::operator= (const Breakpoint& rBp)
|
||||
{
|
||||
if (this == &rBp)
|
||||
return *this;
|
||||
setFilename(rBp.filename());
|
||||
_linenums.clear();
|
||||
for (std::set<int>::const_iterator it = rBp._linenums.begin(); it != rBp._linenums.end(); ++it)
|
||||
_linenums.insert(*it);
|
||||
return *this;
|
||||
}
|
||||
|
||||
Breakpoint::~Breakpoint()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void Breakpoint::setFilename(const QString& fn)
|
||||
{
|
||||
_filename = fn;
|
||||
}
|
||||
|
||||
void Breakpoint::addLine(int line)
|
||||
{
|
||||
_linenums.insert(line);
|
||||
}
|
||||
|
||||
void Breakpoint::removeLine(int line)
|
||||
{
|
||||
_linenums.erase(line);
|
||||
}
|
||||
|
||||
bool Breakpoint::checkLine(int line)
|
||||
{
|
||||
return (_linenums.find(line) != _linenums.end());
|
||||
}
|
||||
|
||||
int Breakpoint::lineIndex(int ind)const
|
||||
{
|
||||
int i = 0;
|
||||
for (std::set<int>::const_iterator it = _linenums.begin(); it != _linenums.end(); ++it)
|
||||
{
|
||||
if (ind == i++)
|
||||
return *it;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
// -----------------------------------------------------
|
||||
|
||||
void PythonDebugModule::init_module(void)
|
||||
{
|
||||
PythonDebugStdout::init_type();
|
||||
PythonDebugStderr::init_type();
|
||||
PythonDebugExcept::init_type();
|
||||
static PythonDebugModule* mod = new PythonDebugModule();
|
||||
Q_UNUSED(mod);
|
||||
}
|
||||
|
||||
PythonDebugModule::PythonDebugModule()
|
||||
: Py::ExtensionModule<PythonDebugModule>("FreeCADDbg")
|
||||
{
|
||||
add_varargs_method("getFunctionCallCount", &PythonDebugModule::getFunctionCallCount,
|
||||
"Get the total number of function calls executed and the number executed since the last call to this function.");
|
||||
add_varargs_method("getExceptionCount", &PythonDebugModule::getExceptionCount,
|
||||
"Get the total number of exceptions and the number executed since the last call to this function.");
|
||||
add_varargs_method("getLineCount", &PythonDebugModule::getLineCount,
|
||||
"Get the total number of lines executed and the number executed since the last call to this function.");
|
||||
add_varargs_method("getFunctionReturnCount", &PythonDebugModule::getFunctionReturnCount,
|
||||
"Get the total number of function returns executed and the number executed since the last call to this function.");
|
||||
|
||||
initialize( "The FreeCAD Python debug module" );
|
||||
|
||||
Py::Dict d(moduleDictionary());
|
||||
Py::Object out(Py::asObject(new PythonDebugStdout()));
|
||||
d["StdOut"] = out;
|
||||
Py::Object err(Py::asObject(new PythonDebugStderr()));
|
||||
d["StdErr"] = err;
|
||||
}
|
||||
|
||||
PythonDebugModule::~PythonDebugModule()
|
||||
{
|
||||
}
|
||||
|
||||
Py::Object PythonDebugModule::getFunctionCallCount(const Py::Tuple &a)
|
||||
{
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
Py::Object PythonDebugModule::getExceptionCount(const Py::Tuple &a)
|
||||
{
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
Py::Object PythonDebugModule::getLineCount(const Py::Tuple &a)
|
||||
{
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
Py::Object PythonDebugModule::getFunctionReturnCount(const Py::Tuple &a)
|
||||
{
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------
|
||||
|
||||
void PythonDebugStdout::init_type()
|
||||
{
|
||||
behaviors().name("PythonDebugStdout");
|
||||
behaviors().doc("Redirection of stdout to FreeCAD's Python debugger window");
|
||||
// you must have overwritten the virtual functions
|
||||
behaviors().supportRepr();
|
||||
add_varargs_method("write",&PythonDebugStdout::write,"write to stdout");
|
||||
add_varargs_method("flush",&PythonDebugStdout::flush,"flush the output");
|
||||
}
|
||||
|
||||
PythonDebugStdout::PythonDebugStdout()
|
||||
{
|
||||
}
|
||||
|
||||
PythonDebugStdout::~PythonDebugStdout()
|
||||
{
|
||||
}
|
||||
|
||||
Py::Object PythonDebugStdout::repr()
|
||||
{
|
||||
std::string s;
|
||||
std::ostringstream s_out;
|
||||
s_out << "PythonDebugStdout";
|
||||
return Py::String(s_out.str());
|
||||
}
|
||||
|
||||
Py::Object PythonDebugStdout::write(const Py::Tuple& args)
|
||||
{
|
||||
char *msg;
|
||||
PyObject* pObj;
|
||||
//args contains a single parameter which is the string to write.
|
||||
if (!PyArg_ParseTuple(args.ptr(), "Os:OutputString", &pObj, &msg))
|
||||
throw Py::Exception();
|
||||
|
||||
if (strlen(msg) > 0)
|
||||
{
|
||||
//send it to our stdout
|
||||
printf("%s\n",msg);
|
||||
|
||||
//send it to the debugger as well
|
||||
//g_DebugSocket.SendMessage(eMSG_OUTPUT, msg);
|
||||
}
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
Py::Object PythonDebugStdout::flush(const Py::Tuple&)
|
||||
{
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------
|
||||
|
||||
void PythonDebugStderr::init_type()
|
||||
{
|
||||
behaviors().name("PythonDebugStderr");
|
||||
behaviors().doc("Redirection of stderr to FreeCAD's Python debugger window");
|
||||
// you must have overwritten the virtual functions
|
||||
behaviors().supportRepr();
|
||||
add_varargs_method("write",&PythonDebugStderr::write,"write to stderr");
|
||||
}
|
||||
|
||||
PythonDebugStderr::PythonDebugStderr()
|
||||
{
|
||||
}
|
||||
|
||||
PythonDebugStderr::~PythonDebugStderr()
|
||||
{
|
||||
}
|
||||
|
||||
Py::Object PythonDebugStderr::repr()
|
||||
{
|
||||
std::string s;
|
||||
std::ostringstream s_out;
|
||||
s_out << "PythonDebugStderr";
|
||||
return Py::String(s_out.str());
|
||||
}
|
||||
|
||||
Py::Object PythonDebugStderr::write(const Py::Tuple& args)
|
||||
{
|
||||
char *msg;
|
||||
PyObject* pObj;
|
||||
//args contains a single parameter which is the string to write.
|
||||
if (!PyArg_ParseTuple(args.ptr(), "Os:OutputDebugString", &pObj, &msg))
|
||||
throw Py::Exception();
|
||||
|
||||
if (strlen(msg) > 0)
|
||||
{
|
||||
//send the message to our own stderr
|
||||
//dprintf(msg);
|
||||
|
||||
//send it to the debugger as well
|
||||
//g_DebugSocket.SendMessage(eMSG_TRACE, msg);
|
||||
}
|
||||
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------
|
||||
|
||||
void PythonDebugExcept::init_type()
|
||||
{
|
||||
behaviors().name("PythonDebugExcept");
|
||||
behaviors().doc("Custom exception handler");
|
||||
// you must have overwritten the virtual functions
|
||||
behaviors().supportRepr();
|
||||
add_varargs_method("fc_excepthook",&PythonDebugExcept::excepthook,"Custom exception handler");
|
||||
}
|
||||
|
||||
PythonDebugExcept::PythonDebugExcept()
|
||||
{
|
||||
}
|
||||
|
||||
PythonDebugExcept::~PythonDebugExcept()
|
||||
{
|
||||
}
|
||||
|
||||
Py::Object PythonDebugExcept::repr()
|
||||
{
|
||||
std::string s;
|
||||
std::ostringstream s_out;
|
||||
s_out << "PythonDebugExcept";
|
||||
return Py::String(s_out.str());
|
||||
}
|
||||
|
||||
Py::Object PythonDebugExcept::excepthook(const Py::Tuple& args)
|
||||
{
|
||||
PyObject *exc, *value, *tb;
|
||||
if (!PyArg_UnpackTuple(args.ptr(), "excepthook", 3, 3, &exc, &value, &tb))
|
||||
throw Py::Exception();
|
||||
|
||||
PyErr_NormalizeException(&exc, &value, &tb);
|
||||
|
||||
PyErr_Display(exc, value, tb);
|
||||
/*
|
||||
if (eEXCEPTMODE_IGNORE != g_eExceptionMode)
|
||||
{
|
||||
assert(tb);
|
||||
|
||||
if (tb && (tb != Py_None))
|
||||
{
|
||||
//get the pointer to the frame held by the bottom traceback object - this
|
||||
//should be where the exception occurred.
|
||||
tracebackobject* pTb = (tracebackobject*)tb;
|
||||
while (pTb->tb_next != NULL)
|
||||
{
|
||||
pTb = pTb->tb_next;
|
||||
}
|
||||
PyFrameObject* frame = (PyFrameObject*)PyObject_GetAttr((PyObject*)pTb, PyString_FromString("tb_frame"));
|
||||
EnterBreakState(frame, (PyObject*)pTb);
|
||||
}
|
||||
}*/
|
||||
|
||||
return Py::None();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------
|
||||
|
||||
namespace Gui {
|
||||
class PythonDebuggerPy : public Py::PythonExtension<PythonDebuggerPy>
|
||||
{
|
||||
public:
|
||||
PythonDebuggerPy(PythonDebugger* d) : dbg(d), depth(0) { }
|
||||
~PythonDebuggerPy() {}
|
||||
PythonDebugger* dbg;
|
||||
int depth;
|
||||
};
|
||||
|
||||
class RunningState
|
||||
{
|
||||
public:
|
||||
RunningState(bool& s) : state(s)
|
||||
{ state = true; }
|
||||
~RunningState()
|
||||
{ state = false; }
|
||||
private:
|
||||
bool& state;
|
||||
};
|
||||
|
||||
struct PythonDebuggerP {
|
||||
PyObject* out_o;
|
||||
PyObject* err_o;
|
||||
PyObject* exc_o;
|
||||
PyObject* out_n;
|
||||
PyObject* err_n;
|
||||
PyObject* exc_n;
|
||||
bool init, trystop, running;
|
||||
QEventLoop loop;
|
||||
PyObject* pydbg;
|
||||
std::vector<Breakpoint> bps;
|
||||
|
||||
PythonDebuggerP(PythonDebugger* that) :
|
||||
init(false), trystop(false), running(false)
|
||||
{
|
||||
Base::PyGILStateLocker lock;
|
||||
out_n = new PythonDebugStdout();
|
||||
err_n = new PythonDebugStderr();
|
||||
PythonDebugExcept* err = new PythonDebugExcept();
|
||||
Py::Object func = err->getattr("fc_excepthook");
|
||||
exc_n = Py::new_reference_to(func);
|
||||
Py_DECREF(err);
|
||||
pydbg = new PythonDebuggerPy(that);
|
||||
}
|
||||
~PythonDebuggerP()
|
||||
{
|
||||
Py_DECREF(out_n);
|
||||
Py_DECREF(err_n);
|
||||
Py_DECREF(exc_n);
|
||||
Py_DECREF(pydbg);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
PythonDebugger::PythonDebugger()
|
||||
: d(new PythonDebuggerP(this))
|
||||
{
|
||||
}
|
||||
|
||||
PythonDebugger::~PythonDebugger()
|
||||
{
|
||||
delete d;
|
||||
}
|
||||
|
||||
Breakpoint PythonDebugger::getBreakpoint(const QString& fn) const
|
||||
{
|
||||
for (std::vector<Breakpoint>::const_iterator it = d->bps.begin(); it != d->bps.end(); ++it) {
|
||||
if (fn == it->filename()) {
|
||||
return *it;
|
||||
}
|
||||
}
|
||||
|
||||
return Breakpoint();
|
||||
}
|
||||
|
||||
bool PythonDebugger::toggleBreakpoint(int line, const QString& fn)
|
||||
{
|
||||
for (std::vector<Breakpoint>::iterator it = d->bps.begin(); it != d->bps.end(); ++it) {
|
||||
if (fn == it->filename()) {
|
||||
if (it->checkLine(line)) {
|
||||
it->removeLine(line);
|
||||
return false;
|
||||
}
|
||||
else {
|
||||
it->addLine(line);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Breakpoint bp;
|
||||
bp.setFilename(fn);
|
||||
bp.addLine(line);
|
||||
d->bps.push_back(bp);
|
||||
return true;
|
||||
}
|
||||
|
||||
void PythonDebugger::runFile(const QString& fn)
|
||||
{
|
||||
try {
|
||||
RunningState state(d->running);
|
||||
Base::Interpreter().runFile((const char*)fn.toUtf8());
|
||||
}
|
||||
catch (const Base::PyException&) {
|
||||
}
|
||||
catch (...) {
|
||||
Base::Console().Warning("Unknown exception thrown during macro debugging\n");
|
||||
}
|
||||
}
|
||||
|
||||
bool PythonDebugger::isRunning() const
|
||||
{
|
||||
return d->running;
|
||||
}
|
||||
|
||||
bool PythonDebugger::start()
|
||||
{
|
||||
if (d->init)
|
||||
return false;
|
||||
d->init = true;
|
||||
d->trystop = false;
|
||||
Base::PyGILStateLocker lock;
|
||||
d->out_o = PySys_GetObject("stdout");
|
||||
d->err_o = PySys_GetObject("stderr");
|
||||
d->exc_o = PySys_GetObject("excepthook");
|
||||
|
||||
PySys_SetObject("stdout", d->out_n);
|
||||
PySys_SetObject("stderr", d->err_n);
|
||||
PySys_SetObject("excepthook", d->exc_o);
|
||||
|
||||
PyEval_SetTrace(tracer_callback, d->pydbg);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PythonDebugger::stop()
|
||||
{
|
||||
if (!d->init)
|
||||
return false;
|
||||
Base::PyGILStateLocker lock;
|
||||
PyEval_SetTrace(NULL, NULL);
|
||||
PySys_SetObject("stdout", d->out_o);
|
||||
PySys_SetObject("stderr", d->err_o);
|
||||
PySys_SetObject("excepthook", d->exc_o);
|
||||
d->init = false;
|
||||
return true;
|
||||
}
|
||||
|
||||
void PythonDebugger::tryStop()
|
||||
{
|
||||
d->trystop = true;
|
||||
signalNextStep();
|
||||
}
|
||||
|
||||
void PythonDebugger::stepOver()
|
||||
{
|
||||
signalNextStep();
|
||||
}
|
||||
|
||||
void PythonDebugger::showDebugMarker(const QString& fn, int line)
|
||||
{
|
||||
PythonEditorView* edit = 0;
|
||||
QList<QWidget*> mdis = getMainWindow()->windows();
|
||||
for (QList<QWidget*>::iterator it = mdis.begin(); it != mdis.end(); ++it) {
|
||||
edit = qobject_cast<PythonEditorView*>(*it);
|
||||
if (edit && edit->fileName() == fn)
|
||||
break;
|
||||
}
|
||||
|
||||
if (!edit) {
|
||||
PythonEditor* editor = new PythonEditor();
|
||||
editor->setWindowIcon(Gui::BitmapFactory().pixmap("python_small"));
|
||||
edit = new PythonEditorView(editor, getMainWindow());
|
||||
edit->open(fn);
|
||||
edit->resize(400, 300);
|
||||
getMainWindow()->addWindow(edit);
|
||||
}
|
||||
|
||||
getMainWindow()->setActiveWindow(edit);
|
||||
edit->showDebugMarker(line);
|
||||
}
|
||||
|
||||
void PythonDebugger::hideDebugMarker(const QString& fn)
|
||||
{
|
||||
PythonEditorView* edit = 0;
|
||||
QList<QWidget*> mdis = getMainWindow()->windows();
|
||||
for (QList<QWidget*>::iterator it = mdis.begin(); it != mdis.end(); ++it) {
|
||||
edit = qobject_cast<PythonEditorView*>(*it);
|
||||
if (edit && edit->fileName() == fn) {
|
||||
edit->hideDebugMarker();
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// http://www.koders.com/cpp/fidBA6CD8A0FE5F41F1464D74733D9A711DA257D20B.aspx?s=PyEval_SetTrace
|
||||
// http://code.google.com/p/idapython/source/browse/trunk/python.cpp
|
||||
// http://www.koders.com/cpp/fid191F7B13CF73133935A7A2E18B7BF43ACC6D1784.aspx?s=PyEval_SetTrace
|
||||
// http://stuff.mit.edu/afs/sipb/project/python/src/python2.2-2.2.2/Modules/_hotshot.c
|
||||
int PythonDebugger::tracer_callback(PyObject *obj, PyFrameObject *frame, int what, PyObject *arg)
|
||||
{
|
||||
PythonDebuggerPy* self = static_cast<PythonDebuggerPy*>(obj);
|
||||
PythonDebugger* dbg = self->dbg;
|
||||
if (dbg->d->trystop)
|
||||
PyErr_SetInterrupt();
|
||||
QCoreApplication::processEvents();
|
||||
//int no;
|
||||
|
||||
//no = frame->f_tstate->recursion_depth;
|
||||
//char* name = PyString_AsString(frame->f_code->co_name);
|
||||
QString file = QString::fromUtf8(PyString_AsString(frame->f_code->co_filename));
|
||||
switch (what) {
|
||||
case PyTrace_CALL:
|
||||
self->depth++;
|
||||
return 0;
|
||||
case PyTrace_RETURN:
|
||||
if (self->depth > 0)
|
||||
self->depth--;
|
||||
return 0;
|
||||
case PyTrace_LINE:
|
||||
{
|
||||
//PyObject *str;
|
||||
//str = PyObject_Str(frame->f_code->co_filename);
|
||||
//no = frame->f_lineno;
|
||||
int line = PyCode_Addr2Line(frame->f_code, frame->f_lasti);
|
||||
//if (str) {
|
||||
// Base::Console().Message("PROFILING: %s:%d\n", PyString_AsString(str), frame->f_lineno);
|
||||
// Py_DECREF(str);
|
||||
//}
|
||||
// For testing only
|
||||
if (!dbg->d->trystop) {
|
||||
Breakpoint bp = dbg->getBreakpoint(file);
|
||||
if (bp.checkLine(line)) {
|
||||
dbg->showDebugMarker(file, line);
|
||||
QEventLoop loop;
|
||||
QObject::connect(dbg, SIGNAL(signalNextStep()), &loop, SLOT(quit()));
|
||||
loop.exec();
|
||||
dbg->hideDebugMarker(file);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
case PyTrace_EXCEPTION:
|
||||
return 0;
|
||||
case PyTrace_C_CALL:
|
||||
return 0;
|
||||
case PyTrace_C_EXCEPTION:
|
||||
return 0;
|
||||
case PyTrace_C_RETURN:
|
||||
return 0;
|
||||
default:
|
||||
/* ignore PyTrace_EXCEPTION */
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
#include "moc_PythonDebugger.cpp"
|
||||
Reference in New Issue
Block a user