Skip to content

Commit d983234

Browse files
authored
(feat) Added memory replace command. (#656)
* (feat) Added memory replace command. * refactor duplicate code.
1 parent 08df3fb commit d983234

File tree

5 files changed

+128
-1
lines changed

5 files changed

+128
-1
lines changed

agent/src/generic/memory.ts

+7
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,13 @@ export const search = (pattern: string, onlyOffsets: boolean = false): string[]
4545
return addresses.reduce((a, b) => a.concat(b));
4646
};
4747

48+
export const replace = (pattern: string, replace: number[]): string[] => {
49+
return search(pattern, true).map((match) => {
50+
write(match, replace);
51+
return match;
52+
})
53+
};
54+
4855
export const write = (address: string, value: number[]): void => {
4956
new NativePointer(address).writeByteArray(value);
5057
};

agent/src/rpc/memory.ts

+2
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,7 @@ export const memory = {
77
memoryListModules: (): Module[] => m.listModules(),
88
memoryListRanges: (protection: string): RangeDetails[] => m.listRanges(protection),
99
memorySearch: (pattern: string, onlyOffsets: boolean): string[] => m.search(pattern, onlyOffsets),
10+
memoryReplace: (pattern: string, replace: number[]): string[] => m.replace(pattern, replace),
1011
memoryWrite: (address: string, value: number[]): void => m.write(address, value),
12+
1113
};

objection/commands/memory.py

+63
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,30 @@ def _should_only_dump_offsets(args: list) -> bool:
3636
return '--offsets-only' in args
3737

3838

39+
def _is_string_pattern(args: list) -> bool:
40+
"""
41+
Checks if --string-pattern is in the list of tokens received form the
42+
command line.
43+
44+
:param args:
45+
:return:
46+
"""
47+
48+
return len(args) > 0 and '--string-pattern' in args
49+
50+
51+
def _is_string_replace(args: list) -> bool:
52+
"""
53+
Checks if --string-replace is in the list of tokens received form the
54+
command line.
55+
56+
:param args:
57+
:return:
58+
"""
59+
60+
return len(args) > 0 and '--string-replace' in args
61+
62+
3963
def _should_output_json(args: list) -> bool:
4064
"""
4165
Checks if --json is in the list of tokens received from the command line.
@@ -296,6 +320,45 @@ def find_pattern(args: list) -> None:
296320
click.secho('Unable to find the pattern in any memory region')
297321

298322

323+
def replace_pattern(args: list) -> None:
324+
"""
325+
Searches the current processes accessible memory for a specific pattern and replaces it with given bytes or string.
326+
327+
:param args:
328+
:return:
329+
"""
330+
331+
if len(clean_argument_flags(args)) < 2:
332+
click.secho('Usage: memory replace "<search pattern eg: 41 41 ?? 41>" "<replace value eg: 41 50>" (--string-pattern) (--string-replace)', bold=True)
333+
return
334+
335+
# if we got a string as search pattern input, convert it to hex
336+
if _is_string_pattern(args):
337+
pattern = ' '.join(hex(ord(x))[2:] for x in args[0])
338+
else:
339+
pattern = args[0]
340+
341+
# if we got a string as replace input, convert it to int[], otherwise convert hex to int[]
342+
replace = args[1]
343+
if _is_string_replace(args):
344+
replace = [ord(x) for x in replace]
345+
else:
346+
replace = [int(x, 16) for x in replace.split(' ')]
347+
348+
click.secho('Searching for: {0}, replacing with: {1}'.format(pattern, args[1]), dim=True)
349+
350+
api = state_connection.get_api()
351+
data = api.memory_replace(pattern, replace)
352+
353+
if len(data) > 0:
354+
click.secho('Pattern replaced at {0} addresses'.format(len(data)), fg='green')
355+
for address in data:
356+
click.secho(address)
357+
358+
else:
359+
click.secho('Unable to find the pattern in any memory region')
360+
361+
299362
def write(args: list) -> None:
300363
"""
301364
Write an arbitrary amount of bytes to an arbitrary memory address.

objection/console/commands.py

+6
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,12 @@
225225
'exec': memory.find_pattern
226226
},
227227

228+
'replace': {
229+
'meta': 'Search and replace pattern in the applications memory',
230+
'flags': ['--string-pattern', '--string-replace'],
231+
'exec': memory.replace_pattern
232+
},
233+
228234
'write': {
229235
'meta': 'Write raw bytes to a memory address. Use with caution!',
230236
'flags': ['--string'],

tests/commands/test_memory.py

+50-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
from unittest import mock
33

44
from objection.commands.memory import _is_string_input, dump_all, dump_from_base, list_modules, list_exports, \
5-
find_pattern
5+
find_pattern, replace_pattern
66
from ..helpers import capture
77

88

@@ -207,6 +207,55 @@ def test_find_pattern_without_string_argument_with_offets_only(self, mock_api):
207207
expected_output = """Searching for: 41 41 41
208208
Pattern matched at 1 addresses
209209
0x08000000
210+
"""
211+
212+
self.assertEqual(output, expected_output)
213+
214+
215+
def test_replace_pattern_validates_arguments(self):
216+
with capture(replace_pattern, []) as o:
217+
output = o
218+
219+
self.assertEqual(output, 'Usage: memory replace "<search pattern eg: 41 41 ?? 41>" "<replace value eg: 41 50>" (--string-pattern) (--string-replace)\n')
220+
221+
@mock.patch('objection.state.connection.state_connection.get_api')
222+
def test_replace_pattern_without_string_argument(self, mock_api):
223+
mock_api.return_value.memory_replace.return_value = ['0x08000000']
224+
225+
with capture(replace_pattern, ['41 41 41','41 42']) as o:
226+
output = o
227+
228+
expected_output = """Searching for: 41 41 41, replacing with: 41 42
229+
Pattern replaced at 1 addresses
230+
0x08000000
231+
"""
232+
233+
self.assertEqual(output, expected_output)
234+
235+
@mock.patch('objection.state.connection.state_connection.get_api')
236+
def test_replace_pattern_with_string_argument(self, mock_api):
237+
mock_api.return_value.memory_replace.return_value = ['0x08000000']
238+
239+
with capture(replace_pattern, ['foo-bar-baz', '41 41', '--string-pattern']) as o:
240+
output = o
241+
242+
expected_output = """Searching for: 66 6f 6f 2d 62 61 72 2d 62 61 7a, replacing with: 41 41
243+
Pattern replaced at 1 addresses
244+
0x08000000
245+
"""
246+
247+
self.assertEqual(output, expected_output)
248+
249+
@mock.patch('objection.state.connection.state_connection.get_api')
250+
def test_replace_pattern_without_string_argument_with_offets_only(self, mock_api):
251+
mock_api.return_value.memory_replace.return_value = ['0x08000000']
252+
253+
with capture(replace_pattern, ['41 41 41', 'ABC', '--string-replace']) as o:
254+
output = o
255+
256+
expected_output = """Searching for: 41 41 41, replacing with: ABC
257+
Pattern replaced at 1 addresses
258+
0x08000000
210259
"""
211260

212261
self.assertEqual(output, expected_output)

0 commit comments

Comments
 (0)