From 2e414069c5a5b6eae78c5ca5e3742ce55001f37a Mon Sep 17 00:00:00 2001 From: karliss Date: Tue, 6 Sep 2022 18:46:39 +0300 Subject: [PATCH] Support editing in hex editor (#3026) --- src/widgets/HexWidget.cpp | 1004 ++++++++++++++++++++++++++++++++++--- src/widgets/HexWidget.h | 178 +++++-- 2 files changed, 1084 insertions(+), 98 deletions(-) diff --git a/src/widgets/HexWidget.cpp b/src/widgets/HexWidget.cpp index 39f63186..d933b944 100644 --- a/src/widgets/HexWidget.cpp +++ b/src/widgets/HexWidget.cpp @@ -26,6 +26,7 @@ static constexpr uint64_t MAX_COPY_SIZE = 128 * 1024 * 1024; static constexpr int MAX_LINE_WIDTH_PRESET = 32; static constexpr int MAX_LINE_WIDTH_BYTES = 128 * 1024; +static constexpr int WARNING_TIME_MS = 500; HexWidget::HexWidget(QWidget *parent) : QScrollArea(parent), @@ -42,7 +43,8 @@ HexWidget::HexWidget(QWidget *parent) showHeader(true), showAscii(true), showExHex(true), - showExAddr(true) + showExAddr(true), + warningTimer(this) { setMouseTracking(true); setFocusPolicy(Qt::FocusPolicy::StrongFocus); @@ -103,7 +105,7 @@ HexWidget::HexWidget(QWidget *parent) actionItemBigEndian = new QAction(tr("Big Endian"), this); actionItemBigEndian->setCheckable(true); actionItemBigEndian->setEnabled(false); - connect(actionItemBigEndian, &QAction::triggered, this, &HexWidget::setItemEndianess); + connect(actionItemBigEndian, &QAction::triggered, this, &HexWidget::setItemEndianness); actionHexPairs = new QAction(tr("Bytes as pairs"), this); actionHexPairs->setCheckable(true); @@ -125,14 +127,14 @@ HexWidget::HexWidget(QWidget *parent) actionComment = new QAction(tr("Add Comment"), this); actionComment->setShortcutContext(Qt::ShortcutContext::WidgetWithChildrenShortcut); actionComment->setShortcut(Qt::Key_Semicolon); - connect(actionComment, &QAction::triggered, this, &HexWidget::on_actionAddComment_triggered); + connect(actionComment, &QAction::triggered, this, &HexWidget::onActionAddCommentTriggered); addAction(actionComment); // delete comment option actionDeleteComment = new QAction(tr("Delete Comment"), this); actionDeleteComment->setShortcutContext(Qt::ShortcutContext::WidgetWithChildrenShortcut); connect(actionDeleteComment, &QAction::triggered, this, - &HexWidget::on_actionDeleteComment_triggered); + &HexWidget::onActionDeleteCommentTriggered); addAction(actionDeleteComment); actionSelectRange = new QAction(tr("Select range"), this); @@ -183,8 +185,13 @@ HexWidget::HexWidget(QWidget *parent) connect(actionIncDec, &QAction::triggered, this, &HexWidget::w_increaseDecrease); actionsWriteOther.append(actionIncDec); + actionKeyboardEdit = new QAction(tr("Edit with keyboard"), this); + actionKeyboardEdit->setCheckable(true); + connect(actionKeyboardEdit, &QAction::triggered, this, &HexWidget::onKeyboardEditTriggered); + connect(actionKeyboardEdit, &QAction::toggled, this, &HexWidget::onKeyboardEditChanged); + connect(this, &HexWidget::selectionChanged, this, - [this](Selection selection) { actionCopy->setEnabled(!selection.empty); }); + [this](Selection newSelection) { actionCopy->setEnabled(!newSelection.empty); }); updateMetrics(); updateItemLength(); @@ -202,9 +209,10 @@ HexWidget::HexWidget(QWidget *parent) cursor.startBlinking(); updateColors(); -} -HexWidget::~HexWidget() {} + warningTimer.setSingleShot(true); + connect(&warningTimer, &QTimer::timeout, this, &HexWidget::hideWarningRect); +} void HexWidget::setMonospaceFont(const QFont &font) { @@ -228,6 +236,8 @@ void HexWidget::setItemSize(int nbytes) if (!values.contains(nbytes)) return; + finishEditingWord(); + itemByteLen = nbytes; if (itemByteLen > rowSizeBytes) { rowSizeBytes = itemByteLen; @@ -236,7 +246,12 @@ void HexWidget::setItemSize(int nbytes) actionsItemFormat.at(ItemFormatFloat)->setEnabled(nbytes >= 4); actionItemBigEndian->setEnabled(nbytes != 1); + refreshWordEditState(); + updateItemLength(); + if (!cursorOnAscii && cursor.address % itemByteLen) { + moveCursor(-int(cursor.address % itemByteLen)); + } fetchData(); updateCursorMeta(); @@ -245,6 +260,8 @@ void HexWidget::setItemSize(int nbytes) void HexWidget::setItemFormat(ItemFormat format) { + finishEditingWord(); + itemFormat = format; bool sizeEnabled = true; @@ -253,6 +270,8 @@ void HexWidget::setItemFormat(ItemFormat format) actionsItemSize.at(0)->setEnabled(sizeEnabled); actionsItemSize.at(1)->setEnabled(sizeEnabled); + refreshWordEditState(); + updateItemLength(); fetchData(); updateCursorMeta(); @@ -382,7 +401,7 @@ void HexWidget::selectRange(RVA start, RVA end) void HexWidget::clearSelection() { - setCursorAddr(cursor.address, false); + setCursorAddr(BasicCursor(cursor.address), false); emit selectionChanged(getSelection()); } @@ -393,7 +412,16 @@ HexWidget::Selection HexWidget::getSelection() void HexWidget::seek(uint64_t address) { - setCursorAddr(address); + if (!cursorOnAscii) { + // when other widget causes seek to the middle of word + // switch to ascii column which operates with byte positions + auto viewOffset = startAddress % itemByteLen; + auto addrOffset = address % itemByteLen; + if ((addrOffset + itemByteLen - viewOffset) % itemByteLen) { + setCursorOnAscii(true); + } + } + setCursorAddr(BasicCursor(address)); } void HexWidget::refresh() @@ -402,8 +430,9 @@ void HexWidget::refresh() viewport()->update(); } -void HexWidget::setItemEndianess(bool bigEndian) +void HexWidget::setItemEndianness(bool bigEndian) { + finishEditingWord(); itemBigEndian = bigEndian; updateCursorMeta(); // Update cached item character @@ -422,6 +451,7 @@ void HexWidget::updateColors() defColor = Config()->getColor("btext"); addrColor = Config()->getColor("func_var_addr"); diffColor = Config()->getColor("graph.diff.unmatch"); + warningColor = QColor("red"); updateCursorMeta(); viewport()->update(); @@ -450,6 +480,11 @@ void HexWidget::paintEvent(QPaintEvent *event) drawItemArea(painter); drawAsciiArea(painter); + if (warningRectVisible) { + painter.setPen(warningColor); + painter.drawRect(warningRect); + } + if (!cursorEnabled) return; @@ -467,6 +502,11 @@ void HexWidget::updateWidth() horizontalScrollBar()->setSingleStep(charWidth); } +bool HexWidget::isFixedWidth() const +{ + return itemFormat == ItemFormatHex || itemFormat == ItemFormatOct; +} + void HexWidget::resizeEvent(QResizeEvent *event) { int oldByteCount = bytesPerScreen(); @@ -526,22 +566,98 @@ void HexWidget::mousePressEvent(QMouseEvent *event) if (event->button() == Qt::LeftButton) { bool selectingData = itemArea.contains(pos); bool selecting = selectingData || asciiArea.contains(pos); + bool holdingShift = event->modifiers() == Qt::ShiftModifier; + + // move cursor within actively edited item + if (selectingData && !holdingShift && editWordState >= EditWordState::WriteNotEdited) { + + auto editWordArea = itemRectangle(cursor.address - startAddress); + if (editWordArea.contains(pos)) { + int wordOffset = 0; + auto cursorPosition = screenPosToAddr(pos, false, &wordOffset); + if (cursorPosition.address == cursor.address + // allow selecting after last character only when cursor limited to current word + && (wordOffset < editWord.length() + || navigationMode == HexNavigationMode::WordChar)) { + editWordPos = std::max(0, wordOffset); + editWordPos = std::min(editWordPos, editWord.length()); + + if (isFixedWidth()) { + updatingSelection = true; + auto selectionCursor = cursorPosition; + if (editWordPos > itemCharLen / 2) { + selectionCursor += itemByteLen; + } + selection.init(selectionCursor); + } + + viewport()->update(); + return; + } + } + } + + // cursor within any item if the mode allows + if (selectingData && !holdingShift && editWordState >= EditWordState::WriteNotStarted + && navigationMode == HexNavigationMode::AnyChar) { + updatingSelection = true; + setCursorOnAscii(false); + int wordOffset = 0; + auto cursorPosition = screenPosToAddr(pos, false, &wordOffset); + finishEditingWord(); + if (isFixedWidth() && wordOffset >= itemCharLen - itemPrefixLen) { + wordOffset = 0; + cursorPosition += itemByteLen; + } + setCursorAddr(cursorPosition, holdingShift); + auto selectionPosition = currentAreaPosToAddr(pos, true); + selection.init(selectionPosition); + emit selectionChanged(getSelection()); + + if (wordOffset > 0) { + startEditWord(); + editWordPos = std::min(wordOffset, editWord.length() - 1); + } + viewport()->update(); + return; + } + if (selecting) { + finishEditingWord(); + updatingSelection = true; setCursorOnAscii(!selectingData); auto cursorPosition = currentAreaPosToAddr(pos, true); - setCursorAddr(cursorPosition, event->modifiers() == Qt::ShiftModifier); + setCursorAddr(cursorPosition, holdingShift); viewport()->update(); } } } +void HexWidget::mouseDoubleClickEvent(QMouseEvent *event) +{ + QPoint pos(event->pos()); + pos.rx() += horizontalScrollBar()->value(); + + if (event->button() == Qt::LeftButton && !isFixedWidth() + && editWordState == EditWordState::WriteNotStarted && itemArea.contains(pos)) { + int wordOffset = 0; + auto cursorPosition = screenPosToAddr(pos, false, &wordOffset); + setCursorAddr(cursorPosition, false); + startEditWord(); + int padding = std::max(0, itemCharLen - editWord.length()); + editWordPos = std::max(0, wordOffset - padding); + editWordPos = std::min(editWordPos, editWord.length()); + } +} + void HexWidget::mouseReleaseEvent(QMouseEvent *event) { if (event->button() == Qt::LeftButton) { if (selection.isEmpty()) { - selection.init(cursor.address); + selection.init(BasicCursor(cursor.address)); cursorEnabled = true; + viewport()->update(); } updatingSelection = false; } @@ -560,13 +676,14 @@ void HexWidget::wheelEvent(QWheelEvent *event) startAddress = 0; } else if (delta > 0 && data->maxIndex() < static_cast(bytesPerScreen())) { startAddress = 0; + } else if ((data->maxIndex() - startAddress) + <= static_cast(bytesPerScreen() + delta - 1)) { + startAddress = (data->maxIndex() - bytesPerScreen()) + 1; } else { startAddress += delta; } + fetchData(); - if ((data->maxIndex() - startAddress) <= static_cast(bytesPerScreen() + delta - 1)) { - startAddress = (data->maxIndex() - bytesPerScreen()) + 1; - } if (cursor.address >= startAddress && cursor.address <= lastVisibleAddr()) { /* Don't enable cursor blinking if selection isn't empty */ cursorEnabled = selection.isEmpty(); @@ -577,6 +694,304 @@ void HexWidget::wheelEvent(QWheelEvent *event) viewport()->update(); } +bool HexWidget::validCharForEdit(QChar digit) +{ + switch (itemFormat) { + case ItemFormatHex: + return (digit >= '0' && digit <= '9') || (digit >= 'a' && digit <= 'f') + || (digit >= 'A' && digit <= 'F'); + case ItemFormatOct: { + if (editWordPos > 0) { + return (digit >= '0' && digit <= '7'); + } else { + int bitsInMSD = (itemByteLen * 8) % 3; + int biggestDigit = (1 << bitsInMSD) - 1; + return digit >= '0' && digit <= char('0' + biggestDigit); + } + } + case ItemFormatDec: + return (digit >= '0' && digit <= '9'); + case ItemFormatSignedDec: + return (digit >= '0' && digit <= '9') || digit == '-'; + case ItemFormatFloat: + return (digit >= '0' && digit <= '9') || digit == '-' || digit == '+' || digit == '.' + || digit == ',' || digit == '+' || digit == 'e' || digit == 'E' || digit == 'i' + || digit == 'n' || digit == 'f' || digit == 'I' || digit == 'N' || digit == 'F' + || digit == 'a' || digit == 'A'; + } + + return false; +} + +void HexWidget::movePrevEditCharAny() +{ + if (!selection.isEmpty()) { + clearSelection(); + } + editWordPos -= 1; + if (editWordPos < 0) { + finishEditingWord(); + if (moveCursor(-itemByteLen, false, OverflowMove::Ignore)) { + startEditWord(); + editWordPos = editWord.length() - 1; + } + } + viewport()->update(); +} + +void HexWidget::typeOverwriteModeChar(QChar c) +{ + if (editWordState < EditWordState::WriteNotEdited || !isFixedWidth()) { + return; + } + editWord[editWordPos] = c; + editWordPos++; + editWordState = EditWordState::WriteEdited; + if (editWordPos >= editWord.length()) { + finishEditingWord(); + bool moved = moveCursor(itemByteLen, false, OverflowMove::Ignore); + startEditWord(); + if (!moved) { + editWordPos = editWord.length() - 1; + } + } +} + +HexWidget::HexNavigationMode HexWidget::defaultNavigationMode() +{ + switch (editWordState) { + case EditWordState::Read: + return HexNavigationMode::Words; + case EditWordState::WriteNotStarted: + return isFixedWidth() ? HexNavigationMode::AnyChar : HexNavigationMode::Words; + case EditWordState::WriteNotEdited: + case EditWordState::WriteEdited: + return isFixedWidth() ? HexNavigationMode::AnyChar : HexNavigationMode::WordChar; + } + return HexNavigationMode::Words; +} + +void HexWidget::refreshWordEditState() +{ + navigationMode = defaultNavigationMode(); +} + +bool HexWidget::handleAsciiWrite(QKeyEvent *event) +{ + if (!cursorOnAscii || !canKeyboardEdit()) { + return false; + } + if (event->key() == Qt::Key_Backspace || event->matches(QKeySequence::Backspace)) { + if (!selection.isEmpty()) { + writeZeros(selection.start(), selection.size()); + } else { + moveCursor(-1, false); + writeZeros(cursor.address, 1); + } + return true; + } + if (event->key() == Qt::Key_Delete || event->matches(QKeySequence::Delete)) { + if (!selection.isEmpty()) { + writeZeros(selection.start(), selection.size()); + } else { + writeZeros(cursor.address, 1); + moveCursor(1, false); + } + return true; + } + QString text; + if (event->matches(QKeySequence::Paste)) { + text = QApplication::clipboard()->text(); + if (text.length() <= 0) { + return false; + } + } else { + text = event->text(); + if (text.length() <= 0) { + return false; + } + QChar c = text[0]; + if (c <= '\x1f' || c == '\x7f') { + return false; + } + } + + auto bytes = text.toUtf8(); // TODO:#3028 use selected text encoding + auto address = getLocationAddress(); + clearSelection(); + data->write(reinterpret_cast(bytes.data()), address, bytes.length()); + seek(address + bytes.length()); + viewport()->update(); + return true; +} + +bool HexWidget::handleNumberWrite(QKeyEvent *event) +{ + if (editWordState < EditWordState::WriteNotStarted) { + return false; + } + bool overwrite = isFixedWidth(); + auto keyText = event->text(); + bool editingWord = editWordState >= EditWordState::WriteNotEdited; + if (keyText.length() > 0 && validCharForEdit(keyText[0])) { + if (!selection.isEmpty()) { + setCursorAddr(BasicCursor(selection.start())); + } + if (!editingWord) { + startEditWord(); + } + if (overwrite) { + typeOverwriteModeChar(keyText[0]); + } else if (!editingWord /* && !overwrite */) { + editWord = keyText; + editWordPos = editWord.length(); + editWordState = EditWordState::WriteEdited; + } else if (itemFormat == ItemFormatFloat || editWord.length() < itemCharLen) { + editWord.insert(editWordPos, keyText); + editWordPos += keyText.length(); + editWordState = EditWordState::WriteEdited; + } + maybeFlushCharEdit(); + return true; + } + if (event->matches(QKeySequence::Paste) && (editingWord || overwrite)) { + QString text = QApplication::clipboard()->text(); + if (text.length() > 0) { + if (overwrite) { + startEditWord(); + for (QChar c : text) { + if (validCharForEdit(c)) { + typeOverwriteModeChar(c); + } + } + } else { + editWord.insert(editWordPos, text); + editWordPos += text.length(); + editWordState = EditWordState::WriteEdited; + } + } + maybeFlushCharEdit(); + return true; + } + if (editingWord) { + if (event->matches(QKeySequence::Cancel)) { + cancelEditedWord(); + return true; + } else if (event->key() == Qt::Key_Return || event->key() == Qt::Key_Enter) { + bool needToAdvance = + !(editWordPos == 0 && overwrite && editWordState < EditWordState::WriteEdited); + if (finishEditingWord(false) && needToAdvance) { + moveCursor(itemByteLen); + } + return true; + } + } + if (event->matches(QKeySequence::Delete)) { + if (!selection.isEmpty()) { + writeZeros(selection.start(), selection.size()); + clearSelection(); + } else { + startEditWord(); + if (overwrite) { + typeOverwriteModeChar('0'); + } else if (editWordPos < editWord.length()) { + editWord.remove(editWordPos, 1); + editWordState = EditWordState::WriteEdited; + } + } + maybeFlushCharEdit(); + return true; + } + if (event->matches(QKeySequence::DeleteEndOfWord) && selection.isEmpty()) { + startEditWord(); + if (overwrite) { + for (int i = editWordPos; i < editWord.length(); i++) { + typeOverwriteModeChar('0'); + } + } else if (editWordPos < editWord.length()) { + editWord.remove(editWordPos, editWord.length()); + editWordState = EditWordState::WriteEdited; + } + maybeFlushCharEdit(); + return true; + } + if (event->matches(QKeySequence::DeleteStartOfWord) && selection.isEmpty()) { + if (!editingWord || (overwrite && editWordPos == 0)) { + if (!moveCursor(-itemByteLen, false, OverflowMove::Ignore)) { + return false; + } + startEditWord(); + editWordPos = editWord.length(); + } + if (overwrite) { + while (editWordPos > 0) { + editWordPos--; + editWord[editWordPos] = '0'; + } + editWordState = EditWordState::WriteEdited; + maybeFlushCharEdit(); + return true; + } else { + if (editWordPos > 0) { + editWord.remove(0, editWordPos); + editWordState = EditWordState::WriteEdited; + editWordPos = 0; + maybeFlushCharEdit(); + return true; + } + } + return false; + } + + if (event->key() == Qt::Key_Backspace) { + if (!selection.isEmpty()) { + writeZeros(selection.start(), selection.size()); + clearSelection(); + setCursorAddr(BasicCursor(selection.start()), false); + return true; + } else { + if (!editingWord || (overwrite && editWordPos == 0)) { + if (!moveCursor(-itemByteLen, false, OverflowMove::Ignore)) { + return false; + } + startEditWord(); + editWordPos = editWord.length(); + } + if (editWordPos > 0) { + editWordPos -= 1; + if (overwrite) { + editWord[editWordPos] = '0'; + } else { + editWord.remove(editWordPos, 1); + } + editWordState = EditWordState::WriteEdited; + maybeFlushCharEdit(); + return true; + } + } + return false; + } + + return false; +} + +bool HexWidget::event(QEvent *event) +{ + // prefer treating keys like 's' 'g' '.' as typing input instead of global shortcuts + if (event->type() == QEvent::ShortcutOverride) { + auto keyEvent = static_cast(event); + auto modifiers = keyEvent->modifiers(); + if ((modifiers == Qt::NoModifier || modifiers == Qt::ShiftModifier + || modifiers == Qt::KeypadModifier) + && keyEvent->key() < Qt::Key_Escape && canKeyboardEdit()) { + keyEvent->accept(); + return true; + } + } + + return QScrollArea::event(event); +} + void HexWidget::keyPressEvent(QKeyEvent *event) { bool select = false; @@ -591,26 +1006,126 @@ void HexWidget::keyPressEvent(QKeyEvent *event) } return false; }; - if (moveOrSelect(QKeySequence::MoveToNextLine, QKeySequence::SelectNextLine)) { - moveCursor(itemRowByteLen(), select); - } else if (moveOrSelect(QKeySequence::MoveToPreviousLine, QKeySequence::SelectPreviousLine)) { - moveCursor(-itemRowByteLen(), select); - } else if (moveOrSelect(QKeySequence::MoveToNextChar, QKeySequence::SelectNextChar)) { - moveCursor(cursorOnAscii ? 1 : itemByteLen, select); - } else if (moveOrSelect(QKeySequence::MoveToPreviousChar, QKeySequence::SelectPreviousChar)) { - moveCursor(cursorOnAscii ? -1 : -itemByteLen, select); - } else if (moveOrSelect(QKeySequence::MoveToNextPage, QKeySequence::SelectNextPage)) { - moveCursor(bytesPerScreen(), select); - } else if (moveOrSelect(QKeySequence::MoveToPreviousPage, QKeySequence::SelectPreviousPage)) { - moveCursor(-bytesPerScreen(), select); - } else if (moveOrSelect(QKeySequence::MoveToStartOfLine, QKeySequence::SelectStartOfLine)) { - int linePos = int((cursor.address % itemRowByteLen()) - (startAddress % itemRowByteLen())); - moveCursor(-linePos, select); - } else if (moveOrSelect(QKeySequence::MoveToEndOfLine, QKeySequence::SelectEndOfLine)) { - int linePos = int((cursor.address % itemRowByteLen()) - (startAddress % itemRowByteLen())); - moveCursor(itemRowByteLen() - linePos, select); + + if (canKeyboardEdit()) { + if (handleAsciiWrite(event)) { + viewport()->update(); + return; + } + if (editWordState >= EditWordState::WriteNotStarted && !cursorOnAscii) { + if (handleNumberWrite(event)) { + viewport()->update(); + return; + } + } + } + + if (cursorOnAscii || navigationMode == HexNavigationMode::Words + || navigationMode == HexNavigationMode::AnyChar) { + if (moveOrSelect(QKeySequence::MoveToNextPage, QKeySequence::SelectNextPage)) { + moveCursor(bytesPerScreen(), select); + } else if (moveOrSelect(QKeySequence::MoveToPreviousPage, + QKeySequence::SelectPreviousPage)) { + moveCursor(-bytesPerScreen(), select); + } else if (moveOrSelect(QKeySequence::MoveToStartOfLine, QKeySequence::SelectStartOfLine)) { + int linePos = + int((cursor.address % itemRowByteLen()) - (startAddress % itemRowByteLen())); + moveCursor(-linePos, select); + } else if (moveOrSelect(QKeySequence::MoveToEndOfLine, QKeySequence::SelectEndOfLine)) { + int linePos = + int((cursor.address % itemRowByteLen()) - (startAddress % itemRowByteLen())); + moveCursor(itemRowByteLen() - linePos, select); + } + } + + if (navigationMode == HexNavigationMode::Words || cursorOnAscii) { + if (moveOrSelect(QKeySequence::MoveToNextLine, QKeySequence::SelectNextLine)) { + moveCursor(itemRowByteLen(), select, OverflowMove::Ignore); + } else if (moveOrSelect(QKeySequence::MoveToPreviousLine, + QKeySequence::SelectPreviousLine)) { + moveCursor(-itemRowByteLen(), select, OverflowMove::Ignore); + } else if (moveOrSelect(QKeySequence::MoveToNextChar, QKeySequence::SelectNextChar) + || moveOrSelect(QKeySequence::MoveToNextWord, QKeySequence::SelectNextWord)) { + moveCursor(cursorOnAscii ? 1 : itemByteLen, select); + } else if (moveOrSelect(QKeySequence::MoveToPreviousChar, QKeySequence::SelectPreviousChar) + || moveOrSelect(QKeySequence::MoveToPreviousWord, + QKeySequence::SelectPreviousWord)) { + moveCursor(cursorOnAscii ? -1 : -itemByteLen, select); + } + } else if (navigationMode == HexNavigationMode::AnyChar && !cursorOnAscii) { + if (moveOrSelect(QKeySequence::MoveToNextLine, QKeySequence::SelectNextLine)) { + moveCursorKeepEditOffset(itemRowByteLen(), select, OverflowMove::Ignore); + } else if (moveOrSelect(QKeySequence::MoveToPreviousLine, + QKeySequence::SelectPreviousLine)) { + moveCursorKeepEditOffset(-itemRowByteLen(), select, OverflowMove::Ignore); + } else if (moveOrSelect(QKeySequence::MoveToNextChar, QKeySequence::SelectNextChar)) { + if (select) { + moveCursor(itemByteLen, select); + } else { + if (!selection.isEmpty()) { + clearSelection(); + } + if (editWordState == EditWordState::WriteNotStarted) { + startEditWord(); + } + editWordPos += 1; + if (editWordPos >= editWord.length()) { + bool moved = moveCursor(itemByteLen, false, OverflowMove::Ignore); + startEditWord(); + if (!moved) { + editWordPos = editWord.length() - 1; + } + } + } + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToPreviousChar)) { + movePrevEditCharAny(); + } else if (event->matches(QKeySequence::SelectPreviousChar)) { + moveCursor(-itemByteLen, true); + } else if (moveOrSelect(QKeySequence::MoveToNextWord, QKeySequence::SelectNextWord)) { + moveCursor(itemByteLen, select); + } else if (event->matches(QKeySequence::MoveToPreviousWord)) { + if (editWordPos > 0) { + editWordPos = 0; + viewport()->update(); + } else { + moveCursor(-itemByteLen, false); + } + } else if (event->matches(QKeySequence::SelectPreviousWord)) { + moveCursor(-itemByteLen, true); + } + } else if (navigationMode == HexNavigationMode::WordChar) { + if (event->matches(QKeySequence::MoveToNextChar)) { + editWordPos = std::min(editWord.length(), editWordPos + 1); + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToPreviousChar)) { + editWordPos = std::max(0, editWordPos - 1); + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToStartOfLine)) { + editWordPos = 0; + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToEndOfLine)) { + editWordPos = editWord.length(); + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToPreviousWord)) { + if (editWordPos > 0) { + editWordPos = 0; + } else { + moveCursor(-itemByteLen, select); + startEditWord(); + } + viewport()->update(); + } else if (event->matches(QKeySequence::MoveToNextWord)) { + if (editWordPos < editWord.length()) { + editWordPos = editWord.length(); + } else { + moveCursor(itemByteLen, select); + startEditWord(); + editWordPos = editWord.length(); + } + viewport()->update(); + } } - // viewport()->update(); } void HexWidget::contextMenuEvent(QContextMenuEvent *event) @@ -645,7 +1160,11 @@ void HexWidget::contextMenuEvent(QContextMenuEvent *event) actionComment->setText(tr("Edit Comment")); } - QMenu *menu = new QMenu(); + if (!ioModesController.canWrite()) { + actionKeyboardEdit->setChecked(false); + } + + auto *menu = new QMenu(this); QMenu *sizeMenu = menu->addMenu(tr("Item size:")); sizeMenu->addActions(actionsItemSize); QMenu *formatMenu = menu->addMenu(tr("Item format:")); @@ -657,6 +1176,8 @@ void HexWidget::contextMenuEvent(QContextMenuEvent *event) writeMenu->addActions(actionsWriteString); writeMenu->addSeparator(); writeMenu->addActions(actionsWriteOther); + menu->addAction(actionKeyboardEdit); + menu->addSeparator(); menu->addAction(actionCopy); disableOutsideSelectionActions(mouseOutsideSelection); @@ -700,23 +1221,20 @@ void HexWidget::copy() void HexWidget::copyAddress() { - uint64_t addr = cursor.address; - if (!selection.isEmpty()) { - addr = selection.start(); - } + uint64_t addr = getLocationAddress(); QClipboard *clipboard = QApplication::clipboard(); clipboard->setText(RzAddressString(addr)); } // slot for add comment action -void HexWidget::on_actionAddComment_triggered() +void HexWidget::onActionAddCommentTriggered() { uint64_t addr = cursor.address; CommentsDialog::addOrEditComment(addr, this); } // slot for deleting comment action -void HexWidget::on_actionDeleteComment_triggered() +void HexWidget::onActionDeleteCommentTriggered() { uint64_t addr = cursor.address; Core()->delComment(addr); @@ -731,6 +1249,20 @@ void HexWidget::onRangeDialogAccepted() selectRange(rangeDialog.getStartAddress(), rangeDialog.getEndAddress()); } +void HexWidget::writeZeros(uint64_t address, uint64_t length) +{ + const uint64_t MAX_BUFFER = 1024; + std::vector zeroes(std::min(MAX_BUFFER, length), 0); + while (length > zeroes.size()) { + data->write(zeroes.data(), address, zeroes.size()); + address += zeroes.size(); + length -= zeroes.size(); + } + if (length > 0) { + data->write(zeroes.data(), address, length); + } +} + void HexWidget::w_writeString() { if (!ioModesController.prepareForWriting()) { @@ -825,10 +1357,7 @@ void HexWidget::w_writeZeros() return; } { - RzCoreLocked core(Core()); - auto *buf = (uint8_t *)calloc(len, sizeof(uint8_t)); - rz_core_write_at(core, getLocationAddress(), buf, len); - free(buf); + writeZeros(getLocationAddress(), len); } refresh(); } @@ -964,15 +1493,42 @@ void HexWidget::w_writeCString() refresh(); } +void HexWidget::onKeyboardEditTriggered(bool enabled) +{ + if (!enabled) { + return; + } + if (!ioModesController.prepareForWriting()) { + actionKeyboardEdit->setChecked(false); + } +} + +void HexWidget::onKeyboardEditChanged(bool enabled) +{ + if (!enabled) { + finishEditingWord(); + navigationMode = HexNavigationMode::Words; + editWordState = EditWordState::Read; + } else { + editWordState = EditWordState::WriteNotStarted; + navigationMode = defaultNavigationMode(); + } + updateCursorMeta(); + viewport()->update(); +} + void HexWidget::updateItemLength() { itemPrefixLen = 0; + itemPrefix.clear(); switch (itemFormat) { case ItemFormatHex: itemCharLen = 2 * itemByteLen; - if (itemByteLen > 1 && showExHex) + if (itemByteLen > 1 && showExHex) { itemPrefixLen = hexPrefix.length(); + itemPrefix = hexPrefix; + } break; case ItemFormatOct: itemCharLen = (itemByteLen * 8 + 3) / 3; @@ -1056,7 +1612,13 @@ void HexWidget::drawCursor(QPainter &painter, bool shadow) QPen pen(Qt::gray); pen.setStyle(Qt::DashLine); painter.setPen(pen); - shadowCursor.screenPos.setWidth(cursorOnAscii ? itemWidth() : charWidth); + qreal shadowWidth = charWidth; + if (cursorOnAscii) { + shadowWidth = itemWidth(); + } else if (editWordState >= EditWordState::WriteNotEdited) { + shadowWidth = itemByteLen * charWidth; + } + shadowCursor.screenPos.setWidth(shadowWidth); painter.drawRect(shadowCursor.screenPos); painter.setPen(Qt::SolidLine); } @@ -1102,20 +1664,26 @@ void HexWidget::drawItemArea(QPainter &painter) fillSelectionBackground(painter); + bool haveEditWord = false; + QRectF editWordRect; + QColor editWordColor; + uint64_t itemAddr = startAddress; for (int line = 0; line < visibleLines; ++line) { itemRect.moveLeft(itemArea.left()); for (int j = 0; j < itemColumns; ++j) { for (int k = 0; k < itemGroupSize && itemAddr <= data->maxIndex(); ++k, itemAddr += itemByteLen) { + itemString = renderItem(itemAddr - startAddress, &itemColor); if (!getFlagsAndComment(itemAddr).isEmpty()) { QColor markerColor(borderColor); markerColor.setAlphaF(0.5); - const auto shape = rangePolygons(itemAddr, itemAddr, false)[0]; painter.setPen(markerColor); - painter.drawPolyline(shape); + for (const auto &shape : rangePolygons(itemAddr, itemAddr, false)) { + painter.drawPolyline(shape); + } } if (selection.contains(itemAddr) && !cursorOnAscii) { itemColor = palette().highlightedText().color(); @@ -1123,19 +1691,53 @@ void HexWidget::drawItemArea(QPainter &painter) if (isItemDifferentAt(itemAddr)) { itemColor.setRgb(diffColor.rgb()); } - painter.setPen(itemColor); - painter.drawText(itemRect, Qt::AlignVCenter, itemString); - itemRect.translate(itemWidth(), 0); - if (cursor.address == itemAddr) { - auto &itemCursor = cursorOnAscii ? shadowCursor : cursor; - itemCursor.cachedChar = itemString.at(0); + + if (editWordState <= EditWordState::WriteNotStarted || cursor.address != itemAddr) { + painter.setPen(itemColor); + painter.drawText(itemRect, Qt::AlignVCenter, itemString); + itemRect.translate(itemWidth(), 0); + if (cursor.address == itemAddr) { + auto &itemCursor = cursorOnAscii ? shadowCursor : cursor; + int itemCharPos = 0; + if (editWordState > EditWordState::Read) { + itemCharPos += itemPrefixLen; + } + if (itemCharPos < itemString.length()) { + itemCursor.cachedChar = itemString.at(itemCharPos); + } else { + itemCursor.cachedChar = ' '; + } + itemCursor.cachedColor = itemColor; + } + } else { + haveEditWord = true; + editWordRect = itemRect; + editWordColor = itemColor; + + auto &itemCursor = cursor; + itemCursor.cachedChar = + editWordPos < editWord.length() ? editWord[editWordPos] : QChar(' '); itemCursor.cachedColor = itemColor; + itemCursor.screenPos.moveTopLeft(itemRect.topLeft()); + itemCursor.screenPos.translate(charWidth * (editWordPos + itemPrefixLen), 0); + + itemRect.translate(itemWidth(), 0); } } itemRect.translate(columnSpacingWidth(), 0); } itemRect.translate(0, lineHeight); } + if (haveEditWord) { + auto length = std::max(itemCharLen, editWord.length()); + auto rect = editWordRect; + rect.setWidth(length * charWidth); + painter.fillRect(rect, backgroundColor); + + painter.setPen(editWordColor); + editWordRect.setWidth(4000); + painter.drawText(editWordRect, Qt::AlignVCenter | Qt::AlignLeft, itemPrefix + editWord); + } painter.setPen(borderColor); @@ -1222,12 +1824,16 @@ QVector HexWidget::rangePolygons(RVA start, RVA last, bool ascii) auto startRect = getRectangle(startOffset); auto endRect = getRectangle(endOffset); + bool startJagged = false; + bool endJagged = false; if (!ascii) { if (int startFraction = startOffset % itemByteLen) { startRect.setLeft(startRect.left() + startFraction * startRect.width() / itemByteLen); + startJagged = true; } if (int endFraction = itemByteLen - 1 - (endOffset % itemByteLen)) { endRect.setRight(endRect.right() - endFraction * endRect.width() / itemByteLen); + endJagged = true; } } if (endOffset - startOffset + 1 <= rowSizeBytes) { @@ -1263,6 +1869,33 @@ QVector HexWidget::rangePolygons(RVA start, RVA last, bool ascii) shape << shape.first(); // close the shape parts.push_back(shape); } + if (!ascii && (startJagged || endJagged) && parts.length() >= 1) { + + QPolygonF top; + top.reserve(3); + top << QPointF(0, 0) << QPointF(charWidth, lineHeight / 3) << QPointF(0, lineHeight / 2); + QPolygonF bottom; + bottom.reserve(3); + bottom << QPointF(0, lineHeight / 2) << QPointF(-charWidth, 2 * lineHeight / 3) + << QPointF(0, lineHeight); + + // small adjustment to make sure that edges don't overlap with rect edges, QPolygonF doesn't + // handle it properly + QPointF adjustment(charWidth / 16, 0); + top.translate(-adjustment); + bottom.translate(adjustment); + + if (startJagged) { + auto movedTop = top.translated(startRect.topLeft()); + auto movedBottom = bottom.translated(startRect.topLeft()); + parts[0] = parts[0].subtracted(movedTop).united(movedBottom); + } + if (endJagged) { + auto movedTop = top.translated(endRect.topRight()); + auto movedBottom = bottom.translated(endRect.topRight()); + parts.last() = parts.last().subtracted(movedBottom).united(movedTop); + } + } return parts; } @@ -1324,18 +1957,44 @@ void HexWidget::updateAreasHeight() asciiArea.setHeight(height); } -void HexWidget::moveCursor(int offset, bool select) +bool HexWidget::moveCursor(int offset, bool select, OverflowMove overflowMove) { - BasicCursor addr = cursor.address; - addr += offset; - if (addr.address > data->maxIndex()) { - addr.address = data->maxIndex(); + BasicCursor addr(cursor.address); + if (overflowMove == OverflowMove::Ignore) { + if (addr.moveChecked(offset)) { + if (addr.address > data->maxIndex()) { + addr.address = data->maxIndex(); + addr.pastEnd = true; + } + setCursorAddr(addr, select); + return true; + } + return false; + } else { + addr += offset; + if (addr.address > data->maxIndex()) { + addr.address = data->maxIndex(); + } + setCursorAddr(addr, select); + return true; + } +} + +void HexWidget::moveCursorKeepEditOffset(int byteOffset, bool select, OverflowMove overflowMove) +{ + int wordOffset = editWordPos; + moveCursor(byteOffset, select, overflowMove); + // preserve position within word when moving vertically in hex or oct modes + if (!cursorOnAscii && !select && wordOffset > 0 && navigationMode == HexNavigationMode::AnyChar + && editWordState > EditWordState::Read) { + startEditWord(); + editWordPos = wordOffset; } - setCursorAddr(addr, select); } void HexWidget::setCursorAddr(BasicCursor addr, bool select) { + finishEditingWord(); if (!select) { bool clearingSelection = !selection.isEmpty(); selection.init(addr); @@ -1345,6 +2004,9 @@ void HexWidget::setCursorAddr(BasicCursor addr, bool select) emit positionChanged(addr.address); cursor.address = addr.address; + if (!cursorOnAscii) { + cursor.address -= cursor.address % itemByteLen; + } /* Pause cursor repainting */ cursorEnabled = false; @@ -1361,7 +2023,9 @@ void HexWidget::setCursorAddr(BasicCursor addr, bool select) addressValue -= (addressValue % itemRowByteLen()); /* FIXME: handling Page Up/Down */ - if (addressValue == startAddress + bytesPerScreen()) { + uint64_t rowAfterVisibleAddress = startAddress + bytesPerScreen(); + if (addressValue == rowAfterVisibleAddress && addressValue > startAddress) { + // when pressing down add only one new row startAddress += itemRowByteLen(); } else { startAddress = addressValue; @@ -1410,6 +2074,10 @@ void HexWidget::updateCursorMeta() point += itemArea.topLeft(); pointAscii += asciiArea.topLeft(); + if (editWordState > EditWordState::Read && !cursorOnAscii) { + point.rx() += itemPrefixLen * charWidth; + } + cursor.screenPos.moveTopLeft(cursorOnAscii ? pointAscii : point); shadowCursor.screenPos.moveTopLeft(cursorOnAscii ? point : pointAscii); } @@ -1419,7 +2087,7 @@ void HexWidget::setCursorOnAscii(bool ascii) cursorOnAscii = ascii; } -const QColor HexWidget::itemColor(uint8_t byte) +QColor HexWidget::itemColor(uint8_t byte) { QColor color(defColor); @@ -1547,7 +2215,7 @@ QString HexWidget::renderItem(int offset, QColor *color) item = QString("%1").arg(itemVal.toLongLong(), itemLen, 10); break; case ItemFormatFloat: - item = QString("%1").arg(itemVal.toDouble(), itemLen); + item = QString("%1").arg(itemVal.toDouble(), itemLen, 'g', itemByteLen == 4 ? 6 : 15); break; } @@ -1588,13 +2256,191 @@ QString HexWidget::getFlagsAndComment(uint64_t address) return metaData; } +bool HexWidget::canKeyboardEdit() +{ + return ioModesController.canWrite() && actionKeyboardEdit->isChecked(); +} + +template +static bool checkRange(BigValue v) +{ + return v >= std::numeric_limits::min() && v <= std::numeric_limits::max(); +} + +template +static bool checkAndWrite(BigInteger value, uint8_t *buf, bool littleEndian) +{ + if (!checkRange(value)) { + return false; + } + if (littleEndian) { + qToLittleEndian((T)value, buf); + } else { + qToBigEndian((T)value, buf); + } + return true; +} + +template +static bool checkAndWriteWithSign(const QVariant &value, uint8_t *buf, bool isSigned, + bool littleEndian) +{ + if (isSigned) { + return checkAndWrite(value.toLongLong(), buf, littleEndian); + } else { + return checkAndWrite(value.toULongLong(), buf, littleEndian); + } +} + +bool HexWidget::parseWord(QString word, uint8_t *buf, size_t bufferSize) const +{ + bool parseOk = false; + if (bufferSize < size_t(itemByteLen)) { + return false; + } + if (itemFormat == ItemFormatFloat) { + if (itemByteLen == 4) { + float value = word.toFloat(&parseOk); + if (!parseOk) { + return false; + } + if (itemBigEndian) { + rz_write_be_float(buf, value); + } else { + rz_write_le_float(buf, value); + } + return true; + } else if (itemByteLen == 8) { + double value = word.toDouble(&parseOk); + if (!parseOk) { + return false; + } + if (itemBigEndian) { + rz_write_be_double(buf, value); + } else { + rz_write_le_double(buf, value); + } + return true; + } + return false; + } else { + QVariant value; + bool isSigned = false; + switch (itemFormat) { + case ItemFormatHex: + value = word.toULongLong(&parseOk, 16); + break; + case ItemFormatOct: + value = word.toULongLong(&parseOk, 8); + break; + case ItemFormatDec: + value = word.toULongLong(&parseOk, 10); + break; + case ItemFormatSignedDec: + isSigned = true; + value = word.toLongLong(&parseOk, 10); + break; + default: + break; + } + if (!parseOk) { + return false; + } + + switch (itemByteLen) { + case 1: + return checkAndWriteWithSign(value, buf, isSigned, !itemBigEndian); + case 2: + return checkAndWriteWithSign(value, buf, isSigned, !itemBigEndian); + case 4: + return checkAndWriteWithSign(value, buf, isSigned, !itemBigEndian); + case 8: + return checkAndWriteWithSign(value, buf, isSigned, !itemBigEndian); + } + } + return false; +} + +bool HexWidget::flushCurrentlyEditedWord() +{ + if (editWordState < EditWordState::WriteEdited) { + return true; + } + uint8_t buf[16]; + if (parseWord(editWord, buf, sizeof(buf))) { + data->write(buf, cursor.address, itemByteLen); + return true; + } + editWordState = EditWordState::WriteNotEdited; + return false; +} + +bool HexWidget::finishEditingWord(bool force) +{ + if (editWordState == EditWordState::WriteEdited) { + if (!flushCurrentlyEditedWord() && !force) { + qWarning() << "Not a valid number in current format or size" << editWord; + showWarningRect(itemRectangle(cursor.address - startAddress).adjusted(-1, -1, 1, 1)); + return false; + } + } + editWord.clear(); + editWordPos = 0; + editWordState = canKeyboardEdit() ? EditWordState::WriteNotStarted : EditWordState::Read; + navigationMode = defaultNavigationMode(); + return true; +} + +void HexWidget::cancelEditedWord() +{ + editWordPos = 0; + editWordState = canKeyboardEdit() ? EditWordState::WriteNotStarted : EditWordState::Read; + editWord.clear(); + navigationMode = defaultNavigationMode(); + updateCursorMeta(); + viewport()->update(); +} + +void HexWidget::maybeFlushCharEdit() +{ + if (editWordState < EditWordState::WriteEdited) { + return; + } + if ((itemFormat == ItemFormatHex && earlyEditFlush >= EarlyEditFlush::EditNibble) + || (isFixedWidth() && earlyEditFlush >= EarlyEditFlush::EditFixedWidthChar)) { + flushCurrentlyEditedWord(); + if (!flushCurrentlyEditedWord()) { + showWarningRect(itemRectangle(cursor.address - startAddress).adjusted(-1, -1, 1, 1)); + } + } + viewport()->update(); +} + +void HexWidget::startEditWord() +{ + if (!canKeyboardEdit()) { + return; + } + if (editWordState >= EditWordState::WriteNotEdited) { + return; + } + editWordPos = 0; + editWordState = EditWordState::WriteNotEdited; + navigationMode = defaultNavigationMode(); + editWord = renderItem(cursor.address - startAddress).trimmed(); + if (itemPrefixLen > 0) { + editWord = editWord.mid(itemPrefixLen); + } + viewport()->update(); +} + void HexWidget::fetchData() { data.swap(oldData); data->fetch(startAddress, bytesPerScreen()); } -BasicCursor HexWidget::screenPosToAddr(const QPoint &point, bool middle) const +BasicCursor HexWidget::screenPosToAddr(const QPoint &point, bool middle, int *wordOffset) const { QPointF pt = point - itemArea.topLeft(); @@ -1605,9 +2451,21 @@ BasicCursor HexWidget::screenPosToAddr(const QPoint &point, bool middle) const relativeAddress += column * itemGroupByteLen(); pt.rx() -= column * columnExWidth(); auto roundingOffset = middle ? itemWidth() / 2 : 0; - relativeAddress += static_cast((pt.x() + roundingOffset) / itemWidth()) * itemByteLen; + int posInGroup = static_cast((pt.x() + roundingOffset) / itemWidth()); + if (!middle) { + posInGroup = std::min(posInGroup, itemGroupSize - 1); + } + relativeAddress += posInGroup * itemByteLen; + pt.rx() -= posInGroup * itemWidth(); BasicCursor result(startAddress); result += relativeAddress; + + if (!middle && wordOffset != nullptr) { + int charPos = static_cast((pt.x() / charWidth) + 0.5); + charPos -= itemPrefixLen; + charPos = std::max(0, charPos); + *wordOffset = charPos; + } return result; } @@ -1679,3 +2537,17 @@ RVA HexWidget::getLocationAddress() { return !selection.isEmpty() ? selection.start() : cursor.address; } + +void HexWidget::hideWarningRect() +{ + warningRectVisible = false; + viewport()->update(); +} + +void HexWidget::showWarningRect(QRectF rect) +{ + warningRect = rect; + warningRectVisible = true; + warningTimer.start(WARNING_TIME_MS); + viewport()->update(); +} diff --git a/src/widgets/HexWidget.h b/src/widgets/HexWidget.h index b46f0574..103040c2 100644 --- a/src/widgets/HexWidget.h +++ b/src/widgets/HexWidget.h @@ -14,7 +14,7 @@ struct BasicCursor { uint64_t address; bool pastEnd; - BasicCursor(uint64_t pos) : address(pos), pastEnd(false) {} + explicit BasicCursor(uint64_t pos) : address(pos), pastEnd(false) {} BasicCursor() : address(0), pastEnd(false) {} BasicCursor &operator+=(int64_t offset) { @@ -35,6 +35,14 @@ struct BasicCursor *this += int64_t(offset); return *this; } + + bool moveChecked(int offset) + { + auto oldAddress = address; + *this += offset; + return address - oldAddress == uint64_t(offset); + } + BasicCursor &operator+=(uint64_t offset) { if (uint64_t(offset) > (UINT64_MAX - address)) { @@ -46,7 +54,10 @@ struct BasicCursor } return *this; } - bool operator<(const BasicCursor &r) { return address < r.address || (pastEnd < r.pastEnd); } + bool operator<(const BasicCursor &r) const + { + return address < r.address || (pastEnd < r.pastEnd); + } }; struct HexCursor @@ -74,9 +85,10 @@ struct HexCursor class AbstractData { public: - virtual ~AbstractData() {} + virtual ~AbstractData() = default; virtual void fetch(uint64_t addr, int len) = 0; virtual bool copy(void *out, uint64_t adr, size_t len) = 0; + virtual bool write(const uint8_t *in, uint64_t adr, size_t len) = 0; virtual uint64_t maxIndex() = 0; virtual uint64_t minIndex() = 0; }; @@ -86,7 +98,7 @@ class BufferData : public AbstractData public: BufferData() { m_buffer.fill(0, 1); } - BufferData(const QByteArray &buffer) + explicit BufferData(const QByteArray &buffer) { if (buffer.isEmpty()) { m_buffer.fill(0, 1); @@ -95,7 +107,7 @@ public: } } - ~BufferData() override {} + ~BufferData() override = default; void fetch(uint64_t, int) override {} @@ -109,6 +121,16 @@ public: return false; } + bool write(const uint8_t *in, uint64_t addr, size_t len) override + { + if (addr < static_cast(m_buffer.size()) + && (static_cast(m_buffer.size()) - addr) < len) { + memcpy(m_buffer.data() + addr, in, len); + return true; + } + return false; + } + uint64_t maxIndex() override { return m_buffer.size() - 1; } private: @@ -118,8 +140,8 @@ private: class MemoryData : public AbstractData { public: - MemoryData() {} - ~MemoryData() override {} + MemoryData() = default; + ~MemoryData() override = default; static constexpr size_t BLOCK_SIZE = 4096; void fetch(uint64_t address, int length) override @@ -144,10 +166,11 @@ public: bool copy(void *out, uint64_t addr, size_t len) override { - if (addr < m_firstBlockAddr || addr > m_lastValidAddr - || (m_lastValidAddr - addr + 1) - < len /* do not merge with last check to handle overflows */ - || m_blocks.isEmpty()) { + if (addr < m_firstBlockAddr + || addr > m_lastValidAddr + /* do not merge with previous check to handle overflows */ + || (m_lastValidAddr - addr + 1) < len || m_blocks.isEmpty()) { + memset(out, 0xff, len); return false; } @@ -165,9 +188,47 @@ public: return true; } - virtual uint64_t maxIndex() override { return m_lastValidAddr; } + void writeToCache(const uint8_t *in, uint64_t adr, size_t len) + { + if (adr < m_firstBlockAddr) { + uint64_t prefix = m_firstBlockAddr - adr; + if (prefix <= len) { + return; + } + in = in + prefix; + adr += prefix; + len -= prefix; + } + if (adr > m_lastValidAddr) { + return; + } + int offset = (int)(adr - m_firstBlockAddr); + int blockId = offset / BLOCK_SIZE; + int blockOffset = offset % BLOCK_SIZE; + while (len > 0 && blockId < m_blocks.size()) { + size_t l = BLOCK_SIZE - blockOffset; + l = std::min(l, len); + memcpy(m_blocks[blockId].data() + blockOffset, in, l); + len -= l; + blockOffset = 0; + adr += l; + in += l; + blockId += 1; + } + } - virtual uint64_t minIndex() override { return m_firstBlockAddr; } + bool write(const uint8_t *in, uint64_t adr, size_t len) override + { + RzCoreLocked core(Core()); + rz_core_write_at(core, adr, in, len); + writeToCache(in, adr, len); + emit Core()->instructionChanged(adr); + return true; + } + + uint64_t maxIndex() override { return std::numeric_limits::max(); } + + uint64_t minIndex() override { return m_firstBlockAddr; } private: QVector m_blocks; @@ -178,7 +239,11 @@ private: class HexSelection { public: - HexSelection() { m_empty = true; } + HexSelection() + { + m_empty = true; + m_start = m_end = 0; + } inline void init(BasicCursor addr) { @@ -189,7 +254,8 @@ public: void set(uint64_t start, uint64_t end) { m_empty = false; - m_init = m_start = start; + m_init = BasicCursor(start); + m_start = start; m_end = end; } @@ -219,7 +285,7 @@ public: bool contains(uint64_t pos) const { return !m_empty && m_start <= pos && pos <= m_end; } - uint64_t size() + uint64_t size() const { uint64_t size = 0; if (!isEmpty()) @@ -227,9 +293,9 @@ public: return size; } - inline bool isEmpty() { return m_empty; } - inline uint64_t start() { return m_start; } - inline uint64_t end() { return m_end; } + inline bool isEmpty() const { return m_empty; } + inline uint64_t start() const { return m_start; } + inline uint64_t end() const { return m_end; } private: BasicCursor m_init; @@ -244,7 +310,7 @@ class HexWidget : public QScrollArea public: explicit HexWidget(QWidget *parent = nullptr); - ~HexWidget(); + ~HexWidget() override = default; void setMonospaceFont(const QFont &font); @@ -258,10 +324,12 @@ public: ItemFormatFloat }; enum class ColumnMode { Fixed, PowerOf2 }; + enum class EditWordState { Read, WriteNotStarted, WriteNotEdited, WriteEdited }; + enum class HexNavigationMode { Words, WordChar, AnyChar }; void setItemSize(int nbytes); void setItemFormat(ItemFormat format); - void setItemEndianess(bool bigEndian); + void setItemEndianness(bool bigEndian); void setItemGroupSize(int size); /** * @brief Sets line size in bytes. @@ -292,7 +360,7 @@ public slots: void refresh(); void updateColors(); signals: - void selectionChanged(Selection selection); + void selectionChanged(HexWidget::Selection selection); void positionChanged(RVA start); protected: @@ -300,10 +368,12 @@ protected: void resizeEvent(QResizeEvent *event) override; void mouseMoveEvent(QMouseEvent *event) override; void mousePressEvent(QMouseEvent *event) override; + void mouseDoubleClickEvent(QMouseEvent *event) override; void mouseReleaseEvent(QMouseEvent *event) override; void wheelEvent(QWheelEvent *event) override; void keyPressEvent(QKeyEvent *event) override; void contextMenuEvent(QContextMenuEvent *event) override; + bool event(QEvent *event) override; private slots: void onCursorBlinked(); @@ -311,13 +381,13 @@ private slots: void copy(); void copyAddress(); void onRangeDialogAccepted(); - void on_actionAddComment_triggered(); - void on_actionDeleteComment_triggered(); + void onActionAddCommentTriggered(); + void onActionDeleteCommentTriggered(); // Write command slots void w_writeString(); void w_increaseDecrease(); - void w_writeBytes(); + void w_writeBytes(); void w_writeZeros(); void w_write64(); void w_writeRandom(); @@ -326,6 +396,9 @@ private slots: void w_writeWideString(); void w_writeCString(); + void onKeyboardEditTriggered(bool enabled); + void onKeyboardEditChanged(bool enabled); + private: void updateItemLength(); void updateCounts(); @@ -338,12 +411,15 @@ private: void updateMetrics(); void updateAreasPosition(); void updateAreasHeight(); - void moveCursor(int offset, bool select = false); + enum class OverflowMove { Clamp, Ignore }; + bool moveCursor(int offset, bool select = false, + OverflowMove overflowMove = OverflowMove::Clamp); + void moveCursorKeepEditOffset(int byteOffset, bool select, OverflowMove overflowMove); void setCursorAddr(BasicCursor addr, bool select = false); void updateCursorMeta(); void setCursorOnAscii(bool ascii); bool isItemDifferentAt(uint64_t address); - const QColor itemColor(uint8_t byte); + QColor itemColor(uint8_t byte); QVariant readItem(int offset, QColor *color = nullptr); QString renderItem(int offset, QColor *color = nullptr); QChar renderAscii(int offset, QColor *color = nullptr); @@ -359,12 +435,13 @@ private: /** * @brief Convert mouse position to address. * @param point mouse position in widget - * @param middle start next position from middle of symbol. Use middle=true for vertical cursror + * @param middle start next position from middle of symbol. Use middle=true for vertical cursor * position between symbols, middle=false for insert mode cursor and getting symbol under * cursor. * @return */ - BasicCursor screenPosToAddr(const QPoint &point, bool middle = false) const; + BasicCursor screenPosToAddr(const QPoint &point, bool middle = false, + int *wordOffset = nullptr) const; BasicCursor asciiPosToAddr(const QPoint &point, bool middle = false) const; BasicCursor currentAreaPosToAddr(const QPoint &point, bool middle = false) const; BasicCursor mousePosToAddr(const QPoint &point, bool middle = false) const; @@ -412,6 +489,27 @@ private: inline uint64_t lastVisibleAddr() const { return (startAddress - 1) + bytesPerScreen(); } const QRectF ¤tArea() const { return cursorOnAscii ? asciiArea : itemArea; } + bool isFixedWidth() const; + + bool canKeyboardEdit(); + bool flushCurrentlyEditedWord(); + bool finishEditingWord(bool force = true); + void maybeFlushCharEdit(); + void cancelEditedWord(); + void startEditWord(); + bool validCharForEdit(QChar digit); + void movePrevEditCharAny(); + void typeOverwriteModeChar(QChar c); + HexNavigationMode defaultNavigationMode(); + void refreshWordEditState(); + bool parseWord(QString word, uint8_t *buf, size_t bufferSize) const; + bool handleAsciiWrite(QKeyEvent *event); + bool handleNumberWrite(QKeyEvent *event); + + void writeZeros(uint64_t address, uint64_t length); + + void hideWarningRect(); + void showWarningRect(QRectF rect); bool cursorEnabled; bool cursorOnAscii; @@ -436,14 +534,13 @@ private: ItemFormat itemFormat; bool itemBigEndian; + QString itemPrefix; int visibleLines; uint64_t startAddress; qreal charWidth; - int byteWidth; qreal lineHeight; int addrCharLen; - int addrAreaWidth; QFont monospaceFont; bool showHeader; @@ -460,6 +557,7 @@ private: QColor b0x7fColor; QColor b0xffColor; QColor printableColor; + QColor warningColor; HexdumpRangeDialog rangeDialog; @@ -479,14 +577,30 @@ private: QAction *actionCopyAddress; QAction *actionComment; QAction *actionDeleteComment; - QAction *actionSetFlag; QAction *actionSelectRange; + QAction *actionKeyboardEdit; QList actionsWriteString; QList actionsWriteOther; std::unique_ptr oldData; std::unique_ptr data; IOModesController ioModesController; + + int editWordPos = 0; + QString editWord; + EditWordState editWordState = EditWordState::Read; + HexNavigationMode navigationMode = HexNavigationMode::Words; + enum class EarlyEditFlush { + OnFinish, + EditNibble, + EditFixedWidthChar, + /* AllFormats(not implemented) */ + }; + EarlyEditFlush earlyEditFlush = EarlyEditFlush::EditFixedWidthChar; + + bool warningRectVisible = false; + QRectF warningRect; + QTimer warningTimer; }; #endif // HEXWIDGET_H