cutter/src/widgets/DisassemblerGraphView.cpp

683 lines
20 KiB
C++
Raw Normal View History

2017-10-09 09:38:57 +00:00
#include "DisassemblerGraphView.h"
#include <QPainter>
#include <QJsonObject>
#include <QJsonArray>
#include <QMouseEvent>
#include <QPropertyAnimation>
#include <QShortcut>
#include <QToolTip>
2017-10-09 09:38:57 +00:00
#include "cutter.h"
#include "utils/Colors.h"
#include "utils/Configuration.h"
#include "utils/CachedFontMetrics.h"
2017-10-09 09:38:57 +00:00
DisassemblerGraphView::DisassemblerGraphView(QWidget *parent)
: GraphView(parent),
mFontMetrics(nullptr),
mMenu(new DisassemblyContextMenu(this))
2017-10-09 09:38:57 +00:00
{
2017-12-14 21:07:48 +00:00
highlight_token = nullptr;
// Signals that require a refresh all
connect(Core(), SIGNAL(refreshAll()), this, SLOT(refreshView()));
connect(Core(), SIGNAL(commentsChanged()), this, SLOT(refreshView()));
connect(Core(), SIGNAL(functionRenamed(QString, QString)), this, SLOT(refreshView()));
connect(Core(), SIGNAL(flagsChanged()), this, SLOT(refreshView()));
connect(Core(), SIGNAL(varsChanged()), this, SLOT(refreshView()));
connect(Core(), SIGNAL(instructionChanged(RVA)), this, SLOT(refreshView()));
connect(Core(), SIGNAL(functionsChanged()), this, SLOT(refreshView()));
connect(Core(), SIGNAL(graphOptionsChanged()), this, SLOT(refreshView()));
2017-10-09 09:38:57 +00:00
connect(Config(), SIGNAL(colorsUpdated()), this, SLOT(colorsUpdatedSlot()));
connect(Config(), SIGNAL(fontsUpdated()), this, SLOT(fontsUpdatedSlot()));
connect(Core(), SIGNAL(seekChanged(RVA)), this, SLOT(onSeekChanged(RVA)));
2017-10-09 09:38:57 +00:00
// Space to switch to disassembly
2017-12-14 21:07:48 +00:00
QShortcut *shortcut_disassembly = new QShortcut(QKeySequence(Qt::Key_Space), this);
shortcut_disassembly->setContext(Qt::WidgetShortcut);
connect(shortcut_disassembly, &QShortcut::activated, this, []{
Core()->setMemoryWidgetPriority(CutterCore::MemoryWidgetType::Disassembly);
Core()->triggerRaisePrioritizedMemoryWidget();
});
// ESC for previous
QShortcut *shortcut_escape = new QShortcut(QKeySequence(Qt::Key_Escape), this);
shortcut_escape->setContext(Qt::WidgetShortcut);
connect(shortcut_escape, SIGNAL(activated()), this, SLOT(seekPrev()));
2017-10-09 09:38:57 +00:00
2017-12-13 22:57:36 +00:00
// Zoom shortcuts
QShortcut *shortcut_zoom_in = new QShortcut(QKeySequence(Qt::Key_Plus), this);
shortcut_zoom_in->setContext(Qt::WidgetShortcut);
connect(shortcut_zoom_in, SIGNAL(activated()), this, SLOT(zoomIn()));
QShortcut *shortcut_zoom_out = new QShortcut(QKeySequence(Qt::Key_Minus), this);
2017-12-13 22:57:36 +00:00
shortcut_zoom_out->setContext(Qt::WidgetShortcut);
connect(shortcut_zoom_out, SIGNAL(activated()), this, SLOT(zoomOut()));
2017-12-13 22:57:36 +00:00
QShortcut *shortcut_zoom_reset = new QShortcut(QKeySequence(Qt::Key_Equal), this);
shortcut_zoom_reset->setContext(Qt::WidgetShortcut);
connect(shortcut_zoom_reset, SIGNAL(activated()), this, SLOT(zoomReset()));
2017-10-09 09:38:57 +00:00
2017-12-13 22:57:36 +00:00
// Branch shortcuts
QShortcut *shortcut_take_true = new QShortcut(QKeySequence(Qt::Key_T), this);
shortcut_take_true->setContext(Qt::WidgetShortcut);
connect(shortcut_take_true, SIGNAL(activated()), this, SLOT(takeTrue()));
QShortcut *shortcut_take_false = new QShortcut(QKeySequence(Qt::Key_F), this);
shortcut_take_false->setContext(Qt::WidgetShortcut);
connect(shortcut_take_false, SIGNAL(activated()), this, SLOT(takeFalse()));
2017-12-14 21:07:48 +00:00
// Navigation shortcuts
QShortcut *shortcut_next_instr = new QShortcut(QKeySequence(Qt::Key_J), this);
shortcut_next_instr->setContext(Qt::WidgetShortcut);
connect(shortcut_next_instr, SIGNAL(activated()), this, SLOT(nextInstr()));
QShortcut *shortcut_prev_instr = new QShortcut(QKeySequence(Qt::Key_K), this);
shortcut_prev_instr->setContext(Qt::WidgetShortcut);
connect(shortcut_prev_instr, SIGNAL(activated()), this, SLOT(prevInstr()));
shortcuts.append(shortcut_disassembly);
shortcuts.append(shortcut_escape);
shortcuts.append(shortcut_zoom_in);
shortcuts.append(shortcut_zoom_out);
shortcuts.append(shortcut_zoom_reset);
shortcuts.append(shortcut_next_instr);
shortcuts.append(shortcut_prev_instr);
initFont();
colorsUpdatedSlot();
2017-10-09 09:38:57 +00:00
}
2017-12-14 21:07:48 +00:00
DisassemblerGraphView::~DisassemblerGraphView()
{
for(QShortcut *shortcut : shortcuts)
{
delete shortcut;
}
}
void DisassemblerGraphView::refreshView()
2017-10-09 09:38:57 +00:00
{
initFont();
2017-10-09 09:38:57 +00:00
loadCurrentGraph();
2017-12-14 21:07:48 +00:00
viewport()->update();
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::loadCurrentGraph()
{
QJsonDocument functionsDoc = Core()->cmdj("agj");
2017-10-09 09:38:57 +00:00
QJsonArray functions = functionsDoc.array();
disassembly_blocks.clear();
2017-12-14 21:07:48 +00:00
blocks.clear();
2017-10-09 09:38:57 +00:00
Analysis anal;
anal.ready = true;
QJsonValue funcRef = functions.first();
QJsonObject func = funcRef.toObject();
Function f;
f.ready = true;
f.entry = func["offset"].toVariant().toULongLong();
2017-11-21 17:48:01 +00:00
QString windowTitle = tr("Graph");
QString funcName = func["name"].toString().trimmed();
if (!funcName.isEmpty())
{
windowTitle += " (" + funcName + ")";
}
2017-12-14 21:07:48 +00:00
parentWidget()->setWindowTitle(windowTitle);
2017-10-09 09:38:57 +00:00
RVA entry = func["offset"].toVariant().toULongLong();
setEntry(entry);
2017-10-09 09:38:57 +00:00
for (QJsonValueRef blockRef : func["blocks"].toArray()) {
QJsonObject block = blockRef.toObject();
RVA block_entry = block["offset"].toVariant().toULongLong();
RVA block_fail = block["fail"].toVariant().toULongLong();
RVA block_jump = block["jump"].toVariant().toULongLong();
2017-10-09 09:38:57 +00:00
DisassemblyBlock db;
GraphBlock gb;
gb.entry = block_entry;
db.entry = block_entry;
db.true_path = RVA_INVALID;
db.false_path = RVA_INVALID;
if(block_fail)
2017-10-09 09:38:57 +00:00
{
db.false_path = block_fail;
gb.exits.push_back(block_fail);
2017-10-09 09:38:57 +00:00
}
if(block_jump)
2017-10-09 09:38:57 +00:00
{
if(block_fail)
{
db.true_path = block_jump;
2017-10-12 12:14:33 +00:00
}
gb.exits.push_back(block_jump);
2017-10-09 09:38:57 +00:00
}
for (QJsonValueRef opRef : block["ops"].toArray()) {
QJsonObject op = opRef.toObject();
Instr i;
i.addr = op["offset"].toVariant().toULongLong();
// Skip last byte, otherwise it will overlap with next instruction
i.size = op["size"].toVariant().toULongLong() - 1;
2017-10-09 09:38:57 +00:00
RichTextPainter::List richText;
2017-10-15 11:38:28 +00:00
Colors::colorizeAssembly(richText, op["opcode"].toString(), op["type_num"].toVariant().toULongLong());
2017-10-14 11:00:23 +00:00
if (op["comment"].toString().length()) {
RichTextPainter::CustomRichText_t comment;
comment.text = QString(" ; %1").arg(QByteArray::fromBase64(op["comment"].toString().toLocal8Bit()).data());
2017-11-05 15:08:17 +00:00
comment.textColor = mCommentColor;
2017-10-14 11:00:23 +00:00
comment.flags = RichTextPainter::FlagColor;
richText.insert(richText.end(), comment);
}
bool cropped;
i.text = Text(RichTextPainter::cropped(richText, Config()->getGraphBlockMaxChars(), "...", &cropped));
if(cropped)
{
i.fullText = richText;
}
else
{
i.fullText = Text();
}
db.instrs.push_back(i);
2017-10-09 09:38:57 +00:00
}
disassembly_blocks[db.entry] = db;
prepareGraphNode(gb);
f.blocks.push_back(db);
addBlock(gb);
2017-10-09 09:38:57 +00:00
}
2017-10-09 09:38:57 +00:00
anal.functions[f.entry] = f;
anal.status = "Ready.";
anal.entry = f.entry;
if(func["blocks"].toArray().size() > 0)
{
computeGraph(entry);
2017-12-14 21:07:48 +00:00
viewport()->update();
if(first_draw)
{
showBlock(blocks[entry]);
first_draw = false;
}
}
}
2017-10-09 09:38:57 +00:00
void DisassemblerGraphView::prepareGraphNode(GraphBlock &block)
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock &db = disassembly_blocks[block.entry];
int width = 0;
int height = 0;
for(auto & line : db.header_text.lines)
{
int lw = 0;
for(auto & part : line)
lw += mFontMetrics->width(part.text);
if(lw > width)
width = lw;
height += 1;
}
for(Instr & instr : db.instrs)
2017-10-09 09:38:57 +00:00
{
for(auto & line : instr.text.lines)
2017-10-09 09:38:57 +00:00
{
int lw = 0;
for(auto & part : line)
lw += mFontMetrics->width(part.text);
if(lw > width)
width = lw;
height += 1;
2017-10-09 09:38:57 +00:00
}
}
2017-12-14 21:07:48 +00:00
int extra = 4 * charWidth + 4;
block.width = width + extra + charWidth;
block.height = (height * charHeight) + extra;
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::initFont()
2017-11-05 15:08:17 +00:00
{
setFont(Config()->getFont());
2017-12-14 21:07:48 +00:00
QFontMetricsF metrics(font());
baseline = int(metrics.ascent());
charWidth = metrics.width('X');
charHeight = metrics.height();
charOffset = 0;
if(mFontMetrics)
delete mFontMetrics;
mFontMetrics = new CachedFontMetrics(this, font());
2017-11-05 15:08:17 +00:00
}
void DisassemblerGraphView::drawBlock(QPainter & p, GraphView::GraphBlock &block)
2017-10-09 09:38:57 +00:00
{
p.setPen(Qt::black);
p.setBrush(Qt::gray);
p.drawRect(block.x, block.y, block.width, block.height);
2017-10-09 09:38:57 +00:00
// Render node
DisassemblyBlock &db = disassembly_blocks[block.entry];
bool block_selected = false;
RVA selected_instruction = RVA_INVALID;
2017-10-09 09:38:57 +00:00
// Figure out if the current block is selected
for(const Instr & instr : db.instrs)
{
RVA addr = Core()->getOffset();
if((instr.addr <= addr) && (addr <= instr.addr+instr.size))
{
block_selected = true;
selected_instruction = instr.addr;
}
// TODO: L219
}
2017-10-16 19:00:47 +00:00
p.setPen(QColor(0, 0, 0, 0));
if(db.terminal)
{
p.setBrush(retShadowColor);
} else if(db.indirectcall) {
p.setBrush(indirectcallShadowColor);
} else {
p.setBrush(QColor(0, 0, 0, 128));
}
p.drawRect(block.x + 4, block.y + 4,
block.width + 4, block.height + 4);
p.setPen(graphNodeColor);
if(block_selected)
{
p.setBrush(disassemblySelectedBackgroundColor);
} else {
p.setBrush(disassemblyBackgroundColor);
}
p.drawRect(block.x, block.y,
block.width, block.height);
// Draw different background for selected instruction
if(selected_instruction != RVA_INVALID)
{
2017-12-14 21:07:48 +00:00
int y = block.y + (2 * charWidth) + (db.header_text.lines.size() * charHeight);
for(Instr & instr : db.instrs)
{
auto selected = instr.addr == selected_instruction;
//auto traceCount = dbgfunctions->GetTraceRecordHitCount(instr.addr);
auto traceCount = 0;
if(selected && traceCount)
{
2017-12-14 21:07:48 +00:00
p.fillRect(QRect(block.x + charWidth, y, block.width - (10 + 2 * charWidth),
int(instr.text.lines.size()) * charHeight), disassemblyTracedSelectionColor);
}
else if(selected)
{
2017-12-14 21:07:48 +00:00
p.fillRect(QRect(block.x + charWidth, y, block.width - (10 + 2 * charWidth),
int(instr.text.lines.size()) * charHeight), disassemblySelectionColor);
}
else if(traceCount)
{
// Color depending on how often a sequence of code is executed
int exponent = 1;
while(traceCount >>= 1) //log2(traceCount)
exponent++;
int colorDiff = (exponent * exponent) / 2;
// If the user has a light trace background color, substract
if(disassemblyTracedColor.blue() > 160)
colorDiff *= -1;
2017-12-14 21:07:48 +00:00
p.fillRect(QRect(block.x + charWidth, y, block.width - (10 + 2 * charWidth), int(instr.text.lines.size()) * charHeight),
QColor(disassemblyTracedColor.red(),
disassemblyTracedColor.green(),
std::max(0, std::min(256, disassemblyTracedColor.blue() + colorDiff))));
}
2017-12-14 21:07:48 +00:00
y += int(instr.text.lines.size()) * charHeight;
}
}
// Render node text
2017-12-14 21:07:48 +00:00
auto x = block.x + (2 * charWidth);
int y = block.y + (2 * charWidth);
for(auto & line : db.header_text.lines)
{
2017-12-14 21:07:48 +00:00
RichTextPainter::paintRichText(&p, x, y, block.width, charHeight, 0, line, mFontMetrics);
y += charHeight;
}
for(Instr & instr : db.instrs)
{
for(auto & line : instr.text.lines)
{
2017-12-14 21:07:48 +00:00
int rectSize = qRound(charWidth);
if(rectSize % 2)
{
rectSize++;
}
// Assume charWidth <= charHeight
2017-12-14 21:07:48 +00:00
QRectF bpRect(x - rectSize / 3.0, y + (charHeight - rectSize) / 2.0, rectSize, rectSize);
// TODO: Breakpoint/Cip stuff
2017-12-14 21:07:48 +00:00
RichTextPainter::paintRichText(&p, x + charWidth, y, block.width - charWidth, charHeight, 0, line, mFontMetrics);
y += charHeight;
}
}
2017-10-09 09:38:57 +00:00
}
GraphView::EdgeConfiguration DisassemblerGraphView::edgeConfiguration(GraphView::GraphBlock &from, GraphView::GraphBlock *to)
2017-10-09 09:38:57 +00:00
{
EdgeConfiguration ec;
DisassemblyBlock &db = disassembly_blocks[from.entry];
if(to->entry == db.true_path)
{
ec.color = brtrueColor;
}
else if(to->entry == db.false_path)
{
ec.color = brfalseColor;
}
else
{
ec.color = jmpColor;
}
ec.start_arrow = false;
ec.end_arrow = true;
return ec;
2017-10-09 09:38:57 +00:00
}
RVA DisassemblerGraphView::getAddrForMouseEvent(GraphBlock &block, QPoint *point)
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock &db = disassembly_blocks[block.entry];
2017-12-14 21:07:48 +00:00
// Remove header and margin
int off_y = (2 * charWidth) + (db.header_text.lines.size() * charHeight);
// Get mouse coordinate over the actual text
int text_point_y = point->y() - off_y;
int mouse_row = text_point_y / charHeight;
int cur_row = db.header_text.lines.size();
if (mouse_row < cur_row)
{
return db.entry;
}
Instr *instr = getInstrForMouseEvent(block, point);
if(instr)
{
return instr->addr;
}
return RVA_INVALID;
}
DisassemblerGraphView::Instr *DisassemblerGraphView::getInstrForMouseEvent(GraphView::GraphBlock &block, QPoint *point)
{
DisassemblyBlock &db = disassembly_blocks[block.entry];
// Remove header and margin
int off_y = (2 * charWidth) + (db.header_text.lines.size() * charHeight);
// Get mouse coordinate over the actual text
int text_point_y = point->y() - off_y;
int mouse_row = text_point_y / charHeight;
int cur_row = db.header_text.lines.size();
for(Instr & instr : db.instrs)
{
if(mouse_row < cur_row + (int)instr.text.lines.size())
{
return &instr;
}
cur_row += instr.text.lines.size();
}
2017-10-09 09:38:57 +00:00
return nullptr;
}
// Public Slots
2017-10-09 09:38:57 +00:00
void DisassemblerGraphView::colorsUpdatedSlot()
{
2017-11-20 11:23:37 +00:00
disassemblyBackgroundColor = ConfigColor("gui.alt_background");
disassemblySelectedBackgroundColor = ConfigColor("gui.background");
2017-10-22 10:21:44 +00:00
mDisabledBreakpointColor = disassemblyBackgroundColor;
2017-10-15 07:14:05 +00:00
graphNodeColor = ConfigColor("gui.border");
2017-11-20 11:23:37 +00:00
backgroundColor = ConfigColor("gui.background");
2017-12-02 15:22:05 +00:00
disassemblySelectionColor = ConfigColor("highlight");
2017-10-15 07:14:05 +00:00
jmpColor = ConfigColor("graph.trufae");
brtrueColor = ConfigColor("graph.true");
brfalseColor = ConfigColor("graph.false");
2017-11-05 15:08:17 +00:00
mCommentColor = ConfigColor("comment");
initFont();
2017-11-21 14:33:15 +00:00
refreshView();
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::fontsUpdatedSlot()
{
2017-12-14 21:07:48 +00:00
initFont();
refreshView();
2017-10-09 09:38:57 +00:00
}
DisassemblerGraphView::DisassemblyBlock *DisassemblerGraphView::blockForAddress(RVA addr)
2017-10-09 09:38:57 +00:00
{
for(auto & blockIt : disassembly_blocks)
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock &db = blockIt.second;
for(Instr i : db.instrs)
{
if((i.addr <= addr) && (addr <= i.addr + i.size))
{
return &db;
}
}
2017-10-09 09:38:57 +00:00
}
return nullptr;
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::onSeekChanged(RVA addr)
2017-10-09 09:38:57 +00:00
{
mMenu->setOffset(addr);
// If this seek was NOT done by us...
if(!sent_seek)
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock *db = blockForAddress(addr);
if(db)
{
// This is a local address! We animated to it.
transition_dont_seek = true;
showBlock(&blocks[db->entry], true);
return;
} else {
refreshView();
DisassemblyBlock *db = blockForAddress(addr);
if(db)
{
// This is a local address! We animated to it.
transition_dont_seek = true;
showBlock(&blocks[db->entry], false);
return;
}
}
2017-10-09 09:38:57 +00:00
}
sent_seek = false;
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::zoomIn()
2017-10-09 09:38:57 +00:00
{
current_scale += 0.1;
2017-12-14 21:07:48 +00:00
auto areaSize = viewport()->size();
adjustSize(areaSize.width(), areaSize.height());
viewport()->update();
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::zoomOut()
2017-10-09 09:38:57 +00:00
{
current_scale -= 0.1;
current_scale = std::max(current_scale, 0.3);
2017-12-14 21:07:48 +00:00
auto areaSize = viewport()->size();
adjustSize(areaSize.width(), areaSize.height());
viewport()->update();
2017-10-09 09:38:57 +00:00
}
2017-12-13 22:57:36 +00:00
void DisassemblerGraphView::zoomReset()
{
current_scale = 1.0;
2017-12-14 21:07:48 +00:00
auto areaSize = viewport()->size();
adjustSize(areaSize.width(), areaSize.height());
viewport()->update();
2017-12-13 22:57:36 +00:00
}
void DisassemblerGraphView::takeTrue()
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock *db = blockForAddress(Core()->getOffset());
if(db->true_path != RVA_INVALID)
2017-10-09 09:38:57 +00:00
{
Core()->seek(db->true_path);
}
else if(blocks[db->entry].exits.size())
{
Core()->seek(blocks[db->entry].exits[0]);
}
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::takeFalse()
2017-10-09 09:38:57 +00:00
{
DisassemblyBlock *db = blockForAddress(Core()->getOffset());
if(db->false_path != RVA_INVALID)
{
Core()->seek(db->false_path);
}
else if(blocks[db->entry].exits.size())
{
Core()->seek(blocks[db->entry].exits[0]);
}
2017-10-09 09:38:57 +00:00
}
2017-12-14 21:07:48 +00:00
void DisassemblerGraphView::seekInstruction(bool previous_instr)
{
RVA addr = Core()->getOffset();
DisassemblyBlock *db = blockForAddress(addr);
if(!db)
{
return;
}
for(size_t i=0; i < db->instrs.size(); i++)
{
Instr &instr = db->instrs[i];
if(!((instr.addr <= addr) && (addr <= instr.addr + instr.size)))
{
continue;
}
// Found the instructon. Check if a next one exists
if(!previous_instr && (i < db->instrs.size()-1))
{
seek(db->instrs[i+1].addr, true);
}
else if(previous_instr && (i > 0))
{
seek(db->instrs[i-1].addr);
}
}
}
void DisassemblerGraphView::nextInstr()
{
seekInstruction(false);
}
void DisassemblerGraphView::prevInstr()
{
seekInstruction(true);
}
void DisassemblerGraphView::seek(RVA addr, bool update_viewport)
2017-10-09 09:38:57 +00:00
{
sent_seek = true;
Core()->seek(addr);
if(update_viewport)
{
2017-12-14 21:07:48 +00:00
viewport()->update();
}
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::seekPrev()
2017-10-09 09:38:57 +00:00
{
Core()->seekPrev();
2017-10-09 09:38:57 +00:00
}
void DisassemblerGraphView::blockClicked(GraphView::GraphBlock &block, QMouseEvent *event, QPoint pos)
2017-10-09 09:38:57 +00:00
{
RVA instr = getAddrForMouseEvent(block, &pos);
if(instr == RVA_INVALID)
2017-10-09 09:38:57 +00:00
{
return;
}
seek(instr, true);
2017-10-09 09:38:57 +00:00
if(event->button() == Qt::RightButton)
2017-10-09 09:38:57 +00:00
{
mMenu->setOffset(instr);
mMenu->exec(event->globalPos());
2017-10-09 09:38:57 +00:00
}
}
2017-12-14 21:07:48 +00:00
void DisassemblerGraphView::blockDoubleClicked(GraphView::GraphBlock &block, QMouseEvent *event, QPoint pos)
{
Q_UNUSED(event);
RVA instr = getAddrForMouseEvent(block, &pos);
2017-12-14 21:07:48 +00:00
if(instr == RVA_INVALID)
{
return;
}
QList<XrefDescription> refs = Core()->getXRefs(instr, false, false);
if (refs.length()) {
sent_seek = false;
Core()->seek(refs.at(0).to);
}
if (refs.length() > 1) {
qWarning() << "Too many references here. Weird behaviour expected.";
}
}
void DisassemblerGraphView::blockHelpEvent(GraphView::GraphBlock &block, QHelpEvent *event, QPoint pos)
{
Instr *instr = getInstrForMouseEvent(block, &pos);
if(!instr || instr->fullText.lines.empty())
{
QToolTip::hideText();
event->ignore();
return;
}
QToolTip::showText(event->globalPos(), instr->fullText.ToQString());
}
bool DisassemblerGraphView::helpEvent(QHelpEvent *event)
{
if(!GraphView::helpEvent(event))
{
QToolTip::hideText();
event->ignore();
}
return true;
}
void DisassemblerGraphView::blockTransitionedTo(GraphView::GraphBlock *to)
2017-10-09 09:38:57 +00:00
{
if(transition_dont_seek)
2017-10-09 09:38:57 +00:00
{
transition_dont_seek = false;
return;
2017-10-09 09:38:57 +00:00
}
seek(to->entry);
2017-10-16 19:00:47 +00:00
}