#!/usr/bin/python
#
# Copyright (c) 2016, NORDUnet A/S All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
# - Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# - Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# - Neither the name of the NORDUnet nor the names of its contributors may
# be used to endorse or promote products derived from this software
# without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
# IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
# PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
# TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
# LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""
Utility to upload new a firmware image or FPGA bitstream
"""
import os
import sys
import time
import struct
import serial
import argparse
import getpass
from binascii import crc32
FIRMWARE_CHUNK_SIZE = 4096
FPGA_CHUNK_SIZE = 4096
default_pins = {'wheel': 'YouReallyNeedToChangeThisPINRightNowWeAreNotKidding',
'ct': 'ct',
}
default_timeout = 2
def parse_args():
"""
Parse the command line arguments
"""
parser = argparse.ArgumentParser(description = "File uploader",
add_help = True,
formatter_class = argparse.ArgumentDefaultsHelpFormatter,
)
parser.add_argument('-d', '--device',
dest='device',
default='/dev/ttyUSB0',
help='Name of management port USB serial device',
)
parser.add_argument("--username",
choices = ("so", "wheel", "ct"),
default = "so",
help = "Username to use when logging into the HSM",
)
parser.add_argument("--timeout",
metavar = 'SECONDS',
type = int,
default = default_timeout,
help = "Timeout of commands (seconds)",
)
parser.add_argument('commands', metavar='CMD', type=str, nargs='+',
help='commands to execute')
return parser.parse_args()
def _write(dst, data, debug=False):
dst.write(data)
if not debug:
return
if len(data) == 4:
print("Wrote 0x{!s}".format(data.encode('hex')))
else:
print("Wrote {!r}".format(data))
def _read(dst, retry=10, verbose=True, debug=False):
res = ''
x = dst.read(1)
while not x and retry:
x = dst.read(1)
retry -= 1
while x:
res += x
x = dst.read(1)
if debug:
print ("Read {!r}".format(res))
if verbose:
sys.stdout.write(res)
return res
def _read_until_prompt(dst, retry):
_write(dst, '\r')
while retry:
prompt = _read(dst, retry = 1)
retry -= 1
if prompt.endswith('Username: ') or \
prompt.endswith('Password: ') or \
prompt.endswith('> '):
return prompt
return ''
def _execute(dst, cmd, args):
global default_pins
prompt = _read_until_prompt(dst, retry = args.timeout)
if prompt.endswith('Username: '):
_write(dst, args.username + "\r", debug = True)
prompt = _read(dst, retry = args.timeout)
if prompt.endswith('Password: '):
pin = default_pins.get(args.username)
if not pin:
pin = getpass.getpass("{} PIN: ".format(args.username))
_write(dst, pin + '\r')
prompt = _read(dst, retry = args.timeout)
if not prompt.endswith('> '):
#sys.stderr.write('Device does not seem to be ready for a file transfer (got {!r})\n'.format(prompt))
return prompt
_write(dst, cmd)
response = _read_until_prompt(dst, retry = args.timeout)
return response
def main(args):
global pin
dst = serial.Serial(args.device, 921600, timeout=1)
for this in args.commands:
_execute(dst, this, args)
dst.close()
return True
if __name__ == '__main__':
try:
args = parse_args()
if main(args):
sys.exit(0)
sys.exit(1)
except KeyboardInterrupt:
pass