2019-10-22 15:51:56 +00:00
|
|
|
#include <QShortcut>
|
|
|
|
#include "ThreadsWidget.h"
|
|
|
|
#include "ui_ThreadsWidget.h"
|
|
|
|
#include "common/JsonModel.h"
|
|
|
|
#include "QuickFilterView.h"
|
|
|
|
#include <r_debug.h>
|
|
|
|
|
|
|
|
#include "core/MainWindow.h"
|
|
|
|
|
|
|
|
#define DEBUGGED_PID (-1)
|
|
|
|
|
|
|
|
enum ColumnIndex {
|
2019-12-06 15:07:05 +00:00
|
|
|
COLUMN_PID = 0,
|
2019-10-22 15:51:56 +00:00
|
|
|
COLUMN_STATUS,
|
|
|
|
COLUMN_PATH,
|
|
|
|
};
|
|
|
|
|
|
|
|
ThreadsWidget::ThreadsWidget(MainWindow *main, QAction *action) :
|
|
|
|
CutterDockWidget(main, action),
|
|
|
|
ui(new Ui::ThreadsWidget)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
|
|
|
|
// Setup threads model
|
2019-12-06 15:07:05 +00:00
|
|
|
modelThreads = new QStandardItemModel(1, 3, this);
|
2019-10-22 15:51:56 +00:00
|
|
|
modelThreads->setHorizontalHeaderItem(COLUMN_PID, new QStandardItem(tr("PID")));
|
|
|
|
modelThreads->setHorizontalHeaderItem(COLUMN_STATUS, new QStandardItem(tr("Status")));
|
|
|
|
modelThreads->setHorizontalHeaderItem(COLUMN_PATH, new QStandardItem(tr("Path")));
|
|
|
|
ui->viewThreads->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft | Qt::AlignVCenter);
|
2019-12-06 09:15:27 +00:00
|
|
|
ui->viewThreads->verticalHeader()->setVisible(false);
|
2019-10-22 15:51:56 +00:00
|
|
|
ui->viewThreads->setFont(Config()->getFont());
|
|
|
|
|
|
|
|
modelFilter = new ThreadsFilterModel(this);
|
|
|
|
modelFilter->setSourceModel(modelThreads);
|
|
|
|
ui->viewThreads->setModel(modelFilter);
|
|
|
|
|
|
|
|
// CTRL+F switches to the filter view and opens it in case it's hidden
|
|
|
|
QShortcut *searchShortcut = new QShortcut(QKeySequence::Find, this);
|
|
|
|
connect(searchShortcut, &QShortcut::activated, ui->quickFilterView, &QuickFilterView::showFilter);
|
|
|
|
searchShortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
|
|
|
|
// ESC switches back to the thread table and clears the buffer
|
|
|
|
QShortcut *clearShortcut = new QShortcut(QKeySequence(Qt::Key_Escape), this);
|
|
|
|
connect(clearShortcut, &QShortcut::activated, this, [this]() {
|
|
|
|
ui->quickFilterView->clearFilter();
|
|
|
|
ui->viewThreads->setFocus();
|
|
|
|
});
|
|
|
|
clearShortcut->setContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
|
|
|
|
refreshDeferrer = createRefreshDeferrer([this]() {
|
|
|
|
updateContents();
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(ui->quickFilterView, &QuickFilterView::filterTextChanged, modelFilter,
|
|
|
|
&ThreadsFilterModel::setFilterWildcard);
|
|
|
|
connect(Core(), &CutterCore::refreshAll, this, &ThreadsWidget::updateContents);
|
2019-12-20 12:15:04 +00:00
|
|
|
connect(Core(), &CutterCore::registersChanged, this, &ThreadsWidget::updateContents);
|
2019-12-09 18:43:42 +00:00
|
|
|
connect(Core(), &CutterCore::debugTaskStateChanged, this, &ThreadsWidget::updateContents);
|
|
|
|
// Seek doesn't necessarily change when switching threads/processes
|
|
|
|
connect(Core(), &CutterCore::switchedThread, this, &ThreadsWidget::updateContents);
|
|
|
|
connect(Core(), &CutterCore::switchedProcess, this, &ThreadsWidget::updateContents);
|
2019-10-22 15:51:56 +00:00
|
|
|
connect(Config(), &Configuration::fontsUpdated, this, &ThreadsWidget::fontsUpdatedSlot);
|
|
|
|
connect(ui->viewThreads, &QTableView::activated, this, &ThreadsWidget::onActivated);
|
|
|
|
}
|
|
|
|
|
|
|
|
ThreadsWidget::~ThreadsWidget() {}
|
|
|
|
|
|
|
|
void ThreadsWidget::updateContents()
|
|
|
|
{
|
|
|
|
if (!refreshDeferrer->attemptRefresh(nullptr)) {
|
|
|
|
return;
|
|
|
|
}
|
2019-10-31 09:50:08 +00:00
|
|
|
|
2019-12-20 12:20:27 +00:00
|
|
|
if (!Core()->currentlyDebugging) {
|
2019-12-09 18:43:42 +00:00
|
|
|
// Remove rows from the previous debugging session
|
|
|
|
modelThreads->removeRows(0, modelThreads->rowCount());
|
2019-12-20 12:20:27 +00:00
|
|
|
return;
|
2019-12-09 18:43:42 +00:00
|
|
|
}
|
2019-10-31 09:50:08 +00:00
|
|
|
|
2019-12-20 12:20:27 +00:00
|
|
|
if (Core()->isDebugTaskInProgress()) {
|
2019-12-09 18:43:42 +00:00
|
|
|
ui->viewThreads->setDisabled(true);
|
|
|
|
} else {
|
2019-12-20 12:20:27 +00:00
|
|
|
setThreadsGrid();
|
2019-12-09 18:43:42 +00:00
|
|
|
ui->viewThreads->setDisabled(false);
|
2019-10-31 09:50:08 +00:00
|
|
|
}
|
2019-10-22 15:51:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QString ThreadsWidget::translateStatus(QString status)
|
|
|
|
{
|
|
|
|
switch (status.toStdString().c_str()[0]) {
|
|
|
|
case R_DBG_PROC_STOP:
|
|
|
|
return "Stopped";
|
|
|
|
case R_DBG_PROC_RUN:
|
|
|
|
return "Running";
|
|
|
|
case R_DBG_PROC_SLEEP:
|
|
|
|
return "Sleeping";
|
|
|
|
case R_DBG_PROC_ZOMBIE:
|
|
|
|
return "Zombie";
|
|
|
|
case R_DBG_PROC_DEAD:
|
|
|
|
return "Dead";
|
|
|
|
case R_DBG_PROC_RAISED:
|
|
|
|
return "Raised event";
|
|
|
|
default:
|
|
|
|
return "Unknown status";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadsWidget::setThreadsGrid()
|
|
|
|
{
|
|
|
|
QJsonArray threadsValues = Core()->getProcessThreads(DEBUGGED_PID).array();
|
|
|
|
int i = 0;
|
2019-12-06 15:07:05 +00:00
|
|
|
QFont font;
|
|
|
|
|
2019-10-22 15:51:56 +00:00
|
|
|
for (const QJsonValue &value : threadsValues) {
|
|
|
|
QJsonObject threadsItem = value.toObject();
|
|
|
|
int pid = threadsItem["pid"].toVariant().toInt();
|
|
|
|
QString status = translateStatus(threadsItem["status"].toString());
|
|
|
|
QString path = threadsItem["path"].toString();
|
|
|
|
bool current = threadsItem["current"].toBool();
|
2019-12-06 15:07:05 +00:00
|
|
|
// Use bold font to highlight active thread
|
|
|
|
font.setBold(current);
|
2019-10-22 15:51:56 +00:00
|
|
|
QStandardItem *rowPid = new QStandardItem(QString::number(pid));
|
2019-12-06 15:07:05 +00:00
|
|
|
rowPid->setFont(font);
|
2019-10-22 15:51:56 +00:00
|
|
|
QStandardItem *rowStatus = new QStandardItem(status);
|
2019-12-06 15:07:05 +00:00
|
|
|
rowStatus->setFont(font);
|
2019-10-22 15:51:56 +00:00
|
|
|
QStandardItem *rowPath = new QStandardItem(path);
|
2019-12-06 15:07:05 +00:00
|
|
|
rowPath->setFont(font);
|
2019-10-22 15:51:56 +00:00
|
|
|
modelThreads->setItem(i, COLUMN_PID, rowPid);
|
|
|
|
modelThreads->setItem(i, COLUMN_STATUS, rowStatus);
|
|
|
|
modelThreads->setItem(i, COLUMN_PATH, rowPath);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove irrelevant old rows
|
|
|
|
if (modelThreads->rowCount() > i) {
|
|
|
|
modelThreads->removeRows(i, modelThreads->rowCount() - i);
|
|
|
|
}
|
|
|
|
|
|
|
|
modelFilter->setSourceModel(modelThreads);
|
|
|
|
ui->viewThreads->resizeColumnsToContents();;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadsWidget::fontsUpdatedSlot()
|
|
|
|
{
|
|
|
|
ui->viewThreads->setFont(Config()->getFont());
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadsWidget::onActivated(const QModelIndex &index)
|
|
|
|
{
|
|
|
|
if (!index.isValid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
int tid = modelFilter->data(index.sibling(index.row(), COLUMN_PID)).toInt();
|
|
|
|
|
|
|
|
// Verify that the selected tid is still in the threads list since dpt= will
|
|
|
|
// attach to any given id. If it isn't found simply update the UI.
|
|
|
|
QJsonArray threadsValues = Core()->getProcessThreads(DEBUGGED_PID).array();
|
|
|
|
for (QJsonValue value : threadsValues) {
|
|
|
|
if (tid == value.toObject()["pid"].toInt()) {
|
|
|
|
Core()->setCurrentDebugThread(tid);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
updateContents();
|
|
|
|
}
|
|
|
|
|
|
|
|
ThreadsFilterModel::ThreadsFilterModel(QObject *parent)
|
|
|
|
: QSortFilterProxyModel(parent)
|
|
|
|
{
|
|
|
|
setFilterCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ThreadsFilterModel::filterAcceptsRow(int row, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
// All columns are checked for a match
|
2019-12-06 15:07:05 +00:00
|
|
|
for (int i = COLUMN_PID; i <= COLUMN_PATH; ++i) {
|
2019-10-22 15:51:56 +00:00
|
|
|
QModelIndex index = sourceModel()->index(row, i, parent);
|
|
|
|
if (sourceModel()->data(index).toString().contains(filterRegExp())) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|