2016-10-06 13:36:02 +02:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
**
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
**
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
**
|
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "pycdbextmodule.h"
|
|
|
|
|
|
|
|
#include "extensioncontext.h"
|
|
|
|
#include "symbolgroup.h"
|
2016-11-03 15:34:37 +01:00
|
|
|
#include "symbolgroupvalue.h"
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
#include "pyfield.h"
|
|
|
|
#include "pystdoutredirect.h"
|
|
|
|
#include "pytype.h"
|
|
|
|
#include "pyvalue.h"
|
|
|
|
|
|
|
|
#include <Python.h>
|
|
|
|
#include <structmember.h>
|
|
|
|
|
|
|
|
// cdbext python module
|
|
|
|
static PyObject *cdbext_parseAndEvaluate(PyObject *, PyObject *args) // -> Value
|
|
|
|
{
|
|
|
|
char *expr;
|
|
|
|
if (!PyArg_ParseTuple(args, "s", &expr))
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
CIDebugControl *control = ExtensionCommandContext::instance()->control();
|
|
|
|
control->SetExpressionSyntax(DEBUG_EXPR_CPLUSPLUS);
|
|
|
|
DEBUG_VALUE value;
|
|
|
|
if (FAILED(control->Evaluate(expr, DEBUG_VALUE_INT64, &value, NULL)))
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
return Py_BuildValue("K", value.I64);
|
|
|
|
}
|
|
|
|
|
|
|
|
static PyObject *cdbext_lookupType(PyObject *, PyObject *args) // -> Type
|
|
|
|
{
|
|
|
|
char *type;
|
|
|
|
if (!PyArg_ParseTuple(args, "s", &type))
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
return lookupType(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
static PyObject *cdbext_listOfLocals(PyObject *, PyObject *) // -> [ Value ]
|
|
|
|
{
|
|
|
|
ExtensionCommandContext *extCmdCtx = ExtensionCommandContext::instance();
|
|
|
|
ULONG frame;
|
|
|
|
if (FAILED(extCmdCtx->symbols()->GetCurrentScopeFrameIndex(&frame)))
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
std::string errorMessage;
|
|
|
|
LocalsSymbolGroup *sg = ExtensionContext::instance().symbolGroup(
|
|
|
|
extCmdCtx->symbols(), extCmdCtx->threadId(), int(frame), &errorMessage);
|
|
|
|
if (!sg)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
const auto children = sg->root()->children();
|
|
|
|
auto locals = PyList_New(0);
|
|
|
|
for (AbstractSymbolGroupNode *abstractChild : children) {
|
2016-11-03 15:32:27 +01:00
|
|
|
if (SymbolGroupNode* child = abstractChild->asSymbolGroupNode())
|
|
|
|
PyList_Append(locals, createValue(child->index(), sg->debugSymbolGroup()));
|
2016-10-06 13:36:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return locals;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PyObject *cdbext_pointerSize(PyObject *, PyObject *)
|
|
|
|
{
|
|
|
|
HRESULT isPointer64Bit = ExtensionCommandContext::instance()->control()->IsPointer64Bit();
|
|
|
|
return Py_BuildValue("i", isPointer64Bit == S_OK ? 8 : 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static PyObject *cdbext_readRawMemory(PyObject *, PyObject *args)
|
|
|
|
{
|
|
|
|
ULONG64 address = 0;
|
|
|
|
ULONG size = 0;
|
|
|
|
if (!PyArg_ParseTuple(args, "Kk", &address, &size))
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
char *buffer = new char[size];
|
|
|
|
|
|
|
|
CIDebugDataSpaces *data = ExtensionCommandContext::instance()->dataSpaces();
|
|
|
|
ULONG bytesWritten = 0;
|
|
|
|
HRESULT hr = data->ReadVirtual(address, buffer, size, &bytesWritten);
|
|
|
|
if (FAILED(hr))
|
|
|
|
bytesWritten = 0;
|
|
|
|
PyObject *ret = Py_BuildValue("y#", buffer, bytesWritten);
|
|
|
|
delete[] buffer;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-11-03 15:34:37 +01:00
|
|
|
static PyObject *cdbext_createValue(PyObject *, PyObject *args)
|
|
|
|
{
|
|
|
|
ULONG64 address = 0;
|
|
|
|
Type *type = 0;
|
|
|
|
if (!PyArg_ParseTuple(args, "KO", &address, &type))
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
|
|
|
|
if (debugPyCdbextModule) {
|
|
|
|
DebugPrint() << "Create Value address: 0x" << std::hex << address
|
|
|
|
<< " type name: " << getTypeName(type->m_module, type->m_typeId);
|
|
|
|
}
|
|
|
|
|
|
|
|
ExtensionCommandContext *extCmdCtx = ExtensionCommandContext::instance();
|
|
|
|
CIDebugSymbols *symbols = extCmdCtx->symbols();
|
|
|
|
|
|
|
|
ULONG frame;
|
|
|
|
if (FAILED(symbols->GetCurrentScopeFrameIndex(&frame)))
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
|
|
|
|
std::string errorMessage;
|
|
|
|
LocalsSymbolGroup *lsg = ExtensionContext::instance().symbolGroup(
|
|
|
|
symbols, extCmdCtx->threadId(), int(frame), &errorMessage);
|
|
|
|
if (!lsg)
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
|
|
|
|
CIDebugSymbolGroup *dsg = lsg->debugSymbolGroup();
|
|
|
|
ULONG numberOfSymbols = 0;
|
|
|
|
dsg->GetNumberSymbols(&numberOfSymbols);
|
|
|
|
ULONG index = 0;
|
|
|
|
for (;index < numberOfSymbols; ++index) {
|
|
|
|
ULONG64 offset;
|
|
|
|
dsg->GetSymbolOffset(index, &offset);
|
|
|
|
if (offset == address) {
|
|
|
|
DEBUG_SYMBOL_PARAMETERS params;
|
|
|
|
if (SUCCEEDED(dsg->GetSymbolParameters(index, 1, ¶ms))) {
|
|
|
|
if (params.TypeId == type->m_typeId && params.Module == type->m_module)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index >= numberOfSymbols) {
|
|
|
|
index = DEBUG_ANY_ID;
|
|
|
|
const std::string name = SymbolGroupValue::pointedToSymbolName(
|
|
|
|
address, getTypeName(type->m_module, type->m_typeId));
|
|
|
|
if (FAILED(dsg->AddSymbol(name.c_str(), &index)))
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
return createValue(index, dsg);
|
|
|
|
}
|
|
|
|
|
2016-10-06 13:36:02 +02:00
|
|
|
static PyMethodDef cdbextMethods[] = {
|
|
|
|
{"parseAndEvaluate", cdbext_parseAndEvaluate, METH_VARARGS,
|
|
|
|
"Returns value of expression or None if the expression can not be resolved"},
|
|
|
|
{"lookupType", cdbext_lookupType, METH_VARARGS,
|
|
|
|
"Returns type object or None if the type can not be resolved"},
|
|
|
|
{"listOfLocals", cdbext_listOfLocals, METH_NOARGS,
|
|
|
|
"Returns list of values that are currently in scope"},
|
|
|
|
{"pointerSize", cdbext_pointerSize, METH_NOARGS,
|
|
|
|
"Returns the size of a pointer"},
|
|
|
|
{"readRawMemory", cdbext_readRawMemory, METH_VARARGS,
|
|
|
|
"Read a block of data from the virtual address space"},
|
2016-11-03 15:34:37 +01:00
|
|
|
{"createValue", cdbext_createValue, METH_VARARGS,
|
|
|
|
"Creates a value with the given type at the given address"},
|
2016-10-06 13:36:02 +02:00
|
|
|
{NULL, NULL, 0,
|
|
|
|
NULL} /* Sentinel */
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct PyModuleDef cdbextModule = {
|
|
|
|
PyModuleDef_HEAD_INIT,
|
|
|
|
"cdbext", /* name of module */
|
|
|
|
"bridge to the creator cdb extension", /* module documentation */
|
|
|
|
-1, /* size of per-interpreter state of the module,
|
|
|
|
or -1 if the module keeps state in global variables. */
|
|
|
|
cdbextMethods
|
|
|
|
};
|
|
|
|
|
|
|
|
PyMODINIT_FUNC
|
|
|
|
PyInit_cdbext(void)
|
|
|
|
{
|
|
|
|
if (PyType_Ready(field_pytype()) < 0)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
if (PyType_Ready(type_pytype()) < 0)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
if (PyType_Ready(value_pytype()) < 0)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
stdoutRedirect_pytype()->tp_new = PyType_GenericNew;
|
|
|
|
if (PyType_Ready(stdoutRedirect_pytype()) < 0)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
PyObject *module = PyModule_Create(&cdbextModule);
|
|
|
|
if (module == NULL)
|
2016-11-03 15:28:12 +01:00
|
|
|
Py_RETURN_NONE;
|
2016-10-06 13:36:02 +02:00
|
|
|
|
|
|
|
Py_INCREF(field_pytype());
|
|
|
|
Py_INCREF(stdoutRedirect_pytype());
|
|
|
|
Py_INCREF(type_pytype());
|
|
|
|
Py_INCREF(value_pytype());
|
|
|
|
|
|
|
|
PyModule_AddObject(module, "Field",
|
|
|
|
reinterpret_cast<PyObject *>(field_pytype()));
|
|
|
|
PyModule_AddObject(module, "StdoutRedirect",
|
|
|
|
reinterpret_cast<PyObject *>(stdoutRedirect_pytype()));
|
|
|
|
PyModule_AddObject(module, "Type",
|
|
|
|
reinterpret_cast<PyObject *>(type_pytype()));
|
|
|
|
PyModule_AddObject(module, "Value",
|
|
|
|
reinterpret_cast<PyObject *>(value_pytype()));
|
|
|
|
|
|
|
|
return module;
|
|
|
|
}
|
|
|
|
|
|
|
|
void initCdbextPythonModule()
|
|
|
|
{
|
|
|
|
PyImport_AppendInittab("cdbext", PyInit_cdbext);
|
|
|
|
}
|
|
|
|
|
|
|
|
PyObject *pyBool(bool b)
|
|
|
|
{
|
|
|
|
if (b)
|
|
|
|
Py_RETURN_TRUE;
|
|
|
|
else
|
|
|
|
Py_RETURN_FALSE;
|
|
|
|
}
|