mirror of
https://github.com/google/pebble.git
synced 2025-03-15 00:31:21 +00:00
155 lines
5.1 KiB
Python
Executable file
155 lines
5.1 KiB
Python
Executable file
#!/usr/bin/env python
|
|
# Copyright 2024 Google LLC
|
|
#
|
|
# 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.
|
|
|
|
|
|
from __future__ import print_function
|
|
import errno
|
|
import sys
|
|
import traceback
|
|
|
|
import insert_firmware_descr
|
|
import pulse
|
|
import stm32_crc
|
|
|
|
|
|
def _load(connection, image, progress, verbose, address):
|
|
image_crc = stm32_crc.crc32(image)
|
|
|
|
progress_cb = None
|
|
if progress or verbose:
|
|
def progress_cb(acked):
|
|
print('.' if acked else 'R', end='')
|
|
sys.stdout.flush()
|
|
|
|
if progress or verbose:
|
|
print('Erasing... ', end='')
|
|
sys.stdout.flush()
|
|
try:
|
|
connection.flash.erase(address, len(image))
|
|
except pulse.PulseError as e:
|
|
detail = ''.join(traceback.format_exception_only(type(e), e))
|
|
if verbose:
|
|
detail = '\n' + traceback.format_exc()
|
|
print('Erase failed! ' + detail)
|
|
return False
|
|
if progress or verbose:
|
|
print('done.')
|
|
sys.stdout.flush()
|
|
|
|
try:
|
|
retries = connection.flash.write(address, image,
|
|
progress_cb=progress_cb)
|
|
except pulse.PulseError as e:
|
|
detail = ''.join(traceback.format_exception_only(type(e), e))
|
|
if verbose:
|
|
detail = '\n' + traceback.format_exc()
|
|
print('Write failed! ' + detail)
|
|
return False
|
|
|
|
result_crc = connection.flash.crc(address, len(image))
|
|
|
|
if progress or verbose:
|
|
print()
|
|
if verbose:
|
|
print('Retries: %d' % retries)
|
|
|
|
return result_crc == image_crc
|
|
|
|
def load_firmware(connection, fin, progress, verbose, address=None):
|
|
if address is None:
|
|
# If address is unspecified, assume we want the prf address
|
|
_, address, length = connection.flash.query_region_geometry(
|
|
connection.flash.REGION_PRF)
|
|
address = int(address)
|
|
|
|
image = insert_firmware_descr.insert_firmware_description_struct(fin)
|
|
if len(image) > length:
|
|
print('Image is too big!')
|
|
return False
|
|
if _load(connection, image, progress, verbose, address):
|
|
connection.flash.finalize_region(
|
|
connection.flash.REGION_PRF)
|
|
return True
|
|
return False
|
|
|
|
def load_resources(connection, fin, progress, verbose):
|
|
_, address, length = connection.flash.query_region_geometry(
|
|
connection.flash.REGION_SYSTEM_RESOURCES)
|
|
|
|
with open(fin, 'rb') as f:
|
|
data = f.read()
|
|
assert len(data) <= length
|
|
if _load(connection, data, progress, verbose, address):
|
|
connection.flash.finalize_region(
|
|
connection.flash.REGION_SYSTEM_RESOURCES)
|
|
return True
|
|
return False
|
|
|
|
|
|
if __name__ == '__main__':
|
|
import argparse
|
|
import logging
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description="A factory tool to load binary data into Pebble's "
|
|
"external flash storage.")
|
|
parser.add_argument('-v', '--verbose', action='store_true',
|
|
help='print verbose status output')
|
|
parser.add_argument('-p', '--progress', action='store_true',
|
|
help='print progress output')
|
|
#parser.add_argument('--reset', help='reset target after loading',
|
|
# action='store_true')
|
|
parser.add_argument('-t', '--tty', metavar='TTY', default=None,
|
|
help='the target serial port')
|
|
|
|
subparsers = parser.add_subparsers(help='commands', dest='which')
|
|
|
|
fw_parser = subparsers.add_parser(
|
|
'firmware', help='load a recovery firmware into flash')
|
|
fw_parser.add_argument(
|
|
'file', metavar='FILE',
|
|
help='a bin containing the recovery firmware to be loaded')
|
|
fw_parser.set_defaults(func=load_firmware)
|
|
|
|
res_parser = subparsers.add_parser('resources',
|
|
help='load firmware resources')
|
|
res_parser.add_argument(
|
|
'file', metavar='FILE',
|
|
help='a pbpack containing the resources to be loaded')
|
|
res_parser.set_defaults(func=load_resources)
|
|
|
|
args = parser.parse_args()
|
|
|
|
if args.verbose:
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
else:
|
|
logging.basicConfig(level=logging.WARNING)
|
|
|
|
with pulse.Connection.open_dbgserial(args.tty) as connection:
|
|
connection.change_baud_rate(921600)
|
|
success = False
|
|
try:
|
|
success = args.func(connection, args.file, args.progress, args.verbose)
|
|
except pulse.PulseError as e:
|
|
detail = ''.join(traceback.format_exception_only(type(e), e))
|
|
if args.verbose:
|
|
detail = traceback.format_exc()
|
|
print(detail)
|
|
|
|
if success:
|
|
print('Success!')
|
|
else:
|
|
print('Fail!')
|
|
sys.exit(-1)
|