2021-10-15 16:39:43 +01:00
|
|
|
#include "stdafx.h"
|
|
|
|
#include "TraceView.h"
|
|
|
|
#include "..\SandMan.h"
|
|
|
|
#include "../QSbieAPI/SbieAPI.h"
|
|
|
|
#include "..\Models\TraceModel.h"
|
|
|
|
#include "..\..\MiscHelpers\Common\Common.h"
|
2021-10-30 08:46:49 +01:00
|
|
|
#include "..\..\MiscHelpers\Common\CheckList.h"
|
2021-10-15 16:39:43 +01:00
|
|
|
#include "SbieView.h"
|
|
|
|
|
|
|
|
//class CTraceFilterProxyModel : public CSortFilterProxyModel
|
|
|
|
//{
|
|
|
|
//public:
|
2022-07-29 09:24:32 +01:00
|
|
|
// CTraceFilterProxyModel(QObject* parrent = 0) : CSortFilterProxyModel(parrent)
|
2021-10-15 16:39:43 +01:00
|
|
|
// {
|
|
|
|
// m_FilterPid = 0;
|
|
|
|
// m_FilterTid = 0;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// bool filterAcceptsRow(int source_row, const QModelIndex& source_parent) const
|
|
|
|
// {
|
|
|
|
// CTraceModel* pTraceModel = (CTraceModel*)sourceModel();
|
|
|
|
//
|
|
|
|
// QModelIndex index = pTraceModel->index(source_row, 0, source_parent);
|
|
|
|
// //CTraceEntryPtr pEntry = pTraceModel->GetEntry(index);
|
|
|
|
// //if (pEntry.data() == NULL)
|
|
|
|
// {
|
|
|
|
// QVariant Id = pTraceModel->GetItemID(index);
|
|
|
|
// StrPair typeId = Split2(Id.toString(), "_");
|
|
|
|
//
|
|
|
|
// if (m_FilterPid != 0 && typeId.first == "pid") {
|
|
|
|
// if (m_FilterPid != typeId.second.toUInt())
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// if (m_FilterTid != 0 && typeId.first == "tid") {
|
|
|
|
// if (m_FilterTid != typeId.second.toUInt())
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// return CSortFilterProxyModel::filterAcceptsRow(source_row, source_parent);
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// quint32 m_FilterPid;
|
|
|
|
// quint32 m_FilterTid;
|
|
|
|
//};
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// CTraceTree
|
|
|
|
|
|
|
|
CTraceTree::CTraceTree(QWidget* parent)
|
|
|
|
: CPanelWidget<QTreeViewEx>(parent)
|
|
|
|
{
|
2022-07-29 09:24:32 +01:00
|
|
|
m_pTreeList->setAlternatingRowColors(theConf->GetBool("Options/AltRowColors", false));
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTreeList->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
|
|
|
|
|
|
m_pTraceModel = new CTraceModel();
|
|
|
|
//connect(m_pTraceModel, SIGNAL(NewBranche()), this, SLOT(UpdateFilters()));
|
|
|
|
|
|
|
|
//m_pSortProxy = new CTraceFilterProxyModel(this);
|
|
|
|
//m_pSortProxy->setSortRole(Qt::EditRole);
|
|
|
|
//m_pSortProxy->setSourceModel(m_pTraceModel);
|
|
|
|
//m_pSortProxy->setDynamicSortFilter(true);
|
|
|
|
|
|
|
|
//m_pTreeList->setModel(m_pSortProxy);
|
|
|
|
//m_pSortProxy->setView(m_pTreeList);
|
|
|
|
|
|
|
|
m_pTreeList->setModel(m_pTraceModel);
|
|
|
|
|
|
|
|
QStyle* pStyle = QStyleFactory::create("windows");
|
|
|
|
m_pTreeList->setStyle(pStyle);
|
2022-08-09 17:19:46 +01:00
|
|
|
m_pTreeList->setItemDelegate(new CTreeItemDelegate());
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTreeList->setExpandsOnDoubleClick(false);
|
|
|
|
//m_pTreeList->setSortingEnabled(true);
|
|
|
|
|
|
|
|
m_pTreeList->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(m_pTreeList, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(OnMenu(const QPoint&)));
|
|
|
|
|
|
|
|
m_pTreeList->setColumnReset(1);
|
|
|
|
//connect(m_pTreeList, SIGNAL(ResetColumns()), m_pTreeList, SLOT(OnResetColumns()));
|
|
|
|
//connect(m_pBoxTree, SIGNAL(ColumnChanged(int, bool)), this, SLOT(OnColumnsChanged()));
|
|
|
|
|
|
|
|
//m_pMainLayout->addWidget(CFinder::AddFinder(m_pTreeList, m_pSortProxy));
|
|
|
|
m_pMainLayout->addWidget(CFinder::AddFinder(m_pTreeList, this));
|
|
|
|
|
|
|
|
|
|
|
|
QByteArray Columns = theConf->GetBlob("MainWindow/TraceLog_Columns");
|
|
|
|
if (!Columns.isEmpty())
|
|
|
|
((QTreeViewEx*)GetView())->restoreState(Columns);
|
|
|
|
else
|
|
|
|
((QTreeViewEx*)GetView())->OnResetColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
CTraceTree::~CTraceTree()
|
|
|
|
{
|
|
|
|
theConf->SetBlob("MainWindow/TraceLog_Columns", GetView()->header()->saveState());
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// CMonitorList
|
|
|
|
|
|
|
|
CMonitorList::CMonitorList(QWidget* parent)
|
|
|
|
: CPanelWidget<QTreeViewEx>(parent)
|
|
|
|
{
|
2022-07-29 09:24:32 +01:00
|
|
|
m_pTreeList->setAlternatingRowColors(theConf->GetBool("Options/AltRowColors", false));
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTreeList->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
|
|
|
|
|
|
m_pMonitorModel = new CMonitorModel();
|
2022-04-16 14:01:01 +01:00
|
|
|
//connect(m_pMonitorModel, SIGNAL(NewBranche()), this, SLOT(UpdateFilters()));
|
2022-03-26 09:09:24 +00:00
|
|
|
|
2022-07-29 09:24:32 +01:00
|
|
|
m_pSortProxy = new CSortFilterProxyModel(this);
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pSortProxy->setSortRole(Qt::EditRole);
|
|
|
|
m_pSortProxy->setSourceModel(m_pMonitorModel);
|
|
|
|
m_pSortProxy->setDynamicSortFilter(true);
|
|
|
|
|
|
|
|
m_pTreeList->setModel(m_pSortProxy);
|
|
|
|
m_pSortProxy->setView(m_pTreeList);
|
|
|
|
|
|
|
|
|
|
|
|
QStyle* pStyle = QStyleFactory::create("windows");
|
|
|
|
m_pTreeList->setStyle(pStyle);
|
2022-07-20 08:35:07 +01:00
|
|
|
m_pTreeList->setItemDelegate(new CTreeItemDelegate());
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTreeList->setExpandsOnDoubleClick(false);
|
|
|
|
m_pTreeList->setSortingEnabled(true);
|
|
|
|
|
|
|
|
m_pTreeList->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(m_pTreeList, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(OnMenu(const QPoint&)));
|
|
|
|
|
|
|
|
m_pTreeList->setColumnReset(1);
|
|
|
|
//connect(m_pTreeList, SIGNAL(ResetColumns()), m_pTreeList, SLOT(OnResetColumns()));
|
|
|
|
//connect(m_pBoxTree, SIGNAL(ColumnChanged(int, bool)), this, SLOT(OnColumnsChanged()));
|
|
|
|
|
|
|
|
m_pMainLayout->addWidget(CFinder::AddFinder(m_pTreeList, m_pSortProxy));
|
|
|
|
|
|
|
|
|
|
|
|
QByteArray Columns = theConf->GetBlob("MainWindow/Monitor_Columns");
|
|
|
|
if (!Columns.isEmpty())
|
|
|
|
((QTreeViewEx*)GetView())->restoreState(Columns);
|
|
|
|
else
|
|
|
|
((QTreeViewEx*)GetView())->OnResetColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
CMonitorList::~CMonitorList()
|
|
|
|
{
|
|
|
|
theConf->SetBlob("MainWindow/Monitor_Columns", GetView()->header()->saveState());
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// CTraceView
|
|
|
|
|
2022-07-09 10:46:07 +01:00
|
|
|
CTraceView::CTraceView(bool bStandAlone, QWidget* parent) : QWidget(parent)
|
2021-10-15 16:39:43 +01:00
|
|
|
{
|
|
|
|
//m_pTreeList->setItemDelegate(theGUI->GetItemDelegate());
|
|
|
|
|
|
|
|
m_FullRefresh = true;
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_LastID = 0;
|
|
|
|
m_LastCount = 0;
|
|
|
|
m_bUpdatePending = false;
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
m_bHighLight = false;
|
|
|
|
//m_FilterCol = -1;
|
|
|
|
m_FilterPid = 0;
|
|
|
|
m_FilterTid = 0;
|
|
|
|
m_FilterStatus = 0;
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pMainLayout = new QVBoxLayout();
|
2022-09-29 17:28:48 +01:00
|
|
|
m_pMainLayout->setContentsMargins(0,0,0,0);
|
2022-03-26 09:09:24 +00:00
|
|
|
this->setLayout(m_pMainLayout);
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
m_pTraceToolBar = new QToolBar();
|
2022-03-26 09:09:24 +00:00
|
|
|
|
|
|
|
m_pMonitorMode = m_pTraceToolBar->addAction(CSandMan::GetIcon("Monitor"), tr("Monitor mode"), this, SLOT(OnSetMode()));
|
|
|
|
m_pMonitorMode->setCheckable(true);
|
|
|
|
m_pMonitorMode->setChecked(theConf->GetBool("Options/UseMonitorMode"));
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
m_pTraceTree = m_pTraceToolBar->addAction(CSandMan::GetIcon("Tree"), tr("Show as task tree"), this, SLOT(OnSetTree()));
|
|
|
|
m_pTraceTree->setCheckable(true);
|
|
|
|
m_pTraceTree->setChecked(theConf->GetBool("Options/UseLogTree"));
|
2022-03-26 09:09:24 +00:00
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
m_pTraceToolBar->addSeparator();
|
|
|
|
m_pTraceToolBar->layout()->setSpacing(3);
|
|
|
|
|
|
|
|
m_pTraceToolBar->addWidget(new QLabel(tr("PID:")));
|
|
|
|
m_pTracePid = new QComboBox();
|
|
|
|
m_pTracePid->addItem(tr("[All]"), 0);
|
|
|
|
m_pTracePid->setMinimumWidth(225);
|
|
|
|
connect(m_pTracePid, SIGNAL(currentIndexChanged(int)), this, SLOT(OnSetPidFilter()));
|
|
|
|
m_pTraceToolBar->addWidget(m_pTracePid);
|
|
|
|
|
|
|
|
m_pTraceToolBar->addWidget(new QLabel(tr("TID:")));
|
|
|
|
m_pTraceTid = new QComboBox();
|
|
|
|
m_pTraceTid->addItem(tr("[All]"), 0);
|
|
|
|
m_pTraceTid->setMinimumWidth(75);
|
|
|
|
connect(m_pTraceTid, SIGNAL(currentIndexChanged(int)), this, SLOT(OnSetTidFilter()));
|
|
|
|
m_pTraceToolBar->addWidget(m_pTraceTid);
|
|
|
|
|
|
|
|
m_pTraceToolBar->addWidget(new QLabel(tr("Type:")));
|
2021-10-30 08:46:49 +01:00
|
|
|
m_pTraceType = new QCheckList();
|
|
|
|
//m_pTraceType->addItem(tr("[All]"), 0);
|
|
|
|
m_pTraceType->setAllCheckedText(tr("[All]"));
|
|
|
|
m_pTraceType->setNoneCheckedText(tr("[All]"));
|
|
|
|
foreach(quint32 type, CTraceEntry::AllTypes())
|
|
|
|
m_pTraceType->addCheckItem(CTraceEntry::GetTypeStr(type), type, Qt::Unchecked);
|
|
|
|
m_pTraceType->setMinimumWidth(100);
|
|
|
|
connect(m_pTraceType, SIGNAL(globalCheckStateChanged(int)), this, SLOT(OnSetFilter()));
|
2021-10-15 16:39:43 +01:00
|
|
|
m_pTraceToolBar->addWidget(m_pTraceType);
|
|
|
|
|
|
|
|
m_pTraceToolBar->addWidget(new QLabel(tr("Status:")));
|
|
|
|
m_pTraceStatus = new QComboBox();
|
|
|
|
m_pTraceStatus->addItem(tr("[All]"), 0);
|
|
|
|
m_pTraceStatus->addItem(tr("Open"), 1);
|
|
|
|
m_pTraceStatus->addItem(tr("Closed"), 2);
|
|
|
|
m_pTraceStatus->addItem(tr("Trace"), 3);
|
|
|
|
m_pTraceStatus->addItem(tr("Other"), 4);
|
|
|
|
m_pTraceStatus->setMinimumWidth(75);
|
|
|
|
connect(m_pTraceStatus, SIGNAL(currentIndexChanged(int)), this, SLOT(OnSetFilter()));
|
|
|
|
m_pTraceToolBar->addWidget(m_pTraceStatus);
|
|
|
|
|
2022-07-09 10:46:07 +01:00
|
|
|
if (bStandAlone)
|
|
|
|
m_pAllBoxes = NULL;
|
|
|
|
else {
|
|
|
|
m_pAllBoxes = m_pTraceToolBar->addAction(CSandMan::GetIcon("All"), tr("Show All Boxes"), this, SLOT(OnSetFilter()));
|
|
|
|
m_pAllBoxes->setCheckable(true);
|
|
|
|
m_pAllBoxes->setChecked(theConf->GetBool("Options/UseLogTree"));
|
|
|
|
}
|
2021-10-16 17:24:16 +01:00
|
|
|
|
|
|
|
m_pTraceToolBar->addSeparator();
|
|
|
|
|
|
|
|
m_pSaveToFile = m_pTraceToolBar->addAction(CSandMan::GetIcon("Save"), tr("Save to file"), this, SLOT(SaveToFile()));
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
m_pMainLayout->setSpacing(0);
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pMainLayout->addWidget(m_pTraceToolBar);
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pView = new QWidget(this);
|
|
|
|
m_pLayout = new QStackedLayout(m_pView);
|
|
|
|
|
|
|
|
m_pTrace = new CTraceTree(this);
|
|
|
|
((CTraceModel*)m_pTrace->GetModel())->SetTree(m_pTraceTree->isChecked());
|
2022-07-09 10:46:07 +01:00
|
|
|
|
|
|
|
if (bStandAlone) {
|
|
|
|
QAction* pAction = new QAction(tr("Cleanup Trace Log"));
|
|
|
|
connect(pAction, SIGNAL(triggered()), this, SLOT(Clear()));
|
|
|
|
m_pTrace->GetMenu()->insertAction(m_pTrace->GetMenu()->actions()[0], pAction);
|
|
|
|
m_pTrace->GetMenu()->insertSeparator(m_pTrace->GetMenu()->actions()[0]);
|
|
|
|
}
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pLayout->addWidget(m_pTrace);
|
2021-10-15 16:39:43 +01:00
|
|
|
|
2022-09-29 17:28:48 +01:00
|
|
|
QObject::connect(m_pTrace, SIGNAL(FilterSet(const QRegularExpression&, bool, int)), this, SLOT(SetFilter(const QRegularExpression&, bool, int)));
|
2021-10-15 16:39:43 +01:00
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pMonitor = new CMonitorList(this);
|
|
|
|
m_pLayout->addWidget(m_pMonitor);
|
2021-10-15 16:39:43 +01:00
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pView->setLayout(m_pLayout);
|
|
|
|
m_pMainLayout->addWidget(m_pView);
|
2021-10-15 16:39:43 +01:00
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
OnSetMode();
|
2022-07-09 10:46:07 +01:00
|
|
|
|
|
|
|
m_uTimerID = startTimer(1000);
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CTraceView::~CTraceView()
|
|
|
|
{
|
2022-07-09 10:46:07 +01:00
|
|
|
killTimer(m_uTimerID);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceView::timerEvent(QTimerEvent* pEvent)
|
|
|
|
{
|
|
|
|
if (pEvent->timerId() != m_uTimerID)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Refresh();
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int CTraceView__Filter(const CTraceEntryPtr& pEntry, void* params)
|
|
|
|
{
|
|
|
|
CTraceView* This = (CTraceView*)params;
|
|
|
|
|
2021-10-30 08:46:49 +01:00
|
|
|
int Ret = 1;
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
if (This->m_pCurrentBox != NULL && This->m_pCurrentBox != pEntry->GetBoxPtr())
|
2021-10-30 08:46:49 +01:00
|
|
|
return 0;
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
if (This->m_FilterExp.isValid()) {
|
2022-03-26 09:09:24 +00:00
|
|
|
if (!pEntry->GetName().contains(This->m_FilterExp)
|
|
|
|
&& !pEntry->GetMessage().contains(This->m_FilterExp)
|
2021-11-13 08:28:32 +00:00
|
|
|
&& !pEntry->GetTypeStr().contains(This->m_FilterExp)
|
2021-10-15 16:39:43 +01:00
|
|
|
//&& !pEntry->GetStautsStr().contains(This->m_FilterExp)
|
|
|
|
&& !pEntry->GetProcessName().contains(This->m_FilterExp))
|
2021-10-30 08:46:49 +01:00
|
|
|
Ret = This->m_bHighLight ? 1 : 0;
|
|
|
|
else
|
|
|
|
Ret = This->m_bHighLight ? 2 : 1;
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (This->m_FilterPid != 0 && This->m_FilterPid != pEntry->GetProcessId())
|
2021-10-30 08:46:49 +01:00
|
|
|
return 0;
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
if (This->m_FilterTid != 0 && This->m_FilterTid != pEntry->GetThreadId())
|
2021-10-30 08:46:49 +01:00
|
|
|
return 0;
|
2021-10-15 16:39:43 +01:00
|
|
|
|
2021-10-30 08:46:49 +01:00
|
|
|
if (!This->m_FilterTypes.isEmpty() && !This->m_FilterTypes.contains(pEntry->GetType()))
|
|
|
|
return 0;
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
if (This->m_FilterStatus != 0) {
|
|
|
|
if (pEntry->IsOpen()) {
|
2021-10-30 08:46:49 +01:00
|
|
|
if(This->m_FilterStatus == 1) return Ret;
|
2021-10-15 16:39:43 +01:00
|
|
|
} else if (pEntry->IsClosed()) {
|
2021-10-30 08:46:49 +01:00
|
|
|
if (This->m_FilterStatus == 2) return Ret;
|
2021-10-15 16:39:43 +01:00
|
|
|
} else if (pEntry->IsTrace()) {
|
2021-10-30 08:46:49 +01:00
|
|
|
if(This->m_FilterStatus == 3) return Ret;
|
2021-10-15 16:39:43 +01:00
|
|
|
} else
|
2021-10-30 08:46:49 +01:00
|
|
|
if(This->m_FilterStatus == 4) return Ret;
|
|
|
|
return 0;
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
2021-10-30 08:46:49 +01:00
|
|
|
return Ret;
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceView::Refresh()
|
|
|
|
{
|
|
|
|
QList<CSandBoxPtr>Boxes;
|
2022-07-09 10:46:07 +01:00
|
|
|
if(m_pAllBoxes && !m_pAllBoxes->isChecked())
|
2021-10-15 16:39:43 +01:00
|
|
|
Boxes = theGUI->GetBoxView()->GetSelectedBoxes();
|
|
|
|
|
|
|
|
if (m_pCurrentBox != (Boxes.count() == 1 ? Boxes.first().data() : NULL)) {
|
|
|
|
m_pCurrentBox = Boxes.count() == 1 ? Boxes.first().data() : NULL;
|
|
|
|
m_FullRefresh = true;
|
|
|
|
}
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
bool bMonitorMode = m_pMonitorMode->isChecked();
|
|
|
|
|
|
|
|
if (m_FullRefresh)
|
|
|
|
{
|
|
|
|
m_LastID = 0;
|
|
|
|
m_LastCount = 0;
|
|
|
|
m_PidMap.clear();
|
|
|
|
|
|
|
|
m_pTrace->m_pTraceModel->Clear();
|
|
|
|
m_pMonitor->m_pMonitorModel->Clear();
|
2021-10-15 16:39:43 +01:00
|
|
|
m_FullRefresh = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVector<CTraceEntryPtr> ResourceLog = theAPI->GetTrace();
|
2022-03-26 09:09:24 +00:00
|
|
|
|
|
|
|
bool bUpdate = false;
|
|
|
|
|
|
|
|
int i = 0;
|
|
|
|
if (ResourceLog.count() >= m_LastCount && m_LastCount > 0)
|
2021-10-15 16:39:43 +01:00
|
|
|
{
|
2022-03-26 09:09:24 +00:00
|
|
|
i = m_LastCount - 1;
|
|
|
|
if (m_LastID == ResourceLog.at(i)->GetUID())
|
|
|
|
i++;
|
|
|
|
else
|
|
|
|
i = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == 0) {
|
|
|
|
m_PidMap.clear();
|
|
|
|
m_MonitorMap.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; i < ResourceLog.count(); i++)
|
|
|
|
{
|
|
|
|
CTraceEntryPtr pEntry = ResourceLog.at(i);
|
|
|
|
|
|
|
|
SProgInfo& Info = m_PidMap[pEntry->GetProcessId()];
|
|
|
|
if (Info.Name.isEmpty()) {
|
|
|
|
Info.Name = pEntry->GetProcessName();
|
|
|
|
bUpdate = true;
|
|
|
|
}
|
|
|
|
if (!Info.Threads.contains(pEntry->GetThreadId())) {
|
|
|
|
Info.Threads.insert(pEntry->GetThreadId());
|
|
|
|
bUpdate = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bMonitorMode)
|
|
|
|
{
|
|
|
|
// filter
|
|
|
|
if (m_pCurrentBox != NULL && m_pCurrentBox != pEntry->GetBoxPtr())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (m_FilterPid != 0 && m_FilterPid != pEntry->GetProcessId())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (m_FilterTid != 0 && m_FilterTid != pEntry->GetThreadId())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!m_FilterTypes.isEmpty() && !m_FilterTypes.contains(pEntry->GetType()))
|
|
|
|
continue;
|
|
|
|
//
|
|
|
|
|
|
|
|
CMonitorEntryPtr &pItem = m_MonitorMap[pEntry->GetName()];
|
|
|
|
if (pItem.data() == NULL)
|
|
|
|
pItem = CMonitorEntryPtr(new CMonitorEntry(pEntry->GetName(), pEntry->GetType()));
|
|
|
|
|
|
|
|
pItem->Merge(pEntry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_LastCount = ResourceLog.count();
|
|
|
|
if(m_LastCount)
|
|
|
|
m_LastID = ResourceLog.last()->GetUID();
|
|
|
|
|
|
|
|
|
|
|
|
if (bUpdate && !m_bUpdatePending)
|
|
|
|
{
|
|
|
|
m_bUpdatePending = true;
|
|
|
|
QTimer::singleShot(500, this, SLOT(UpdateFilters()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (bMonitorMode)
|
|
|
|
{
|
|
|
|
m_pMonitor->m_pMonitorModel->Sync(m_MonitorMap, this);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QList<QVariant> Added = m_pTrace->m_pTraceModel->Sync(ResourceLog, CTraceView__Filter, this);
|
|
|
|
|
|
|
|
if (m_pTrace->m_pTraceModel->IsTree())
|
|
|
|
{
|
|
|
|
QTimer::singleShot(100, this, [this, Added]() {
|
|
|
|
//CSortFilterProxyModel* pSortProxy = (CSortFilterProxyModel*)GetModel();
|
|
|
|
foreach(const QVariant ID, Added) {
|
|
|
|
// m_pTreeList->expand(pSortProxy->mapFromSource(m_pTraceModel->FindIndex(ID)));
|
|
|
|
m_pTrace->GetTree()->expand(m_pTrace->m_pTraceModel->FindIndex(ID));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceView::Clear()
|
|
|
|
{
|
|
|
|
m_pTracePid->clear();
|
|
|
|
m_pTracePid->addItem(tr("[All]"), 0);
|
|
|
|
|
|
|
|
m_pTraceTid->clear();
|
|
|
|
m_pTraceTid->addItem(tr("[All]"), 0);
|
|
|
|
|
|
|
|
theAPI->ClearTrace();
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTrace->m_pTraceModel->Clear();
|
|
|
|
m_pMonitor->m_pMonitorModel->Clear();
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
2022-04-07 19:49:13 +01:00
|
|
|
void CTraceView::AddAction(QAction* pAction)
|
|
|
|
{
|
|
|
|
m_pTrace->GetMenu()->insertAction(m_pTrace->GetMenu()->actions()[0], pAction);
|
|
|
|
m_pTrace->GetMenu()->insertSeparator(m_pTrace->GetMenu()->actions()[0]);
|
|
|
|
|
|
|
|
m_pMonitor->GetMenu()->insertAction(m_pMonitor->GetMenu()->actions()[0], pAction);
|
|
|
|
m_pMonitor->GetMenu()->insertSeparator(m_pMonitor->GetMenu()->actions()[0]);
|
|
|
|
}
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
void CTraceView::OnSetTree()
|
|
|
|
{
|
2022-03-26 09:09:24 +00:00
|
|
|
((CTraceModel*)m_pTrace->GetModel())->SetTree(m_pTraceTree->isChecked());
|
2022-09-29 17:28:48 +01:00
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
m_pTrace->m_pTraceModel->Clear();
|
2022-09-29 17:28:48 +01:00
|
|
|
Refresh();
|
|
|
|
m_pTrace->GetTree()->expandAll();
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
theConf->SetValue("Options/UseLogTree", m_pTraceTree->isChecked());
|
|
|
|
}
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
void CTraceView::OnSetMode()
|
|
|
|
{
|
|
|
|
if (m_pMonitorMode->isChecked())
|
|
|
|
m_pLayout->setCurrentIndex(1); // monitor
|
|
|
|
else
|
|
|
|
m_pLayout->setCurrentIndex(0); // trace
|
|
|
|
|
|
|
|
m_pTraceTree->setEnabled(!m_pMonitorMode->isChecked());
|
|
|
|
m_pTraceStatus->setEnabled(!m_pMonitorMode->isChecked());
|
|
|
|
|
|
|
|
m_FullRefresh = true;
|
|
|
|
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
void CTraceView::UpdateFilters()
|
|
|
|
{
|
2022-03-26 09:09:24 +00:00
|
|
|
m_bUpdatePending = false;
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
quint32 cur_pid = m_pTracePid->currentData().toUInt();
|
|
|
|
|
2022-03-26 09:09:24 +00:00
|
|
|
QMap<quint32, SProgInfo> pids = m_PidMap;
|
2022-09-29 17:28:48 +01:00
|
|
|
foreach(quint32 pid, pids.keys()) {
|
2022-03-26 09:09:24 +00:00
|
|
|
SProgInfo& Info = pids[pid];
|
2021-10-15 16:39:43 +01:00
|
|
|
|
|
|
|
if(m_pTracePid->findData(pid) == -1)
|
|
|
|
m_pTracePid->addItem(tr("%1 (%2)").arg(Info.Name).arg(pid), pid);
|
|
|
|
|
|
|
|
if (cur_pid != 0 && cur_pid != pid)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
foreach(quint32 tid, Info.Threads) {
|
|
|
|
if (m_pTraceTid->findData(tid) == -1)
|
|
|
|
m_pTraceTid->addItem(tr("%1").arg(tid), tid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-29 17:28:48 +01:00
|
|
|
void CTraceView::SetFilter(const QRegularExpression& Exp, bool bHighLight, int Col)
|
2021-10-15 16:39:43 +01:00
|
|
|
{
|
|
|
|
m_FilterExp = Exp;
|
|
|
|
m_bHighLight = bHighLight;
|
|
|
|
//m_FilterCol = Col;
|
|
|
|
|
|
|
|
m_FullRefresh = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceView::OnSetPidFilter()
|
|
|
|
{
|
|
|
|
m_FilterPid = m_pTracePid->currentData().toUInt();
|
|
|
|
m_FilterTid = 0;
|
|
|
|
//m_pSortProxy->m_FilterPid = m_pTracePid->currentData().toUInt();
|
|
|
|
//m_pSortProxy->m_FilterTid = 0;
|
|
|
|
|
|
|
|
QTimer::singleShot(100, this, [this]() {
|
|
|
|
|
|
|
|
m_pTraceTid->clear();
|
|
|
|
m_pTraceTid->addItem(tr("[All]"), 0);
|
|
|
|
|
|
|
|
UpdateFilters();
|
|
|
|
});
|
|
|
|
|
|
|
|
//m_pSortProxy->setFilterKeyColumn(m_pSortProxy->filterKeyColumn());
|
|
|
|
m_FullRefresh = true;
|
2022-03-26 09:09:24 +00:00
|
|
|
if(!m_pMonitorMode->isChecked())
|
|
|
|
m_pTrace->GetTree()->expandAll();
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceView::OnSetTidFilter()
|
|
|
|
{
|
|
|
|
m_FilterTid = m_pTraceTid->currentData().toUInt();
|
|
|
|
//m_pSortProxy->m_FilterTid = m_pTraceTid->currentData().toUInt();
|
|
|
|
|
|
|
|
//m_pSortProxy->setFilterKeyColumn(m_pSortProxy->filterKeyColumn());
|
|
|
|
m_FullRefresh = true;
|
2022-03-26 09:09:24 +00:00
|
|
|
if(!m_pMonitorMode->isChecked())
|
|
|
|
m_pTrace->GetTree()->expandAll();
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CTraceView::OnSetFilter()
|
|
|
|
{
|
2021-10-30 08:46:49 +01:00
|
|
|
m_FilterTypes.clear();
|
|
|
|
for (int i = 0; i < m_pTraceType->count(); i++) {
|
|
|
|
if (m_pTraceType->itemData(i, Qt::CheckStateRole).toInt() == Qt::Checked) {
|
|
|
|
m_FilterTypes.append(m_pTraceType->itemData(i, Qt::UserRole + 1).toUInt());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-15 16:39:43 +01:00
|
|
|
m_FilterStatus = m_pTraceStatus->currentData().toUInt();
|
|
|
|
|
|
|
|
m_FullRefresh = true;
|
2022-03-26 09:09:24 +00:00
|
|
|
if(!m_pMonitorMode->isChecked())
|
|
|
|
m_pTrace->GetTree()->expandAll();
|
2021-10-15 16:39:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-10-16 17:24:16 +01:00
|
|
|
void CTraceView::SaveToFile()
|
|
|
|
{
|
|
|
|
QString Path = QFileDialog::getSaveFileName(this, tr("Save trace log to file"), "", QString("Log files (*.log)")).replace("/", "\\");
|
2021-10-16 19:13:16 +01:00
|
|
|
if (Path.isEmpty())
|
|
|
|
return;
|
2021-10-16 17:24:16 +01:00
|
|
|
|
|
|
|
QFile File(Path);
|
|
|
|
if (!File.open(QFile::WriteOnly)) {
|
|
|
|
QMessageBox::critical(this, "Sandboxie-Plus", tr("Failed to open log file for writing"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-05-15 11:27:33 +01:00
|
|
|
if (m_pMonitorMode->isChecked())
|
2021-10-16 17:24:16 +01:00
|
|
|
{
|
2022-05-15 11:27:33 +01:00
|
|
|
QList<QStringList> Rows = m_pMonitor->DumpPanel();
|
|
|
|
foreach(const QStringList& Row, Rows)
|
|
|
|
File.write(Row.join("\t").toLatin1() + "\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QVector<CTraceEntryPtr> ResourceLog = theAPI->GetTrace();
|
|
|
|
for (int i = 0; i < ResourceLog.count(); i++)
|
|
|
|
{
|
|
|
|
CTraceEntryPtr pEntry = ResourceLog.at(i);
|
|
|
|
|
|
|
|
//int iFilter = CTraceView__Filter(pEntry, this);
|
|
|
|
//if (!iFilter)
|
|
|
|
// continue;
|
|
|
|
|
|
|
|
QStringList Line;
|
|
|
|
Line.append(pEntry->GetTimeStamp().toString("hh:mm:ss.zzz"));
|
|
|
|
QString Name = pEntry->GetProcessName();
|
|
|
|
Line.append(Name.isEmpty() ? tr("Unknown") : Name);
|
|
|
|
Line.append(QString("%1").arg(pEntry->GetProcessId()));
|
|
|
|
Line.append(QString("%1").arg(pEntry->GetThreadId()));
|
|
|
|
Line.append(pEntry->GetTypeStr());
|
|
|
|
Line.append(pEntry->GetStautsStr());
|
|
|
|
Line.append(pEntry->GetName());
|
|
|
|
Line.append(pEntry->GetMessage());
|
|
|
|
|
|
|
|
File.write(Line.join("\t").toLatin1() + "\n");
|
|
|
|
}
|
2021-10-16 17:24:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
File.close();
|
2022-07-09 10:46:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// CTraceWindow
|
|
|
|
|
|
|
|
CTraceWindow::CTraceWindow(QWidget *parent)
|
|
|
|
: QDialog(parent)
|
|
|
|
{
|
|
|
|
Qt::WindowFlags flags = windowFlags();
|
|
|
|
flags |= Qt::CustomizeWindowHint;
|
|
|
|
//flags &= ~Qt::WindowContextHelpButtonHint;
|
|
|
|
//flags &= ~Qt::WindowSystemMenuHint;
|
|
|
|
//flags &= ~Qt::WindowMinMaxButtonsHint;
|
|
|
|
//flags |= Qt::WindowMinimizeButtonHint;
|
|
|
|
//flags &= ~Qt::WindowCloseButtonHint;
|
|
|
|
flags &= ~Qt::WindowContextHelpButtonHint;
|
|
|
|
//flags &= ~Qt::WindowSystemMenuHint;
|
|
|
|
setWindowFlags(flags);
|
|
|
|
|
|
|
|
this->setWindowTitle(tr("Sandboxie-Plus - Trace Monitor"));
|
|
|
|
|
2022-07-11 18:30:09 +01:00
|
|
|
bool bAlwaysOnTop = theConf->GetBool("Options/AlwaysOnTop", false);
|
|
|
|
this->setWindowFlag(Qt::WindowStaysOnTopHint, bAlwaysOnTop);
|
|
|
|
|
2022-07-09 10:46:07 +01:00
|
|
|
QGridLayout* pLayout = new QGridLayout();
|
2022-09-29 17:28:48 +01:00
|
|
|
pLayout->setContentsMargins(3,3,3,3);
|
2022-07-09 10:46:07 +01:00
|
|
|
pLayout->addWidget(new CTraceView(true, this), 0, 0);
|
|
|
|
this->setLayout(pLayout);
|
|
|
|
|
|
|
|
restoreGeometry(theConf->GetBlob("TraceWindow/Window_Geometry"));
|
|
|
|
}
|
|
|
|
|
|
|
|
CTraceWindow::~CTraceWindow()
|
|
|
|
{
|
|
|
|
theConf->SetBlob("TraceWindow/Window_Geometry", saveGeometry());
|
|
|
|
|
2022-07-11 18:30:09 +01:00
|
|
|
if(!theAPI) theAPI->EnableMonitor(false);
|
2022-07-09 10:46:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CTraceWindow::closeEvent(QCloseEvent *e)
|
|
|
|
{
|
|
|
|
emit Closed();
|
|
|
|
this->deleteLater();
|
2021-10-16 17:24:16 +01:00
|
|
|
}
|