Skip to content

Commit

Permalink
wip
Browse files Browse the repository at this point in the history
  • Loading branch information
lievenhey committed Jun 30, 2023
1 parent 136afd8 commit 33ca598
Show file tree
Hide file tree
Showing 6 changed files with 210 additions and 22 deletions.
1 change: 1 addition & 0 deletions src/models/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ add_library(
codedelegate.cpp
costdelegate.cpp
data.cpp
disassemblyentry.cpp
disassemblymodel.cpp
disassemblyoutput.cpp
eventmodel.cpp
Expand Down
36 changes: 36 additions & 0 deletions src/models/disassemblyentry.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
/*
SPDX-FileCopyrightText: Lieven Hey <[email protected]>
SPDX-FileCopyrightText: 2023 Klarälvdalens Datakonsult AB, a KDAB Group company, [email protected]
SPDX-License-Identifier: GPL-2.0-or-later
*/

#include "disassemblyentry.h"

DisassemblyEntry::DisassemblyEntry(DisassemblyEntry* parent, int row,
const DisassemblyOutput::DisassemblyLine& disassemblyLine, QTextLine textLine)
: m_parent(parent)
, m_disassemblyLine(disassemblyLine)
, m_textLine(textLine)
, m_row(row)
{
}

DisassemblyEntry* DisassemblyEntry::lastChild()
{
if (m_lines.isEmpty()) {
return nullptr;
}

return &m_lines[m_lines.size() - 1];
}

void DisassemblyEntry::addChild(const DisassemblyEntry& line)
{
m_lines.push_back(line);
}

void DisassemblyEntry::clear()
{
m_lines.clear();
}
63 changes: 63 additions & 0 deletions src/models/disassemblyentry.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
/*
SPDX-FileCopyrightText: Lieven Hey <[email protected]>
SPDX-FileCopyrightText: 2023 Klarälvdalens Datakonsult AB, a KDAB Group company, [email protected]
SPDX-License-Identifier: GPL-2.0-or-later
*/

#pragma once

#include "disassemblyoutput.h"
#include <QTextLine>
#include <QVector>

class DisassemblyEntry
{
public:
DisassemblyEntry(DisassemblyEntry* parent, int row, const DisassemblyOutput::DisassemblyLine& disassemblyLine = {},
QTextLine textLine = {});

~DisassemblyEntry() = default;

DisassemblyEntry* parent() const
{
return m_parent;
}

int row() const
{
return m_row;
}

DisassemblyEntry* child(int row)
{
return &m_lines[row];
}

DisassemblyEntry* lastChild();

DisassemblyOutput::DisassemblyLine disassemblyLine() const
{
return m_disassemblyLine;
}

QTextLine textLine() const
{
return m_textLine;
}

int childCount() const
{
return m_lines.size();
}

void clear();
void addChild(const DisassemblyEntry& line);

private:
DisassemblyEntry* m_parent;
QVector<DisassemblyEntry> m_lines;
DisassemblyOutput::DisassemblyLine m_disassemblyLine;
QTextLine m_textLine;
int m_row;
};
116 changes: 96 additions & 20 deletions src/models/disassemblymodel.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,10 @@
#include "sourcecodemodel.h"

DisassemblyModel::DisassemblyModel(KSyntaxHighlighting::Repository* repository, QObject* parent)
: QAbstractTableModel(parent)
: QAbstractItemModel(parent)
, m_document(new QTextDocument(this))
, m_highlighter(new Highlighter(m_document, repository, this))
, m_disassemblyLines(nullptr, 0)
{
m_document->setUndoRedoEnabled(false);
}
Expand Down Expand Up @@ -47,6 +48,43 @@ QModelIndex DisassemblyModel::findIndexWithOffset(int offset)
return {};
}

QModelIndex DisassemblyModel::parent(const QModelIndex& index) const
{
if (!index.isValid()) {
return QModelIndex();
}

DisassemblyEntry* line = static_cast<DisassemblyEntry*>(index.internalPointer());
auto parentItem = line->parent();

if (parentItem == &m_disassemblyLines) {
return QModelIndex();
}

return createIndex(parentItem->row(), 0, parentItem);
}

QModelIndex DisassemblyModel::index(int row, int column, const QModelIndex& parent) const
{
if (!hasIndex(row, column, parent)) {
return QModelIndex();
}

DisassemblyEntry* parentItem;

if (!parent.isValid()) {
parentItem = const_cast<DisassemblyEntry*>(&m_disassemblyLines);
} else {
parentItem = static_cast<DisassemblyEntry*>(parent.internalPointer());
}

if (row < parentItem->childCount()) {
return createIndex(row, column, parentItem->child(row));
} else {
return QModelIndex();
}
}

void DisassemblyModel::setDisassembly(const DisassemblyOutput& disassemblyOutput,
const Data::CallerCalleeResults& results)
{
Expand All @@ -57,9 +95,11 @@ void DisassemblyModel::setDisassembly(const DisassemblyOutput& disassemblyOutput
m_numTypes = results.selfCosts.numTypes();

m_document->clear();
m_disassemblyLines.clear();

QTextCursor cursor(m_document);
cursor.beginEditBlock();
DisassemblyEntry* lastChild = nullptr;
for (const auto& it : disassemblyOutput.disassemblyLines) {
cursor.insertText(it.disassembly);
cursor.insertBlock();
Expand All @@ -68,6 +108,30 @@ void DisassemblyModel::setDisassembly(const DisassemblyOutput& disassemblyOutput

m_document->setTextWidth(m_document->idealWidth());

int childRow = 0;
int row = 0;
int linecounter = 0;
for (const auto& it : disassemblyOutput.disassemblyLines) {
auto textLine = m_document->findBlockByLineNumber(linecounter++).layout()->lineAt(0);

if (it.fileLine.file != disassemblyOutput.mainSourceFileName) {
if (lastChild) {
lastChild->addChild({lastChild, childRow, it, textLine});
childRow++;
} else {
childRow = 1;
m_disassemblyLines.addChild({&m_disassemblyLines, row, {0, QLatin1String("[inlined]"), {}, {}}});
lastChild = m_disassemblyLines.lastChild();
Q_ASSERT(lastChild);
lastChild->addChild({lastChild, 0, it, textLine});
}
} else {
lastChild = nullptr;
m_disassemblyLines.addChild({&m_disassemblyLines, row, it, textLine});
row++;
}
}

endResetModel();
}

Expand Down Expand Up @@ -107,29 +171,32 @@ QVariant DisassemblyModel::data(const QModelIndex& index, int role) const
return static_cast<int>(Qt::AlignLeft | Qt::AlignVCenter);
}

const auto& data = m_data.disassemblyLines.at(index.row());
// const auto& data = m_data.disassemblyLines.at(index.row());

DisassemblyEntry* entry = static_cast<DisassemblyEntry*>(index.internalPointer());
Q_ASSERT(entry);
auto line = entry->disassemblyLine();

if (role == Qt::DisplayRole || role == CostRole || role == TotalCostRole || role == SyntaxHighlightRole
|| role == Qt::ToolTipRole) {
if (role != Qt::ToolTipRole) {
if (index.column() == AddrColumn) {
if (!data.addr)
if (!line.addr)
return {};
return QString::number(data.addr, 16);
return QString::number(line.addr, 16);
} else if (index.column() == DisassemblyColumn) {
const auto block = m_document->findBlockByLineNumber(index.row());
if (role == SyntaxHighlightRole)
return QVariant::fromValue(block.layout()->lineAt(0));
return block.text();
return QVariant::fromValue(entry->textLine());
return line.disassembly;
}
}

if (data.addr == 0) {
if (line.addr == 0) {
return {};
}

const auto entry = m_results.entries.value(m_data.symbol);
auto it = entry.offsetMap.find(data.addr);
auto it = entry.offsetMap.find(line.addr);
if (it != entry.offsetMap.end()) {
int event = index.column() - COLUMN_COUNT;

Expand All @@ -143,7 +210,7 @@ QVariant DisassemblyModel::data(const QModelIndex& index, int role) const
return totalCost;
} else if (role == Qt::ToolTipRole) {
auto tooltip = tr("addr: <tt>%1</tt><br/>assembly: <tt>%2</tt><br/>disassembly: <tt>%3</tt>")
.arg(QString::number(data.addr, 16), data.disassembly);
.arg(QString::number(line.addr, 16), line.disassembly);
return Util::formatTooltip(tooltip, locationCost, m_results.selfCosts);
}

Expand All @@ -153,37 +220,46 @@ QVariant DisassemblyModel::data(const QModelIndex& index, int role) const
} else {
if (role == Qt::ToolTipRole)
return tr("<qt><tt>%1</tt><hr/>No samples at this location.</qt>")
.arg(data.disassembly.toHtmlEscaped());
.arg(line.disassembly.toHtmlEscaped());
else
return QString();
}
} else if (role == DisassemblyModel::HighlightRole) {
return data.fileLine.line == m_highlightLine;
return line.fileLine.line == m_highlightLine;
} else if (role == LinkedFunctionNameRole) {
return data.linkedFunction.name;
return line.linkedFunction.name;
} else if (role == LinkedFunctionOffsetRole) {
return data.linkedFunction.offset;
} else if (role == RainbowLineNumberRole && data.addr) {
return data.fileLine.line;
return line.linkedFunction.offset;
} else if (role == RainbowLineNumberRole && line.addr) {
return line.fileLine.line;
}

return {};
}

int DisassemblyModel::columnCount(const QModelIndex& parent) const
{
return parent.isValid() ? 0 : COLUMN_COUNT + m_numTypes;
Q_UNUSED(parent);
return COLUMN_COUNT + m_numTypes;
}

int DisassemblyModel::rowCount(const QModelIndex& parent) const
{
return parent.isValid() ? 0 : m_data.disassemblyLines.count();
// return parent.isValid() ? 0 : m_data.disassemblyLines.count();
if (parent.column() > 0)
return 0;
if (parent.isValid()) {
auto item = static_cast<DisassemblyEntry*>(parent.internalPointer());
return item->childCount();
}
return m_disassemblyLines.childCount();
}

void DisassemblyModel::updateHighlighting(int line)
{
m_highlightLine = line;
emit dataChanged(createIndex(0, Columns::DisassemblyColumn), createIndex(rowCount(), Columns::DisassemblyColumn));
// emit dataChanged(createIndex(0, Columns::DisassemblyColumn), createIndex(rowCount(),
// Columns::DisassemblyColumn));
}

Data::FileLine DisassemblyModel::fileLineForIndex(const QModelIndex& index) const
Expand Down Expand Up @@ -220,5 +296,5 @@ QModelIndex DisassemblyModel::indexForFileLine(const Data::FileLine& fileLine) c

if (bestMatch == -1)
return {};
return index(bestMatch, 0);
return QModelIndex(); // index(bestMatch, 0);
}
9 changes: 7 additions & 2 deletions src/models/disassemblymodel.h
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,11 @@
#pragma once

#include <memory>
#include <QAbstractTableModel>
#include <QAbstractItemModel>
#include <QTextLine>

#include "data.h"
#include "disassemblyentry.h"
#include "disassemblyoutput.h"

class QTextDocument;
Expand All @@ -23,7 +24,7 @@ class Definition;
class Repository;
}

class DisassemblyModel : public QAbstractTableModel
class DisassemblyModel : public QAbstractItemModel
{
Q_OBJECT
public:
Expand All @@ -35,6 +36,9 @@ class DisassemblyModel : public QAbstractTableModel
void clear();
QModelIndex findIndexWithOffset(int offset);

QModelIndex parent(const QModelIndex& index) const override;
QModelIndex index(int row, int column, const QModelIndex& parent) const override;

int rowCount(const QModelIndex& parent = QModelIndex()) const override;
int columnCount(const QModelIndex& parent = QModelIndex()) const override;

Expand Down Expand Up @@ -73,6 +77,7 @@ public slots:
private:
QTextDocument* m_document;
Highlighter* m_highlighter;
DisassemblyEntry m_disassemblyLines;
DisassemblyOutput m_data;
Data::CallerCalleeResults m_results;
int m_numTypes = 0;
Expand Down
7 changes: 7 additions & 0 deletions src/models/disassemblyoutput.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,13 @@ static ObjectdumpOutput objdumpParse(const QByteArray& output)
continue;
}

// inlining create lines like these
// std::ostream::operator<<(std::ostream& (*)(std::ostream&)):
// we want to skip those
if (asmLine.endsWith(QLatin1Char(':'))) {
continue;
}

// we don't care about the file name
if (asmLine.startsWith(QLatin1Char('/')) && asmLine.contains(QStringLiteral("file format"))) {
continue;
Expand Down

0 comments on commit 33ca598

Please sign in to comment.