android13/system/update_engine/scripts/cow_converter.py

88 lines
2.8 KiB
Python

#!/usr/bin/env python3
#
# Copyright (C) 2021 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
"""Command-line tool for converting OTA payloads to VABC style COW images."""
import os
import sys
import tempfile
import zipfile
import subprocess
def IsSparseImage(filepath):
"""Determine if an image is a sparse image
Args:
filepath: str, a path to an .img file
Returns:
return true iff the filepath is a sparse image.
"""
with open(filepath, 'rb') as fp:
# Magic for android sparse image format
# https://source.android.com/devices/bootloader/images
return fp.read(4) == b'\x3A\xFF\x26\xED'
def ConvertCOW(ota_path, target_file_path, tmp_dir, output_dir):
"""Convert ota payload to COW IMAGE
Args:
ota_path: str, path to ota.zip
target_file_path: str, path to target_file.zip,
must be the target build for OTA.
tmp_dir: A temp dir as scratch space
output_dir: A directory where all converted COW images will be written.
"""
with zipfile.ZipFile(ota_path) as ota_zip:
payload_path = ota_zip.extract("payload.bin", output_dir)
with zipfile.ZipFile(target_file_path) as zfp:
for fileinfo in zfp.infolist():
img_name = os.path.basename(fileinfo.filename)
if not fileinfo.filename.endswith(".img"):
continue
if fileinfo.filename.startswith("IMAGES/") or \
fileinfo.filename.startswith("RADIO/"):
img_path = zfp.extract(fileinfo, tmp_dir)
target_img_path = os.path.join(output_dir, img_name)
if IsSparseImage(img_path):
subprocess.check_call(["simg2img", img_path, target_img_path])
else:
os.rename(img_path, target_img_path)
print("Extracted", fileinfo.filename, "size:", fileinfo.file_size)
subprocess.call(["cow_converter", payload_path,
output_dir])
def main():
if len(sys.argv) != 4:
print(
"Usage:", sys.argv[0], "<your_ota.zip> <target_file.zip> <output dir>")
return 1
ota_path = sys.argv[1]
target_file_path = sys.argv[2]
output_dir = sys.argv[3]
os.makedirs(output_dir, exist_ok=True)
with tempfile.TemporaryDirectory() as tmp_dir:
ConvertCOW(ota_path, target_file_path, tmp_dir, output_dir)
return 0
if __name__ == '__main__':
sys.exit(main())