## along with this program; if not, see <http://www.gnu.org/licenses/>.
##
+# This utility extracts FX2 MCU firmware and FPGA bitstream images from
+# the "KingstVIS" vendor software. The blobs are kept in Qt resources
+# sections. The script was tested with several v3.5 software versions.
+
+import argparse
import os
import sys
import re
import struct
import codecs
import importlib.util
+import zlib
-# reuse parseelf.py module from saleae-logic16:
+# Reuse the parseelf.py module from saleae-logic16.
fwdir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
parseelf_py = os.path.join(fwdir, "saleae-logic16", "parseelf.py")
spec = importlib.util.spec_from_file_location("parseelf", parseelf_py)
s = self._elf.read_section(shdr), shdr
self._elf_sections[idx] = s
return s
-
+
def _get_elf_sym_value(self, sname):
sym = self._elf.symtab[sname]
section, shdr = self._get_elf_section(sym["st_shndx"])
print("warning: symbol %s should be %d bytes, but in section is only %d bytes" % (
sname, sym["st_size"], len(value)))
return value
-
- # qt resource stuff:
+
+ # Qt resource stuff.
def _get_resource_name(self, offset):
length, i = struct.unpack(">HI", self._res_names[offset:offset + 2 + 4])
offset += 2 + 4
name = self._res_names[offset:offset + 2 * length].decode("utf-16be")
return name
-
+
def _get_resource_data(self, offset):
length = struct.unpack(">I", self._res_datas[offset:offset + 4])[0]
offset += 4
return self._res_datas[offset:offset + length]
-
+
def _read_resources(self):
RCCFileInfo_Directory = 0x02
def read_table():
child_count, first_child = which[2:]
for i in range(child_count):
child = table[first_child + i]
- if child[1] & RCCFileInfo_Directory:
+ if child[1] & RCCFileInfo_Directory:
read_dir_entries(table, child, parents + [child[0]])
else:
country, language, data_offset = child[2:]
full_name = "/".join(parents + [child[0]])
self._resources[full_name] = data_offset
-
+
self._res_datas = self._get_elf_sym_value("_ZL16qt_resource_data")
self._res_names = self._get_elf_sym_value("_ZL16qt_resource_name")
self._res_struct = self._get_elf_sym_value("_ZL18qt_resource_struct")
-
+
self._resources = {} # res_fn -> res_offset
table = read_table()
read_dir_entries(table, table[0])
m = re.match(res_fn_re, key)
if m is not None:
yield key
-
+
class res_writer(object):
def __init__(self, res):
self.res = res
-
+
def _write_file(self, fn, data, decoder=None, zero_pad_to=None):
if decoder is not None:
data = decoder(data)
data += b"\0" * (zero_pad_to - len(data))
with open(fn, "wb") as fp:
fp.write(data)
- print("saved %d bytes to %s" % (len(data), fn))
-
+ data_crc32 = zlib.crc32(data) & 0xffffffff
+ print("saved %d bytes to %s (crc32=%08x)" % (len(data), fn, data_crc32))
+
def extract(self, res_fn, out_fn, decoder=None, zero_pad_to=None):
self._write_file(out_fn, self.res.get_resource(res_fn), decoder=decoder, zero_pad_to=zero_pad_to)
-
+
def extract_re(self, res_fn_re, out_fn, decoder=None):
for res_fn in res.find_resource_names(res_fn_re):
fn = re.sub(res_fn_re, out_fn, res_fn).lower()
""" return list of (address, data)
"""
datas = []
- # assume \n or \r\n*
+ # Assume LF-only or CR-LF style end-of-line.
for line in hexdata.split(b"\n"):
line = line.strip()
if chr(line[0]) != ":": raise Exception("invalid line: %r" % line)
def maybe_intel_hex_as_blob(data):
if data[0] == ord(":") and max(data) < 127:
return intel_hex_as_blob(data)
- return data # keep binary data
+ return data # Keep binary data.
if __name__ == "__main__":
- if len(sys.argv) != 2:
- print("sigrok-fwextract-kingst-la2016 <programfile>")
- sys.exit()
-
- res = qt_resources(sys.argv[1])
-
+ parser = argparse.ArgumentParser(description = "KingstVIS firmware extraction")
+ parser.add_argument('executable', help = "KingstVIS executable file")
+ options = parser.parse_args()
+ exe_fn = options.executable
+
+ res = qt_resources(exe_fn)
writer = res_writer(res)
- writer.extract("fwfpga/LA2016A", "kingst-la2016a-fpga.bitstream", zero_pad_to=180224)
+
+ # Extract all MCU firmware and FPGA bitstream images. The sigrok
+ # project may not cover all KingstVIS supported devices. Users can
+ # either just copy those files which are strictly required for their
+ # specific device (diagnostics will identify those). Or just copy a
+ # few more files while some of them remain unused later (their size
+ # is small). Seeing which files would be contained is considered
+ # valuable, to identify device variants or candidate models.
writer.extract_re(r"fwusb/fw(.*)", r"kingst-la-\1.fw", decoder=maybe_intel_hex_as_blob)
+ writer.extract_re(r"fwfpga/(.*)", r"kingst-\1-fpga.bitstream")