Skip to content

Commit f41a0db

Browse files
committed
Initial release
0 parents  commit f41a0db

34 files changed

+5551
-0
lines changed

.github/workflows/package-plugin.yaml

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
name: Package IDA Plugin 📦
2+
3+
on: push
4+
5+
jobs:
6+
package:
7+
runs-on: ubuntu-latest
8+
steps:
9+
- uses: actions/checkout@v2
10+
11+
- name: Download workflow artifact
12+
uses: dawidd6/[email protected]
13+
with:
14+
15+
# the target repo for external artifacts (built libs)
16+
repo: gaasedelen/keystone
17+
branch: master
18+
19+
# token to fetch artifacts from the repo
20+
github_token: ${{secrets.KEYSTONE_PATCHING_TOKEN}}
21+
22+
# which workflow to search for artifacts
23+
workflow: python-publish.yml
24+
workflow_conclusion: success
25+
26+
- name: Package distributions
27+
shell: bash
28+
run: |
29+
mkdir dist && cd dist
30+
mkdir win32 && cp -r ../plugins/* ./win32/ && cp -r ../artifact/keystone_win32/* ./win32/patching/keystone && cd ./win32 && zip -r ../patching_win32.zip ./* && cd ..
31+
mkdir linux && cp -r ../plugins/* ./linux/ && cp -r ../artifact/keystone_linux/* ./linux/patching/keystone && cd ./linux && zip -r ../patching_linux.zip ./* && cd ..
32+
mkdir darwin && cp -r ../plugins/* ./darwin/ && cp -r ../artifact/keystone_darwin/* ./darwin/patching/keystone && cd ./darwin && zip -r ../patching_macos.zip ./* && cd ..
33+
34+
- uses: actions/upload-artifact@v2
35+
with:
36+
path: ${{ github.workspace }}/dist/*.zip

.gitignore

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
# Byte-compiled / optimized / DLL files
2+
__pycache__/
3+
*.py[cod]
4+
*$py.class
5+
6+
# C extensions
7+
*.so
8+
9+
# Distribution / packaging
10+
.Python
11+
build/
12+
develop-eggs/
13+
dist/
14+
downloads/
15+
eggs/
16+
.eggs/
17+
lib/
18+
lib64/
19+
parts/
20+
sdist/
21+
var/
22+
wheels/
23+
pip-wheel-metadata/
24+
share/python-wheels/
25+
*.egg-info/
26+
.installed.cfg
27+
*.egg
28+
MANIFEST
29+
30+
# PyInstaller
31+
# Usually these files are written by a python script from a template
32+
# before PyInstaller builds the exe, so as to inject date/other infos into it.
33+
*.manifest
34+
*.spec
35+
36+
# Installer logs
37+
pip-log.txt
38+
pip-delete-this-directory.txt
39+
40+
# Unit test / coverage reports
41+
htmlcov/
42+
.tox/
43+
.nox/
44+
.coverage
45+
.coverage.*
46+
.cache
47+
nosetests.xml
48+
coverage.xml
49+
*.cover
50+
*.py,cover
51+
.hypothesis/
52+
.pytest_cache/
53+
54+
# Translations
55+
*.mo
56+
*.pot
57+
58+
# Django stuff:
59+
*.log
60+
local_settings.py
61+
db.sqlite3
62+
db.sqlite3-journal
63+
64+
# Flask stuff:
65+
instance/
66+
.webassets-cache
67+
68+
# Scrapy stuff:
69+
.scrapy
70+
71+
# Sphinx documentation
72+
docs/_build/
73+
74+
# PyBuilder
75+
target/
76+
77+
# Jupyter Notebook
78+
.ipynb_checkpoints
79+
80+
# IPython
81+
profile_default/
82+
ipython_config.py
83+
84+
# pyenv
85+
.python-version
86+
87+
# pipenv
88+
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
89+
# However, in case of collaboration, if having platform-specific dependencies or dependencies
90+
# having no cross-platform support, pipenv may install dependencies that don't work, or not
91+
# install all needed dependencies.
92+
#Pipfile.lock
93+
94+
# PEP 582; used by e.g. github.com/David-OConnor/pyflow
95+
__pypackages__/
96+
97+
# Celery stuff
98+
celerybeat-schedule
99+
celerybeat.pid
100+
101+
# SageMath parsed files
102+
*.sage.py
103+
104+
# Environments
105+
.env
106+
.venv
107+
env/
108+
venv/
109+
ENV/
110+
env.bak/
111+
venv.bak/
112+
113+
# Spyder project settings
114+
.spyderproject
115+
.spyproject
116+
117+
# Rope project settings
118+
.ropeproject
119+
120+
# mkdocs documentation
121+
/site
122+
123+
# mypy
124+
.mypy_cache/
125+
.dmypy.json
126+
dmypy.json
127+
128+
# Pyre type checker
129+
.pyre/
130+
131+
# ida dev
132+
*.id0
133+
*.id1
134+
*.id2
135+
*.nam
136+
*.til
137+
*.idb
138+
*.i64
139+
140+
# ida test suite
141+
samples/*
142+
plugins/patching/keystone/*

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2022 Markus Gaasedelen
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

README.md

+133
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,133 @@
1+
# Patching - Interactive Binary Patching for IDA Pro
2+
3+
<p align="center"><img alt="Patching Plugin" src="screenshots/title.png"/></p>
4+
5+
## Overview
6+
7+
Patching assembly code to change the behavior of an existing program is not uncommon in malware analysis, software reverse engineering, and broader domains of security research. This project extends the popular [IDA Pro](https://www.hex-rays.com/products/ida/) disassembler to create a more robust interactive binary patching workflow designed for rapid iteration.
8+
9+
This project is currently powered by a minor [fork](https://github.com/gaasedelen/keystone) of the ubiquitous [Keystone Engine](https://github.com/keystone-engine/keystone), supporting x86/x64 and Arm/Arm64 patching with plans to enable the remaining Keystone architectures in a future release.
10+
11+
Special thanks to [Hex-Rays](https://hex-rays.com/) for supporting the development of this plugin.
12+
13+
## Releases
14+
15+
* v0.1 -- Initial release
16+
17+
# Installation
18+
19+
This plugin requires IDA 7.6 and Python 3. It supports Windows, Linux, and macOS.
20+
21+
## Easy Install
22+
23+
Run the following line in the IDA console to automatically install the plugin:
24+
25+
### Windows / Linux
26+
27+
```python
28+
import urllib.request as r; exec(r.urlopen('https://github.com/gaasedelen/patching/raw/main/install.py').read())
29+
```
30+
31+
### macOS
32+
33+
```python
34+
import urllib.request as r; exec(r.urlopen('https://github.com/gaasedelen/patching/raw/main/install.py', cafile='/etc/ssl/cert.pem').read())
35+
```
36+
37+
## Manual Install
38+
39+
Alternatively, the plugin can be manually installed by downloading the distributable plugin package for your respective platform from the [releases](https://github.com/gaasedelen/patching/releases) page and unzipping it to your plugins folder.
40+
41+
It is __*strongly*__ recommended you install this plugin into IDA's user plugin directory:
42+
43+
```python
44+
import ida_diskio, os; print(os.path.join(ida_diskio.get_user_idadir(), "plugins"))
45+
```
46+
47+
# Usage
48+
49+
The patching plugin will automatically load for supported architectures (x86/x64/Arm/Arm64) and inject relevant patching actions into the right click context menu of the IDA disassembly views:
50+
51+
<p align="center"><img alt="Patching plugin right click context menu" src="screenshots/usage.gif"/></p>
52+
53+
A complete listing of the contextual patching actions are described in the following sections.
54+
55+
## Assemble
56+
57+
The main patching dialog can be launched via the Assemble action in the right click context menu. It simulates a basic IDA disassembly view that can be used to edit one or several instructions in rapid succession.
58+
59+
<p align="center"><img alt="The interactive patching dialog" src="screenshots/assemble.gif"/></p>
60+
61+
The assembly line is an editable field that can be used to modify instructions in real-time. Pressing enter will commit (patch) the entered instruction into the database.
62+
63+
Your current location (a.k.a your cursor) will always be highlighted in green. Instructions that will be clobbered as a result of your patch / edit will be highlighted in red prior to committing the patch.
64+
65+
<p align="center"><img alt="Additional instructions that will be clobbered by a patch show up as red" src="screenshots/clobber.png"/></p>
66+
67+
Finally, the `UP` and `DOWN` arrow keys can be used while still focused on the editable assembly text field to quickly move the cursor up and down the disassembly view without using the mouse.
68+
69+
## NOP
70+
71+
The most common patching action is to NOP out one or more instructions. For this reason, the NOP action will always be visible in the right click menu for quick access.
72+
73+
<p align="center"><img alt="Right click NOP instruction" src="screenshots/nop.gif"/></p>
74+
75+
Individual instructions can be NOP'ed, as well as a selected range of instructions.
76+
77+
## Force Conditional Jump
78+
79+
Forcing a conditional jump to always execute a 'good' path is another common patching action. The plugin will only show this action when right clicking a conditional jump instruction.
80+
81+
<p align="center"><img alt="Forcing a conditional jump" src="screenshots/forcejump.gif"/></p>
82+
83+
If you *never* want a conditional jump to be taken, you can just NOP it instead!
84+
85+
## Save & Quick Apply
86+
87+
Patches can be saved (applied) to a selected executable via the patching submenu at any time. The quick-apply action makes it even faster to save subsequent patches using the same settings.
88+
89+
<p align="center"><img alt="Applying patches to the original executable" src="screenshots/save.gif"/></p>
90+
91+
The plugin will also make an active effort to retain a backup (`.bak`) of the original executable which it uses to 'cleanly' apply the current set of database patches during each save.
92+
93+
## Revert Patch
94+
95+
Finally, if you are ever unhappy with a patch you can simply right click patched (yellow) blocks of instructions to revert them to their original value.
96+
97+
<p align="center"><img alt="Reverting patches" src="screenshots/revert.gif"/></p>
98+
99+
While it is 'easy' to revert bytes back to their original value, it can be 'hard' to restore analysis to its previous state. Reverting a patch may *occasionally* require additional human fixups.
100+
101+
# Known Bugs
102+
103+
* Further improve ARM / ARM64 / THUMB correctness
104+
* Define 'better' behavior for cpp::like::symbols(...) / IDBs (very sketchy right now)
105+
* Adding / Updating / Modifying / Showing / Warning about Relocation Entries??
106+
* Handle renamed registers (like against dwarf annotated idb)?
107+
* A number of new instructions (circa 2017 and later) are not supported by Keystone
108+
* A few problematic instruction encodings by Keystone
109+
110+
# Future Work
111+
112+
Time and motivation permitting, future work may include:
113+
114+
* Enable the remaining major architectures supported by Keystone:
115+
* PPC32 / PPC64 / MIPS32 / MIPS64 / SPARC / SystemZ
116+
* Multi instruction assembly (eg. `xor eax, eax; ret;`)
117+
* Multi line assembly (eg. shellcode / asm labels)
118+
* Interactive byte / data / string editing
119+
* Symbol hinting / auto-complete / fuzzy-matching
120+
* Syntax highlighting the editable assembly line
121+
* Better hinting of errors, syntax issues, etc
122+
* NOP / Force Jump from Hex-Rays view (sounds easy, but probably pretty hard!)
123+
* radio button toggle between 'pretty print' mode vs 'raw' mode? or display both?
124+
```
125+
Pretty: mov [rsp+48h+dwCreationDisposition], 3
126+
Raw: mov [rsp+20h], 3
127+
```
128+
129+
I welcome external contributions, issues, and feature requests. Please make any pull requests to the `develop` branch of this repository if you would like them to be considered for a future release.
130+
131+
# Authors
132+
133+
* Markus Gaasedelen ([@gaasedelen](https://twitter.com/gaasedelen))

0 commit comments

Comments
 (0)