summaryrefslogtreecommitdiff
path: root/src/plugins/debugger/disassembleragent.cpp
diff options
context:
space:
mode:
authorFriedemann Kleint <Friedemann.Kleint@nokia.com>2011-07-01 16:57:17 +0200
committerFriedemann Kleint <Friedemann.Kleint@nokia.com>2011-07-04 10:25:54 +0200
commitb3812bf94f0f0e771ed832c14335ff868cfc0cff (patch)
treec96c9c23ccd9c1dd799fe4f49052f5db131116f8 /src/plugins/debugger/disassembleragent.cpp
parent53295db969cef0b233a56faab3ed216c66961a37 (diff)
downloadqt-creator-b3812bf94f0f0e771ed832c14335ff868cfc0cff.tar.gz
Debugger: Make disassembler caching smarter.
Check if address is contained in range of lines. Task-number: QTCREATORBUG-5205 Change-Id: I12a4f2f2f3837e164fd093e80fb427e9234136af Reviewed-on: http://codereview.qt.nokia.com/1017 Reviewed-by: Qt Sanity Bot <qt_sanity_bot@ovi.com> Reviewed-by: Friedemann Kleint <Friedemann.Kleint@nokia.com>
Diffstat (limited to 'src/plugins/debugger/disassembleragent.cpp')
-rw-r--r--src/plugins/debugger/disassembleragent.cpp103
1 files changed, 78 insertions, 25 deletions
diff --git a/src/plugins/debugger/disassembleragent.cpp b/src/plugins/debugger/disassembleragent.cpp
index ca63193ee7..80d58ad7f2 100644
--- a/src/plugins/debugger/disassembleragent.cpp
+++ b/src/plugins/debugger/disassembleragent.cpp
@@ -56,6 +56,8 @@
#include <QtGui/QTextBlock>
#include <QtGui/QIcon>
#include <QtCore/QPointer>
+#include <QtCore/QPair>
+#include <QtCore/QDir>
using namespace Core;
using namespace TextEditor;
@@ -69,6 +71,27 @@ namespace Internal {
//
///////////////////////////////////////////////////////////////////////
+class FrameKey
+{
+public:
+ FrameKey() : startAddress(0), endAddress(0) {}
+ inline bool matches(const Location &loc) const;
+
+ QString functionName;
+ QString fileName;
+ quint64 startAddress;
+ quint64 endAddress;
+};
+
+bool FrameKey::matches(const Location &loc) const
+{
+ return loc.address() >= startAddress
+ && loc.address() < endAddress
+ && loc.fileName() == fileName && loc.functionName() == functionName;
+}
+
+typedef QPair<FrameKey, DisassemblerLines> CacheEntry;
+
class DisassemblerAgentPrivate
{
public:
@@ -83,8 +106,9 @@ public:
QPointer<DebuggerEngine> engine;
ITextMark *locationMark;
QList<ITextMark *> breakpointMarks;
-
- QHash<QString, DisassemblerLines> cache;
+
+ QList<CacheEntry> cache;
+
QString mimeType;
bool m_resetLocationScheduled;
};
@@ -130,6 +154,14 @@ DisassemblerAgent::~DisassemblerAgent()
d = 0;
}
+int DisassemblerAgent::indexOf(const Location &loc) const
+{
+ for (int i = 0; i < d->cache.size(); i++)
+ if (d->cache.at(i).first.matches(loc))
+ return i;
+ return -1;
+}
+
void DisassemblerAgent::cleanup()
{
d->cache.clear();
@@ -150,12 +182,6 @@ void DisassemblerAgent::resetLocation()
}
}
-static QString frameKey(const Location &loc)
-{
- return _("%1:%2:%3").arg(loc.functionName())
- .arg(loc.fileName()).arg(loc.address());
-}
-
const Location &DisassemblerAgent::location() const
{
return d->location;
@@ -172,20 +198,28 @@ bool DisassemblerAgent::isMixed() const
void DisassemblerAgent::setLocation(const Location &loc)
{
d->location = loc;
- if (isMixed()) {
- QHash<QString, DisassemblerLines>::ConstIterator it =
- d->cache.find(frameKey(loc));
- if (it != d->cache.end()) {
- QString msg = _("Use cache disassembler for '%1' in '%2'")
- .arg(loc.functionName()).arg(loc.fileName());
- d->engine->showMessage(msg);
- setContents(*it);
- updateBreakpointMarkers();
- updateLocationMarker();
- return;
+ int index = indexOf(loc);
+ if (index != -1) {
+ // Refresh when not displaying a function and there is not sufficient
+ // context left past the address.
+ if (!isMixed() && d->cache.at(index).first.endAddress - loc.address() < 24) {
+ index = -1;
+ d->cache.removeAt(index);
}
}
- d->engine->fetchDisassembler(this);
+ if (index != -1) {
+ const FrameKey &key = d->cache.at(index).first;
+ const QString msg =
+ _("Using cached disassembly for 0x%1 (0x%2-0x%3) in '%4'/ '%5'")
+ .arg(loc.address(), 0, 16)
+ .arg(key.startAddress, 0, 16).arg(key.endAddress, 0, 16)
+ .arg(loc.functionName(), QDir::toNativeSeparators(loc.fileName()));
+ d->engine->showMessage(msg);
+ setContentsToEditor(d->cache.at(index).second);
+ d->m_resetLocationScheduled = false; // In case reset from previous run still pending.
+ } else {
+ d->engine->fetchDisassembler(this);
+ }
}
void DisassemblerAgentPrivate::configureMimeType()
@@ -225,6 +259,24 @@ void DisassemblerAgent::setMimeType(const QString &mt)
void DisassemblerAgent::setContents(const DisassemblerLines &contents)
{
QTC_ASSERT(d, return);
+ if (contents.size()) {
+ const quint64 startAddress = contents.startAddress();
+ const quint64 endAddress = contents.endAddress();
+ if (startAddress) {
+ FrameKey key;
+ key.fileName = d->location.fileName();
+ key.functionName = d->location.functionName();
+ key.startAddress = startAddress;
+ key.endAddress = endAddress;
+ d->cache.append(CacheEntry(key, contents));
+ }
+ }
+ setContentsToEditor(contents);
+}
+
+void DisassemblerAgent::setContentsToEditor(const DisassemblerLines &contents)
+{
+ QTC_ASSERT(d, return);
using namespace Core;
using namespace TextEditor;
@@ -260,7 +312,6 @@ void DisassemblerAgent::setContents(const DisassemblerLines &contents)
plainTextEdit->setPlainText(str);
plainTextEdit->setReadOnly(true);
- d->cache.insert(frameKey(d->location), contents);
d->editor->setDisplayName(_("Disassembler (%1)")
.arg(d->location.functionName()));
@@ -272,9 +323,10 @@ void DisassemblerAgent::updateLocationMarker()
{
QTC_ASSERT(d->editor, return);
- const DisassemblerLines &contents = d->cache.value(frameKey(d->location));
+ const int index = indexOf(d->location);
+ const DisassemblerLines contents = index != -1 ?
+ d->cache.at(index).second : DisassemblerLines();
int lineNumber = contents.lineForAddress(d->location.address());
-
if (d->location.needsMarker()) {
d->editor->markableInterface()->removeMark(d->locationMark);
if (lineNumber)
@@ -300,8 +352,9 @@ void DisassemblerAgent::updateBreakpointMarkers()
if (ids.isEmpty())
return;
- const DisassemblerLines &contents = d->cache.value(frameKey(d->location));
-
+ const int index = indexOf(d->location);
+ const DisassemblerLines contents = index != -1 ?
+ d->cache.at(index).second : DisassemblerLines();
foreach (TextEditor::ITextMark *marker, d->breakpointMarks)
d->editor->markableInterface()->removeMark(marker);
d->breakpointMarks.clear();