clean up for 0.1.0 publish
This commit is contained in:
parent
0cbb014230
commit
934fe099df
7 changed files with 245 additions and 205 deletions
46
README.md
46
README.md
|
@ -1,26 +1,38 @@
|
|||
# payload dumper
|
||||
Script tested on Yandex Amber OTA's (full and incremental) under Linux(but may works on Windows too)
|
||||
|
||||
## System requirement
|
||||
Dumps the `payload.bin` image found in Android update images.
|
||||
|
||||
- Python3, pip
|
||||
- google protobuf for python `pip install protobuf`
|
||||
## Installation
|
||||
|
||||
## Guide
|
||||
### Requirements
|
||||
|
||||
- Make you sure you have Python 3.6 installed.
|
||||
- Download payload_dumper.py and update_metadata_pb2.py
|
||||
- Extract your OTA zip and place payload.bin in the same folder as these files.
|
||||
- Open PowerShell, Command Prompt, or Terminal depending on your OS.
|
||||
- Enter the following command: python -m pip install protobuf
|
||||
- Python3
|
||||
- pip
|
||||
|
||||
### Full OTA
|
||||
### Install using pip
|
||||
|
||||
- When that’s finished, enter this command: python payload_dumper.py payload.bin
|
||||
- This will start to extract the images within the payload.bin file to the output folder you are in.
|
||||
```sh
|
||||
pip install --user payload_dumper
|
||||
```
|
||||
|
||||
### Incremental OTA
|
||||
## Usage
|
||||
|
||||
- Copy original images (from full OTA or dumped from devices) to old folder (with part name + .img, ex: boot.img, system.img)
|
||||
- run python payload_dumper.py --diff payload.bin
|
||||
- file extracted to the output folder you are in.
|
||||
### Dumping the entirety of `payload.bin`
|
||||
|
||||
```
|
||||
payload_dumper payload.bin
|
||||
```
|
||||
|
||||
### Dumping specific partitions
|
||||
|
||||
Use a comma-separated list of partitions to dump:
|
||||
```
|
||||
payload_dumper --partitions boot,dtbo,vendor
|
||||
```
|
||||
|
||||
### Patching older image with OTA
|
||||
|
||||
Assuming the old partitions are in a directory named `old/`:
|
||||
```
|
||||
payload_dumper --diff payload.bin
|
||||
```
|
||||
|
|
|
@ -1,185 +0,0 @@
|
|||
#!/usr/bin/env python
|
||||
import struct
|
||||
import hashlib
|
||||
import bz2
|
||||
import sys
|
||||
import argparse
|
||||
import bsdiff4
|
||||
import io
|
||||
import os
|
||||
from enlighten import get_manager
|
||||
|
||||
try:
|
||||
import lzma
|
||||
except ImportError:
|
||||
from backports import lzma
|
||||
|
||||
import update_metadata_pb2 as um
|
||||
|
||||
flatten = lambda l: [item for sublist in l for item in sublist]
|
||||
|
||||
manager = get_manager()
|
||||
|
||||
|
||||
def u32(x):
|
||||
return struct.unpack(">I", x)[0]
|
||||
|
||||
|
||||
def u64(x):
|
||||
return struct.unpack(">Q", x)[0]
|
||||
|
||||
|
||||
def verify_contiguous(exts):
|
||||
blocks = 0
|
||||
|
||||
for ext in exts:
|
||||
if ext.start_block != blocks:
|
||||
return False
|
||||
|
||||
blocks += ext.num_blocks
|
||||
|
||||
return True
|
||||
|
||||
|
||||
def data_for_op(op, out_file, old_file):
|
||||
args.payloadfile.seek(data_offset + op.data_offset)
|
||||
data = args.payloadfile.read(op.data_length)
|
||||
|
||||
# assert hashlib.sha256(data).digest() == op.data_sha256_hash, 'operation data hash mismatch'
|
||||
|
||||
if op.type == op.REPLACE_XZ:
|
||||
dec = lzma.LZMADecompressor()
|
||||
data = dec.decompress(data)
|
||||
out_file.seek(op.dst_extents[0].start_block * block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.REPLACE_BZ:
|
||||
dec = bz2.BZ2Decompressor()
|
||||
data = dec.decompress(data)
|
||||
out_file.seek(op.dst_extents[0].start_block * block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.REPLACE:
|
||||
out_file.seek(op.dst_extents[0].start_block * block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.SOURCE_COPY:
|
||||
if not args.diff:
|
||||
print("SOURCE_COPY supported only for differential OTA")
|
||||
sys.exit(-2)
|
||||
out_file.seek(op.dst_extents[0].start_block * block_size)
|
||||
for ext in op.src_extents:
|
||||
old_file.seek(ext.start_block * block_size)
|
||||
data = old_file.read(ext.num_blocks * block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.SOURCE_BSDIFF:
|
||||
if not args.diff:
|
||||
print("SOURCE_BSDIFF supported only for differential OTA")
|
||||
sys.exit(-3)
|
||||
out_file.seek(op.dst_extents[0].start_block * block_size)
|
||||
tmp_buff = io.BytesIO()
|
||||
for ext in op.src_extents:
|
||||
old_file.seek(ext.start_block * block_size)
|
||||
old_data = old_file.read(ext.num_blocks * block_size)
|
||||
tmp_buff.write(old_data)
|
||||
tmp_buff.seek(0)
|
||||
old_data = tmp_buff.read()
|
||||
tmp_buff.seek(0)
|
||||
tmp_buff.write(bsdiff4.patch(old_data, data))
|
||||
n = 0
|
||||
tmp_buff.seek(0)
|
||||
for ext in op.dst_extents:
|
||||
tmp_buff.seek(n * block_size)
|
||||
n += ext.num_blocks
|
||||
data = tmp_buff.read(ext.num_blocks * block_size)
|
||||
out_file.seek(ext.start_block * block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.ZERO:
|
||||
for ext in op.dst_extents:
|
||||
out_file.seek(ext.start_block * block_size)
|
||||
out_file.write(b"\x00" * ext.num_blocks * block_size)
|
||||
else:
|
||||
print("Unsupported type = %d" % op.type)
|
||||
sys.exit(-1)
|
||||
|
||||
return data
|
||||
|
||||
|
||||
def dump_part(part):
|
||||
print("Processing partition: %s" % part.partition_name)
|
||||
|
||||
out_file = open("%s/%s.img" % (args.out, part.partition_name), "wb")
|
||||
h = hashlib.sha256()
|
||||
|
||||
if args.diff:
|
||||
old_file = open("%s/%s.img" % (args.old, part.partition_name), "rb")
|
||||
else:
|
||||
old_file = None
|
||||
|
||||
operation_progress = manager.counter(
|
||||
total=len(part.operations), desc="Operations", unit="op", leave=False
|
||||
)
|
||||
for op in part.operations:
|
||||
data = data_for_op(op, out_file, old_file)
|
||||
operation_progress.update()
|
||||
operation_progress.close()
|
||||
|
||||
|
||||
parser = argparse.ArgumentParser(description="OTA payload dumper")
|
||||
parser.add_argument(
|
||||
"payloadfile", type=argparse.FileType("rb"), help="payload file name"
|
||||
)
|
||||
parser.add_argument("--out", default="output", help="output directory (defaul: output)")
|
||||
parser.add_argument(
|
||||
"--diff",
|
||||
action="store_true",
|
||||
help="extract differential OTA, you need put original images to old dir",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--old",
|
||||
default="old",
|
||||
help="directory with original images for differential OTA (defaul: old)",
|
||||
)
|
||||
parser.add_argument("--images", default="", help="images to extract (default: empty)")
|
||||
args = parser.parse_args()
|
||||
|
||||
# Check for --out directory exists
|
||||
if not os.path.exists(args.out):
|
||||
os.makedirs(args.out)
|
||||
|
||||
magic = args.payloadfile.read(4)
|
||||
assert magic == b"CrAU"
|
||||
|
||||
file_format_version = u64(args.payloadfile.read(8))
|
||||
assert file_format_version == 2
|
||||
|
||||
manifest_size = u64(args.payloadfile.read(8))
|
||||
|
||||
metadata_signature_size = 0
|
||||
|
||||
if file_format_version > 1:
|
||||
metadata_signature_size = u32(args.payloadfile.read(4))
|
||||
|
||||
manifest = args.payloadfile.read(manifest_size)
|
||||
metadata_signature = args.payloadfile.read(metadata_signature_size)
|
||||
|
||||
data_offset = args.payloadfile.tell()
|
||||
|
||||
dam = um.DeltaArchiveManifest()
|
||||
dam.ParseFromString(manifest)
|
||||
block_size = dam.block_size
|
||||
|
||||
if args.images == "":
|
||||
progress = manager.counter(
|
||||
total=len(dam.partitions), desc="Partitions", unit="part", position=1
|
||||
)
|
||||
for part in dam.partitions:
|
||||
dump_part(part)
|
||||
progress.update()
|
||||
else:
|
||||
images = args.images.split(",")
|
||||
for image in images:
|
||||
partition = [part for part in dam.partitions if part.partition_name == image]
|
||||
if partition:
|
||||
dump_part(partition[0])
|
||||
else:
|
||||
print("Partition %s not found in payload!" % image)
|
||||
|
||||
manager.stop()
|
0
payload_dumper/__init__.py
Normal file
0
payload_dumper/__init__.py
Normal file
213
payload_dumper/dumper.py
Executable file
213
payload_dumper/dumper.py
Executable file
|
@ -0,0 +1,213 @@
|
|||
#!/usr/bin/env python
|
||||
import struct
|
||||
import hashlib
|
||||
import bz2
|
||||
import sys
|
||||
import argparse
|
||||
import bsdiff4
|
||||
import io
|
||||
import os
|
||||
from enlighten import get_manager
|
||||
import lzma
|
||||
import payload_dumper.update_metadata_pb2 as um
|
||||
|
||||
flatten = lambda l: [item for sublist in l for item in sublist]
|
||||
|
||||
|
||||
def u32(x):
|
||||
return struct.unpack(">I", x)[0]
|
||||
|
||||
|
||||
def u64(x):
|
||||
return struct.unpack(">Q", x)[0]
|
||||
|
||||
|
||||
def verify_contiguous(exts):
|
||||
blocks = 0
|
||||
for ext in exts:
|
||||
if ext.start_block != blocks:
|
||||
return False
|
||||
|
||||
blocks += ext.num_blocks
|
||||
|
||||
return True
|
||||
|
||||
|
||||
class Dumper:
|
||||
def __init__(self, payloadfile, out, diff=None, old=None, images=""):
|
||||
self.payloadfile = payloadfile
|
||||
self.out = out
|
||||
self.diff = diff
|
||||
self.old = old
|
||||
self.images = images
|
||||
self.validate_magic()
|
||||
self.manager = get_manager()
|
||||
|
||||
def run(self):
|
||||
if self.images == "":
|
||||
progress = self.manager.counter(
|
||||
total=len(self.dam.partitions),
|
||||
desc="Partitions",
|
||||
unit="part",
|
||||
position=1,
|
||||
)
|
||||
for part in self.dam.partitions:
|
||||
self.dump_part(part)
|
||||
progress.update()
|
||||
else:
|
||||
images = [image.strip() for image in self.images.split(",")]
|
||||
progress = self.manager.counter(
|
||||
total=len(images),
|
||||
desc="Partitions",
|
||||
unit="part",
|
||||
position=1,
|
||||
)
|
||||
for image in images:
|
||||
partition = [
|
||||
part for part in self.dam.partitions if part.partition_name == image
|
||||
]
|
||||
if partition:
|
||||
self.dump_part(partition[0])
|
||||
else:
|
||||
print("Partition %s not found in payload!" % image)
|
||||
progress.update()
|
||||
|
||||
self.manager.stop()
|
||||
|
||||
def validate_magic(self):
|
||||
magic = self.payloadfile.read(4)
|
||||
assert magic == b"CrAU"
|
||||
|
||||
file_format_version = u64(self.payloadfile.read(8))
|
||||
assert file_format_version == 2
|
||||
|
||||
manifest_size = u64(self.payloadfile.read(8))
|
||||
|
||||
metadata_signature_size = 0
|
||||
|
||||
if file_format_version > 1:
|
||||
metadata_signature_size = u32(self.payloadfile.read(4))
|
||||
|
||||
manifest = self.payloadfile.read(manifest_size)
|
||||
self.metadata_signature = self.payloadfile.read(metadata_signature_size)
|
||||
self.data_offset = self.payloadfile.tell()
|
||||
|
||||
self.dam = um.DeltaArchiveManifest()
|
||||
self.dam.ParseFromString(manifest)
|
||||
self.block_size = self.dam.block_size
|
||||
|
||||
def data_for_op(self, op, out_file, old_file):
|
||||
self.payloadfile.seek(self.data_offset + op.data_offset)
|
||||
data = self.payloadfile.read(op.data_length)
|
||||
|
||||
# assert hashlib.sha256(data).digest() == op.data_sha256_hash, 'operation data hash mismatch'
|
||||
|
||||
if op.type == op.REPLACE_XZ:
|
||||
dec = lzma.LZMADecompressor()
|
||||
data = dec.decompress(data)
|
||||
out_file.seek(op.dst_extents[0].start_block * self.block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.REPLACE_BZ:
|
||||
dec = bz2.BZ2Decompressor()
|
||||
data = dec.decompress(data)
|
||||
out_file.seek(op.dst_extents[0].start_block * self.block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.REPLACE:
|
||||
out_file.seek(op.dst_extents[0].start_block * self.block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.SOURCE_COPY:
|
||||
if not self.diff:
|
||||
print("SOURCE_COPY supported only for differential OTA")
|
||||
sys.exit(-2)
|
||||
out_file.seek(op.dst_extents[0].start_block * self.block_size)
|
||||
for ext in op.src_extents:
|
||||
old_file.seek(ext.start_block * self.block_size)
|
||||
data = old_file.read(ext.num_blocks * self.block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.SOURCE_BSDIFF:
|
||||
if not self.diff:
|
||||
print("SOURCE_BSDIFF supported only for differential OTA")
|
||||
sys.exit(-3)
|
||||
out_file.seek(op.dst_extents[0].start_block * self.block_size)
|
||||
tmp_buff = io.BytesIO()
|
||||
for ext in op.src_extents:
|
||||
old_file.seek(ext.start_block * self.block_size)
|
||||
old_data = old_file.read(ext.num_blocks * self.block_size)
|
||||
tmp_buff.write(old_data)
|
||||
tmp_buff.seek(0)
|
||||
old_data = tmp_buff.read()
|
||||
tmp_buff.seek(0)
|
||||
tmp_buff.write(bsdiff4.patch(old_data, data))
|
||||
n = 0
|
||||
tmp_buff.seek(0)
|
||||
for ext in op.dst_extents:
|
||||
tmp_buff.seek(n * self.block_size)
|
||||
n += ext.num_blocks
|
||||
data = tmp_buff.read(ext.num_blocks * self.block_size)
|
||||
out_file.seek(ext.start_block * self.block_size)
|
||||
out_file.write(data)
|
||||
elif op.type == op.ZERO:
|
||||
for ext in op.dst_extents:
|
||||
out_file.seek(ext.start_block * self.block_size)
|
||||
out_file.write(b"\x00" * ext.num_blocks * self.block_size)
|
||||
else:
|
||||
print("Unsupported type = %d" % op.type)
|
||||
sys.exit(-1)
|
||||
|
||||
return data
|
||||
|
||||
def dump_part(self, part):
|
||||
print("Processing %s" % part.partition_name)
|
||||
|
||||
out_file = open("%s/%s.img" % (self.out, part.partition_name), "wb")
|
||||
h = hashlib.sha256()
|
||||
|
||||
if self.diff:
|
||||
old_file = open("%s/%s.img" % (self.old, part.partition_name), "rb")
|
||||
else:
|
||||
old_file = None
|
||||
|
||||
operation_progress = self.manager.counter(
|
||||
total=len(part.operations), desc="Operations", unit="op", leave=False
|
||||
)
|
||||
for op in part.operations:
|
||||
data = self.data_for_op(op, out_file, old_file)
|
||||
operation_progress.update()
|
||||
operation_progress.close()
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser(description="OTA payload dumper")
|
||||
parser.add_argument(
|
||||
"payloadfile", type=argparse.FileType("rb"), help="payload file name"
|
||||
)
|
||||
parser.add_argument(
|
||||
"--out", default="output", help="output directory (default: output)"
|
||||
)
|
||||
parser.add_argument(
|
||||
"--diff",
|
||||
action="store_true",
|
||||
help="extract differential OTA",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--old",
|
||||
default="old",
|
||||
help="directory with original images for differential OTA (default: old)",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--partitions", default="", help="comma separated list of partitions to extract (default: extract all)"
|
||||
)
|
||||
args = parser.parse_args()
|
||||
|
||||
# Check for --out directory exists
|
||||
if not os.path.exists(args.out):
|
||||
os.makedirs(args.out)
|
||||
|
||||
dumper = Dumper(
|
||||
args.payloadfile, args.out, diff=args.diff, old=args.old, images=args.partitions
|
||||
)
|
||||
dumper.run()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
|
@ -12,6 +12,9 @@ enlighten = "^1.10.2"
|
|||
|
||||
[tool.poetry.dev-dependencies]
|
||||
|
||||
[tool.poetry.scripts]
|
||||
payload_dumper = "payload_dumper:dumper.main"
|
||||
|
||||
[build-system]
|
||||
requires = ["poetry-core>=1.0.0"]
|
||||
build-backend = "poetry.core.masonry.api"
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
protobuf
|
||||
six
|
||||
bsdiff4
|
Loading…
Reference in a new issue