File: DialogBinaryString.cpp

package info (click to toggle)
edb-debugger 1.3.0-2.2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 5,124 kB
  • sloc: cpp: 46,241; xml: 4,998; ansic: 3,088; sh: 52; asm: 33; makefile: 5
file content (125 lines) | stat: -rw-r--r-- 3,526 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/*
Copyright (C) 2006 - 2015 Evan Teran
                          evan.teran@gmail.com

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

#include "DialogBinaryString.h"
#include "DialogResults.h"
#include "IDebugger.h"
#include "IRegion.h"
#include "MemoryRegions.h"
#include "edb.h"
#include "util/Math.h"

#include <QListWidget>
#include <QMessageBox>
#include <QPushButton>
#include <QVector>
#include <cstring>

namespace BinarySearcherPlugin {

/**
 * @brief DialogBinaryString::DialogBinaryString
 * @param parent
 * @param f
 */
DialogBinaryString::DialogBinaryString(QWidget *parent, Qt::WindowFlags f)
	: QDialog(parent, f) {

	ui.setupUi(this);
	ui.progressBar->setValue(0);

	// NOTE(eteran): address issue #574
	ui.binaryString->setShowKeepSize(false);

	buttonFind_ = new QPushButton(QIcon::fromTheme("edit-find"), tr("Find"));
	connect(buttonFind_, &QPushButton::clicked, this, [this]() {
		buttonFind_->setEnabled(false);
		ui.progressBar->setValue(0);
		doFind();
		ui.progressBar->setValue(100);
		buttonFind_->setEnabled(true);
	});

	ui.buttonBox->addButton(buttonFind_, QDialogButtonBox::ActionRole);
}

/**
 * @brief DialogBinaryString::doFind
 */
void DialogBinaryString::doFind() {

	const QByteArray b = ui.binaryString->value();

	auto results = new DialogResults(this);

	const int sz = b.size();
	if (sz != 0) {
		edb::v1::memory_regions().sync();
		const QList<std::shared_ptr<IRegion>> regions = edb::v1::memory_regions().regions();
		const size_t page_size                        = edb::v1::debugger_core->pageSize();

		int i = 0;
		for (const std::shared_ptr<IRegion> &region : regions) {
			const size_t region_size = region->size();

			// a short circut for speading things up
			if (ui.chkSkipNoAccess->isChecked() && !region->accessible()) {
				ui.progressBar->setValue(util::percentage(++i, regions.size()));
				continue;
			}

			const size_t page_count      = region_size / page_size;
			const QVector<uint8_t> pages = edb::v1::read_pages(region->start(), page_count);

			if (!pages.isEmpty()) {

				const uint8_t *p               = &pages[0];
				const uint8_t *const pages_end = &pages[0] + region_size - sz;

				while (p < pages_end) {
					// compare values..
					if (std::memcmp(p, b.constData(), sz) == 0) {
						const edb::address_t addr  = p - &pages[0] + region->start();
						const edb::address_t align = 1 << (ui.cmbAlignment->currentIndex() + 1);

						if (!ui.chkAlignment->isChecked() || (addr % align) == 0) {
							results->addResult(DialogResults::RegionType::Data, addr);
						}
					}

					// update progress bar every 64KB
					if ((uint64_t(p) & 0xFFFF) == 0) {
						ui.progressBar->setValue(util::percentage(i, regions.size(), p - &pages[0], region_size));
					}

					++p;
				}
			}
			++i;
		}

		if (results->resultCount() == 0) {
			QMessageBox::information(nullptr, tr("No Results"), tr("No Results were found!"));
			delete results;
		} else {
			results->show();
		}
	}
}

}