567 lines
17 KiB
Python
Executable File
567 lines
17 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# tools/flash_writer.py
|
|
|
|
# Licensed to the Apache Software Foundation (ASF) under one or more
|
|
# contributor license agreements. See the NOTICE file distributed with
|
|
# this work for additional information regarding copyright ownership. The
|
|
# ASF licenses this file to you 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.
|
|
|
|
import time
|
|
import sys
|
|
import os
|
|
import struct
|
|
import glob
|
|
import fnmatch
|
|
import errno
|
|
import telnetlib
|
|
import argparse
|
|
import shutil
|
|
import subprocess
|
|
import re
|
|
import xmodem
|
|
|
|
import_serial_module = True
|
|
|
|
# When SDK release, please set SDK_RELEASE as True.
|
|
SDK_RELEASE = False
|
|
|
|
if SDK_RELEASE :
|
|
PRINT_RAW_COMMAND = False
|
|
REBOOT_AT_END = True
|
|
else :
|
|
PRINT_RAW_COMMAND = True
|
|
REBOOT_AT_END = True
|
|
|
|
try:
|
|
import serial
|
|
except:
|
|
import_serial_module = False
|
|
|
|
# supported environment various
|
|
# CXD56_PORT
|
|
# CXD56_TELNETSRV_PORT
|
|
# CXD56_TELNETSRV_IP
|
|
|
|
PROTOCOL_SERIAL = 0
|
|
PROTOCOL_TELNET = 1
|
|
|
|
MAX_DOT_COUNT = 70
|
|
|
|
# configure parameters and default value
|
|
class ConfigArgs:
|
|
PROTOCOL_TYPE = None
|
|
SERIAL_PORT = "COM1"
|
|
SERVER_PORT = 4569
|
|
SERVER_IP = "localhost"
|
|
EOL = bytes([10])
|
|
WAIT_RESET = True
|
|
AUTO_RESET = False
|
|
DTR_RESET = False
|
|
XMODEM_BAUD = 0
|
|
NO_SET_BOOTABLE = False
|
|
PACKAGE_NAME = []
|
|
FILE_NAME = []
|
|
ERASE_NAME = []
|
|
PKGSYS_NAME = []
|
|
PKGAPP_NAME = []
|
|
PKGUPD_NAME = []
|
|
|
|
ROM_MSG = [b"Welcome to nash"]
|
|
XMDM_MSG = "Waiting for XMODEM (CRC or 1K) transfer. Ctrl-X to cancel."
|
|
|
|
class ConfigArgsLoader():
|
|
def __init__(self):
|
|
self.parser = argparse.ArgumentParser(formatter_class=argparse.RawTextHelpFormatter)
|
|
self.parser.add_argument("package_name", help="the name of the package to install", nargs='*')
|
|
self.parser.add_argument("-f", "--file", dest="file_name", help="save file", action='append')
|
|
self.parser.add_argument("-e", "--erase", dest="erase_name", help="erase file", action='append')
|
|
|
|
self.parser.add_argument("-S", "--sys", dest="pkgsys_name", help="the name of the system package to install", action='append')
|
|
self.parser.add_argument("-A", "--app", dest="pkgapp_name", help="the name of the application package to install", action='append')
|
|
self.parser.add_argument("-U", "--upd", dest="pkgupd_name", help="the name of the updater package to install", action='append')
|
|
|
|
self.parser.add_argument("-a", "--auto-reset", dest="auto_reset",
|
|
action="store_true", default=None,
|
|
help="try to auto reset develop board if possible")
|
|
self.parser.add_argument("-d", "--dtr-reset", dest="dtr_reset",
|
|
action="store_true", default=None,
|
|
help="try to auto reset develop board if possible")
|
|
self.parser.add_argument("-n", "--no-set-bootable", dest="no_set_bootable",
|
|
action="store_true", default=None,
|
|
help="not to set bootable")
|
|
|
|
group = self.parser.add_argument_group()
|
|
group.add_argument("-i", "--server-ip", dest="server_ip",
|
|
help="the ip address connected to the telnet server")
|
|
group.add_argument("-p", "--server-port", dest="server_port", type=int,
|
|
help="the port connected to the telnet server")
|
|
|
|
group = self.parser.add_argument_group()
|
|
group.add_argument("-c", "--serial-port", dest="serial_port", help="the serial port")
|
|
group.add_argument("-b", "--xmodem-baudrate", dest="xmodem_baud", help="Use the faster baudrate in xmodem")
|
|
|
|
mutually_group = self.parser.add_mutually_exclusive_group()
|
|
mutually_group.add_argument("-t", "--telnet-protocol", dest="telnet_protocol",
|
|
action="store_true", default=None,
|
|
help="use the telnet protocol for binary transmission")
|
|
mutually_group.add_argument("-s", "--serial-protocol", dest="serial_protocol",
|
|
action="store_true", default=None,
|
|
help="use the serial port for binary transmission, default options")
|
|
|
|
mutually_group2 = self.parser.add_mutually_exclusive_group()
|
|
mutually_group2.add_argument("-F", "--force-wait-reset", dest="wait_reset",
|
|
action="store_true", default=None,
|
|
help="force wait for pressing RESET button")
|
|
mutually_group2.add_argument("-N", "--no-wait-reset", dest="wait_reset",
|
|
action="store_false", default=None,
|
|
help="if possible, skip to wait for pressing RESET button")
|
|
|
|
def update_config(self):
|
|
args = self.parser.parse_args()
|
|
|
|
ConfigArgs.PACKAGE_NAME = args.package_name
|
|
ConfigArgs.FILE_NAME = args.file_name
|
|
ConfigArgs.ERASE_NAME = args.erase_name
|
|
ConfigArgs.PKGSYS_NAME = args.pkgsys_name
|
|
ConfigArgs.PKGAPP_NAME = args.pkgapp_name
|
|
ConfigArgs.PKGUPD_NAME = args.pkgupd_name
|
|
|
|
# Get serial port or telnet server ip etc
|
|
if args.serial_protocol == True:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_SERIAL
|
|
elif args.telnet_protocol == True:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_TELNET
|
|
|
|
if ConfigArgs.PROTOCOL_TYPE == None:
|
|
proto = os.environ.get("CXD56_PROTOCOL")
|
|
if proto is not None:
|
|
if 's' in proto:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_SERIAL
|
|
elif 't' in proto:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_TELNET
|
|
|
|
if ConfigArgs.PROTOCOL_TYPE == None:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_SERIAL
|
|
|
|
if ConfigArgs.PROTOCOL_TYPE == PROTOCOL_SERIAL:
|
|
if args.serial_port is not None:
|
|
ConfigArgs.SERIAL_PORT = args.serial_port
|
|
else:
|
|
# Get serial port from the environment
|
|
port = os.environ.get("CXD56_PORT")
|
|
if port is not None:
|
|
ConfigArgs.SERIAL_PORT = port
|
|
else:
|
|
print("CXD56_PORT is not set, Use " + ConfigArgs.SERIAL_PORT + ".")
|
|
else:
|
|
ConfigArgs.PROTOCOL_TYPE = PROTOCOL_TELNET
|
|
if args.server_port is not None:
|
|
ConfigArgs.SERVER_PORT = args.server_port
|
|
else:
|
|
port = os.environ.get("CXD56_TELNETSRV_PORT")
|
|
if port is not None:
|
|
ConfigArgs.SERVER_PORT = port
|
|
else:
|
|
print("CXD56_TELNETSRV_PORT is not set, Use " + str(ConfigArgs.SERVER_PORT) + ".")
|
|
if args.server_ip is not None:
|
|
ConfigArgs.SERVER_IP = args.server_ip
|
|
else:
|
|
ip = os.environ.get("CXD56_TELNETSRV_IP")
|
|
if ip is not None:
|
|
ConfigArgs.SERVER_IP = ip
|
|
else:
|
|
print("CXD56_TELNETSRV_IP is not set, Use " + ConfigArgs.SERVER_IP + ".")
|
|
|
|
if args.xmodem_baud is not None:
|
|
ConfigArgs.XMODEM_BAUD = args.xmodem_baud
|
|
|
|
if args.auto_reset is not None:
|
|
ConfigArgs.AUTO_RESET = args.auto_reset
|
|
|
|
if args.dtr_reset is not None:
|
|
ConfigArgs.DTR_RESET = args.dtr_reset
|
|
|
|
if args.no_set_bootable is not None:
|
|
ConfigArgs.NO_SET_BOOTABLE = args.no_set_bootable
|
|
|
|
if args.wait_reset is not None:
|
|
ConfigArgs.WAIT_RESET = args.wait_reset
|
|
|
|
class TelnetDev:
|
|
def __init__(self):
|
|
srv_ipaddr = ConfigArgs.SERVER_IP
|
|
srv_port = ConfigArgs.SERVER_PORT
|
|
self.recvbuf = b''
|
|
try:
|
|
self.telnet = telnetlib.Telnet(host=srv_ipaddr, port=srv_port, timeout=10)
|
|
# There is a ack to be sent after connecting to the telnet server.
|
|
self.telnet.write(b"\xff")
|
|
except Exception as e:
|
|
print("Cannot connect to the server %s:%d" % (srv_ipaddr, srv_port))
|
|
sys.exit(e.args[0])
|
|
|
|
def readline(self, size=None):
|
|
res = b''
|
|
ch = b''
|
|
while ch != ConfigArgs.EOL:
|
|
ch = self.getc_raw(1, timeout=0.1)
|
|
if ch == b'':
|
|
return res
|
|
res += ch
|
|
return res
|
|
|
|
def getc_raw(self, size, timeout=1):
|
|
res = b''
|
|
tm = time.monotonic()
|
|
while size > 0:
|
|
while self.recvbuf == b'':
|
|
self.recvbuf = self.telnet.read_eager()
|
|
if self.recvbuf == b'':
|
|
if (time.monotonic() - tm) > timeout:
|
|
return res
|
|
time.sleep(0.1)
|
|
res += self.recvbuf[0:1]
|
|
self.recvbuf = self.recvbuf[1:]
|
|
size -= 1
|
|
return res
|
|
|
|
def write(self, buffer):
|
|
self.telnet.write(buffer)
|
|
|
|
def discard_inputs(self, timeout=1.0):
|
|
while True:
|
|
ch = self.getc_raw(1, timeout=timeout)
|
|
if ch == b'':
|
|
break
|
|
|
|
def getc(self, size, timeout=1):
|
|
c = self.getc_raw(size, timeout)
|
|
return c
|
|
|
|
def putc(self, buffer, timeout=1):
|
|
self.telnet.write(buffer)
|
|
self.show_progress(len(buffer))
|
|
|
|
def reboot(self):
|
|
# no-op
|
|
pass
|
|
|
|
def set_file_size(self, filesize):
|
|
self.bytes_transfered = 0
|
|
self.filesize = filesize
|
|
self.count = 0
|
|
|
|
def show_progress(self, sendsize):
|
|
if PRINT_RAW_COMMAND:
|
|
if self.count < MAX_DOT_COUNT:
|
|
self.bytes_transfered = self.bytes_transfered + sendsize
|
|
cur_count = int(self.bytes_transfered * MAX_DOT_COUNT / self.filesize)
|
|
if MAX_DOT_COUNT < cur_count:
|
|
cur_count = MAX_DOT_COUNT
|
|
for idx in range(cur_count - self.count):
|
|
print('#',end='')
|
|
sys.stdout.flush()
|
|
self.count = cur_count
|
|
if self.count == MAX_DOT_COUNT:
|
|
print("\n")
|
|
|
|
class SerialDev:
|
|
def __init__(self):
|
|
if import_serial_module is False:
|
|
print("Cannot import serial module, maybe it's not install yet.")
|
|
print("\n", end="")
|
|
print("Please install python-setuptool by Cygwin installer.")
|
|
print("After that use easy_intall command to install serial module")
|
|
print(" $ cd tool/")
|
|
print(" $ python3 -m easy_install pyserial-2.7.tar.gz")
|
|
quit()
|
|
else:
|
|
port = ConfigArgs.SERIAL_PORT
|
|
try:
|
|
self.serial = serial.Serial(port, baudrate=115200,
|
|
parity=serial.PARITY_NONE, stopbits=serial.STOPBITS_ONE,
|
|
bytesize=serial.EIGHTBITS, timeout=0.1)
|
|
except Exception as e:
|
|
print("Cannot open port : " + port)
|
|
sys.exit(e.args[0])
|
|
|
|
def readline(self, size=None):
|
|
return self.serial.readline(size)
|
|
|
|
def write(self, buffer):
|
|
self.serial.write(buffer)
|
|
self.serial.flush()
|
|
|
|
def discard_inputs(self, timeout=1.0):
|
|
time.sleep(timeout)
|
|
self.serial.flushInput()
|
|
|
|
def getc(self, size, timeout=1):
|
|
self.serial.timeout = timeout
|
|
c = self.serial.read(size)
|
|
self.serial.timeout = 0.1
|
|
return c
|
|
|
|
def putc(self, buffer, timeout=1):
|
|
self.serial.timeout = timeout
|
|
self.serial.write(buffer)
|
|
self.serial.flush()
|
|
self.serial.timeout = 0.1
|
|
self.show_progress(len(buffer))
|
|
|
|
# Note: windows platform dependent code
|
|
def putc_win(self, buffer, timeout=1):
|
|
self.serial.write(buffer)
|
|
self.show_progress(len(buffer))
|
|
while True:
|
|
if self.serial.out_waiting == 0:
|
|
break
|
|
|
|
def setBaudrate(self, baudrate):
|
|
# self.serial.setBaudrate(baudrate)
|
|
self.serial.baudrate = baudrate
|
|
|
|
def reboot(self):
|
|
# Target Reset by DTR
|
|
self.serial.setDTR(False)
|
|
self.serial.setDTR(True)
|
|
self.serial.setDTR(False)
|
|
|
|
def set_file_size(self, filesize):
|
|
self.bytes_transfered = 0
|
|
self.filesize = filesize
|
|
self.count = 0
|
|
|
|
def show_progress(self, sendsize):
|
|
if PRINT_RAW_COMMAND:
|
|
if self.count < MAX_DOT_COUNT:
|
|
self.bytes_transfered = self.bytes_transfered + sendsize
|
|
cur_count = int(self.bytes_transfered * MAX_DOT_COUNT / self.filesize)
|
|
if MAX_DOT_COUNT < cur_count:
|
|
cur_count = MAX_DOT_COUNT
|
|
for idx in range(cur_count - self.count):
|
|
print('#',end='')
|
|
sys.stdout.flush()
|
|
self.count = cur_count
|
|
if self.count == MAX_DOT_COUNT:
|
|
print("\n")
|
|
|
|
class FlashWriter:
|
|
def __init__(self, protocol_sel=PROTOCOL_SERIAL):
|
|
if protocol_sel == PROTOCOL_TELNET:
|
|
self.serial = TelnetDev()
|
|
else:
|
|
self.serial = SerialDev()
|
|
|
|
def cancel_autoboot(self) :
|
|
boot_msg = ''
|
|
self.serial.reboot() # Target reboot before send 'r'
|
|
while boot_msg == '' :
|
|
rx = self.serial.readline().strip()
|
|
self.serial.write(b"r") # Send "r" key to avoid auto boot
|
|
for msg in ROM_MSG :
|
|
if msg in rx :
|
|
boot_msg = msg
|
|
break
|
|
while True :
|
|
rx = self.serial.readline().decode(errors="replace").strip()
|
|
if "updater" in rx :
|
|
# Workaround : Sometime first character is dropped.
|
|
# Send line feed as air shot before actual command.
|
|
self.serial.write(b"\n") # Send line feed
|
|
self.serial.discard_inputs()# Clear input buffer to sync
|
|
return boot_msg.decode(errors="ignore")
|
|
|
|
def recv(self):
|
|
rx = self.serial.readline()
|
|
if PRINT_RAW_COMMAND :
|
|
serial_line = rx.decode(errors="replace")
|
|
if serial_line.strip() != "" and not serial_line.startswith(XMDM_MSG):
|
|
print(serial_line, end="")
|
|
return rx
|
|
|
|
def wait(self, string):
|
|
while True:
|
|
rx = self.recv()
|
|
if string.encode() in rx:
|
|
time.sleep(0.1)
|
|
break
|
|
|
|
def wait_for_prompt(self):
|
|
prompt_pat = re.compile(b"updater")
|
|
while True:
|
|
rx = self.recv()
|
|
if prompt_pat.search(rx):
|
|
time.sleep(0.1)
|
|
break
|
|
|
|
def send(self, string):
|
|
self.serial.write(str(string).encode() + b"\n")
|
|
rx = self.serial.readline()
|
|
if PRINT_RAW_COMMAND :
|
|
print(rx.decode(errors="replace"), end="")
|
|
|
|
def read_output(self, prompt_text) :
|
|
output = []
|
|
while True :
|
|
rx = self.serial.readline()
|
|
if prompt_text.encode() in rx :
|
|
time.sleep(0.1)
|
|
break
|
|
if rx != "" :
|
|
output.append(rx.decode(errors="ignore").rstrip())
|
|
return output
|
|
|
|
def install_files(self, files, command) :
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
command += " -b " + ConfigArgs.XMODEM_BAUD
|
|
if os.name == 'nt':
|
|
modem = xmodem.XMODEM(self.serial.getc, self.serial.putc_win, 'xmodem1k')
|
|
else:
|
|
modem = xmodem.XMODEM(self.serial.getc, self.serial.putc, 'xmodem1k')
|
|
for file in files:
|
|
with open(file, "rb") as bin :
|
|
self.send(command)
|
|
print("Install " + file)
|
|
self.wait(XMDM_MSG)
|
|
print("|0%" +
|
|
"-" * (int(MAX_DOT_COUNT / 2) - 6) +
|
|
"50%" +
|
|
"-" * (MAX_DOT_COUNT - int(MAX_DOT_COUNT / 2) - 5) +
|
|
"100%|")
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
self.serial.setBaudrate(ConfigArgs.XMODEM_BAUD)
|
|
self.serial.discard_inputs() # Clear input buffer to sync
|
|
self.serial.set_file_size(os.path.getsize(file))
|
|
modem.send(bin)
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
self.serial.setBaudrate(115200)
|
|
self.wait_for_prompt()
|
|
|
|
def save_files(self, files) :
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
command = "save_file -b " + ConfigArgs.XMODEM_BAUD + " -x "
|
|
else:
|
|
command = "save_file -x "
|
|
if os.name == 'nt':
|
|
modem = xmodem.XMODEM(self.serial.getc, self.serial.putc_win, 'xmodem1k')
|
|
else:
|
|
modem = xmodem.XMODEM(self.serial.getc, self.serial.putc, 'xmodem1k')
|
|
for file in files:
|
|
with open(file, "rb") as bin :
|
|
self.send(command + os.path.basename(file))
|
|
print("Save " + file)
|
|
self.wait(XMDM_MSG)
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
self.serial.setBaudrate(ConfigArgs.XMODEM_BAUD)
|
|
self.serial.discard_inputs() # Clear input buffer to sync
|
|
self.serial.set_file_size(os.path.getsize(file))
|
|
modem.send(bin)
|
|
if ConfigArgs.XMODEM_BAUD:
|
|
self.serial.setBaudrate(115200)
|
|
self.wait_for_prompt()
|
|
self.send("chmod d+rw " + os.path.basename(file))
|
|
self.wait_for_prompt()
|
|
|
|
def delete_files(self, files) :
|
|
for file in files :
|
|
self.delete_binary(file)
|
|
|
|
def delete_binary(self, bin_name) :
|
|
self.send("rm " + bin_name)
|
|
self.wait_for_prompt()
|
|
|
|
def main():
|
|
try:
|
|
config_loader = ConfigArgsLoader()
|
|
config_loader.update_config()
|
|
except:
|
|
return errno.EINVAL
|
|
|
|
# Wait to reset the board
|
|
writer = FlashWriter(ConfigArgs.PROTOCOL_TYPE)
|
|
|
|
do_wait_reset = True
|
|
if ConfigArgs.AUTO_RESET:
|
|
if subprocess.call("cd " + sys.path[0] + "; ./reset_board.sh", shell=True) == 0:
|
|
print("auto reset board success!!")
|
|
do_wait_reset = False
|
|
bootrom_msg = writer.cancel_autoboot()
|
|
|
|
if ConfigArgs.DTR_RESET:
|
|
do_wait_reset = False
|
|
bootrom_msg = writer.cancel_autoboot()
|
|
|
|
if ConfigArgs.WAIT_RESET == False and do_wait_reset == True:
|
|
rx = writer.recv()
|
|
time.sleep(1)
|
|
for i in range(3):
|
|
writer.send("")
|
|
rx = writer.recv()
|
|
if "updater".encode() in rx:
|
|
# No need to wait for reset
|
|
do_wait_reset = False
|
|
break
|
|
time.sleep(1)
|
|
|
|
if do_wait_reset:
|
|
# Wait to reset the board
|
|
print('Please press RESET button on target board')
|
|
sys.stdout.flush()
|
|
bootrom_msg = writer.cancel_autoboot()
|
|
|
|
# Remove files
|
|
if ConfigArgs.ERASE_NAME :
|
|
print(">>> Remove existing files ...")
|
|
writer.delete_files(ConfigArgs.ERASE_NAME)
|
|
|
|
# Install files
|
|
if ConfigArgs.PACKAGE_NAME or ConfigArgs.PKGSYS_NAME or ConfigArgs.PKGAPP_NAME or ConfigArgs.PKGUPD_NAME:
|
|
print(">>> Install files ...")
|
|
if ConfigArgs.PACKAGE_NAME :
|
|
writer.install_files(ConfigArgs.PACKAGE_NAME, "install")
|
|
if ConfigArgs.PKGSYS_NAME :
|
|
writer.install_files(ConfigArgs.PKGSYS_NAME, "install")
|
|
if ConfigArgs.PKGAPP_NAME :
|
|
writer.install_files(ConfigArgs.PKGAPP_NAME, "install")
|
|
if ConfigArgs.PKGUPD_NAME :
|
|
writer.install_files(ConfigArgs.PKGUPD_NAME, "install -k updater.key")
|
|
|
|
# Save files
|
|
if ConfigArgs.FILE_NAME :
|
|
print(">>> Save files ...")
|
|
writer.save_files(ConfigArgs.FILE_NAME)
|
|
|
|
# Set auto boot
|
|
if not ConfigArgs.NO_SET_BOOTABLE:
|
|
print(">>> Save Configuration to FlashROM ...")
|
|
writer.send("set bootable M0P")
|
|
writer.wait_for_prompt()
|
|
|
|
# Sync all cached data to flash
|
|
writer.send("sync")
|
|
writer.wait_for_prompt()
|
|
|
|
if REBOOT_AT_END :
|
|
print("Restarting the board ...")
|
|
writer.send("reboot")
|
|
|
|
return 0
|
|
|
|
if __name__ == "__main__":
|
|
try:
|
|
sys.exit(main())
|
|
except KeyboardInterrupt:
|
|
print("Canceled by keyboard interrupt.")
|
|
pass
|