Skip to content

Commit e5509c1

Browse files
authored
Merge pull request NordicSemiconductor#78 from wbober/thread
Add support for Thread DFU
2 parents 5bd13cb + fe02f2b commit e5509c1

11 files changed

+15611
-13
lines changed

README.md

+20-4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ This application and its library offer the following features:
1313
* Cryptographic key generation, management and storage
1414
* Bootloader DFU settings generation and display
1515
* Device Firmware Update procedure over Bluetooth Low Energy
16+
* Device Firmware Update procedure over Thread
1617

1718
## License
1819

@@ -41,7 +42,7 @@ This will also retrieve and install all additional required packages.
4142

4243
**Note**: Please refer to the [pc-ble-driver-py PyPI installation note on Windows](https://github.com/NordicSemiconductor/pc-ble-driver-py#installing-from-pypi) if you are running nrfutil on this operating system.
4344

44-
**Note**: To use the `dfu ble` option you will need to set up your boards to be able to communicate with your computer. You can find additional information here: [Hardware setup](https://github.com/NordicSemiconductor/pc-ble-driver/blob/master/Installation.md#hardware-setup).
45+
**Note**: To use the `dfu ble` or `dfu thread` option you will need to set up your boards to be able to communicate with your computer. You can find additional information here: [Hardware setup](https://github.com/NordicSemiconductor/pc-ble-driver/blob/master/Installation.md#hardware-setup).
4546

4647
## Downloading precompiled Windows executable
4748

@@ -99,11 +100,11 @@ pyinstaller /full/path/to/nrfutil.spec
99100

100101
**Note**: Please refer to the [pc-ble-driver-py PyPI installation note on Windows](https://github.com/NordicSemiconductor/pc-ble-driver-py#installing-from-pypi) if you are running nrfutil on this operating system.
101102

102-
**Note**: To use the `dfu ble` option you will need to set up your boards to be able to communicate with your computer. You can find additional information here: [Hardware setup](https://github.com/NordicSemiconductor/pc-ble-driver/blob/master/Installation.md#hardware-setup).
103+
**Note**: To use the `dfu ble` or `dfu thread` option you will need to set up your boards to be able to communicate with your computer. You can find additional information here: [Hardware setup](https://github.com/NordicSemiconductor/pc-ble-driver/blob/master/Installation.md#hardware-setup).
103104

104105
## Usage
105106

106-
To get info on usage of nrfutil:
107+
To get info on usage of nrfutil:
107108
```
108109
nrfutil --help
109110
```
@@ -144,6 +145,8 @@ SoftDevice | FWID (sd-req)
144145
`s132_nrf52_4.0.2` | 0x98
145146
`s132_nrf52_4.0.3` | 0x99
146147

148+
**Note**: The Thread stack doesn't use a SoftDevice but --sd-req option is required for compatibility reasons. You can provide any value for the option as it is ignored during DFU.
149+
147150
Not all combinations of Bootloader, SoftDevice and Application are possible when generating a package. The table below summarizes the support for different combinations.
148151

149152
The following conventions are used on the table:
@@ -169,7 +172,7 @@ nrfutil pkg display package.zip
169172
```
170173

171174
#### dfu
172-
This set of commands allow you to perform an actual firmware update over a serial or BLE connection.
175+
This set of commands allow you to perform an actual firmware update over a serial, BLE, or Thread connection.
173176

174177
##### ble
175178
Perform a full DFU procedure over a BLE connection. This command takes several options that you can list using:
@@ -182,6 +185,19 @@ nrfutil dfu ble -ic NRF52 -pkg app_dfu_package.zip -p COM3 -n "MyDevice" -f
182185
```
183186
The `-f` option instructs nrfutil to actually program the board connected to COM3 with the connectivity software required to operate as a serialized SoftDevice. Use with caution as this will overwrite the contents of the IC's flash memory.
184187

188+
##### Thread
189+
**Note**: DFU over Thread is experimental
190+
191+
Perform a full DFU procedure over a Thread. This command takes several options that you can list using:
192+
```
193+
nrfutil dfu thread --help
194+
```
195+
Below is an example of the execution of a DFU procedure on all devices in a Thread network using a file generated above and a connectivity IC connected to COM3.
196+
```
197+
nrfutil dfu ble -pkg app_dfu_package.zip -p COM3 -f
198+
```
199+
The `-f` option instructs nrfutil to actually program the board connected to COM3 with the connectivity software required to operate as a network co-processor (NCP). Use with caution as this will overwrite the contents of the IC's flash memory.
200+
185201
##### serial
186202
Perform a full DFU procedure over a serial (UART) line. This command takes several options that you can list using:
187203
```

nordicsemi/__main__.py

+130-3
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,9 @@
3434
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
3535
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
3636
#
37+
import ipaddress
38+
import signal
39+
from nordicsemi.thread.dfu_server import ThreadDfuServer
3740

3841
"""nrfutil command line tool."""
3942
import os
@@ -110,6 +113,11 @@ def int_as_text_to_int(value):
110113
except ValueError:
111114
raise NordicSemiException('%s is not a valid integer' % value)
112115

116+
def pause():
117+
try:
118+
raw_input()
119+
except (KeyboardInterrupt, EOFError):
120+
pass
113121

114122
class BasedIntOrNoneParamType(click.ParamType):
115123
name = 'Integer'
@@ -547,7 +555,7 @@ def update_progress(progress=0):
547555
if global_bar:
548556
global_bar.update(progress)
549557

550-
@cli.group(short_help='Perform a Device Firmware Update over a BLE or serial transport.')
558+
@cli.group(short_help='Perform a Device Firmware Update over, BLE, Thread, or serial transport.')
551559
def dfu():
552560
"""
553561
This set of commands supports Device Firmware Upgrade procedures over both BLE and serial transports.
@@ -598,7 +606,7 @@ def serial(package, port, flow_control, packet_receipt_notification, baud_rate):
598606
serial_backend.register_events_callback(DfuEvent.PROGRESS_EVENT, update_progress)
599607
dfu = Dfu(zip_file_path = package, dfu_transport = serial_backend)
600608

601-
if logger.getEffectiveLevel() > logging.INFO:
609+
if logger.getEffectiveLevel() > logging.INFO:
602610
with click.progressbar(length=dfu.dfu_get_total_size()) as bar:
603611
global global_bar
604612
global_bar = bar
@@ -607,7 +615,7 @@ def serial(package, port, flow_control, packet_receipt_notification, baud_rate):
607615
dfu.dfu_send_images()
608616

609617
click.echo("Device programmed.")
610-
618+
611619

612620
def enumerate_ports():
613621
descs = BLEDriver.enum_serial_ports()
@@ -620,6 +628,13 @@ def enumerate_ports():
620628
index = click.prompt('Enter your choice: ', type=click.IntRange(0, len(descs)))
621629
return descs[index].port
622630

631+
def get_port_by_snr(snr):
632+
serial_ports = BLEDriver.enum_serial_ports()
633+
try:
634+
serial_port = [d.port for d in serial_ports if d.serial_number.lstrip('0') == snr][0]
635+
except IndexError:
636+
raise NordicSemiException('board not found')
637+
return serial_port
623638

624639
@dfu.command(short_help="Update the firmware on a device over a BLE connection.")
625640
@click.option('-pkg', '--package',
@@ -697,6 +712,118 @@ def convert_version_string_to_int(s):
697712
js = [10000, 100, 1]
698713
return sum([js[i] * int(numbers[i]) for i in range(3)])
699714

715+
@dfu.command(short_help="Update the firmware on a device over a Thread connection.")
716+
@click.option('-pkg', '--package',
717+
help='Filename of the DFU package.',
718+
type=click.Path(exists=True, resolve_path=True, file_okay=True, dir_okay=False),
719+
required=True)
720+
@click.option('-p', '--port',
721+
help='Serial port COM port to which the NCP is connected.',
722+
type=click.STRING)
723+
@click.option('-a', '--address',
724+
help='Device IPv6 address. If address is not specified then perform DFU'
725+
+ 'on all capable devices.',
726+
type=click.STRING)
727+
@click.option('-sp', '--server_port',
728+
help='UDP port to which the DFU server binds. If not specified the 5683 is used.',
729+
type=click.INT,
730+
default=5683)
731+
@click.option('--prefix',
732+
help='URI prefix used added to DFU resources. Defaults to ''dfu''.',
733+
type=click.STRING,
734+
default='dfu')
735+
@click.option('--panid',
736+
help='802.15.4 PAN ID. If not specified then 1234 is used as PAN ID.',
737+
type=click.INT)
738+
@click.option('--channel',
739+
help='802.15.4 Channel. If not specified then channel 11 is used.',
740+
type=click.INT)
741+
@click.option('-snr', '--jlink_snr',
742+
help='Jlink serial number.',
743+
type=click.STRING)
744+
@click.option('-f', '--flash_connectivity',
745+
help='Flash connectivity firmware automatically. Default: disabled.',
746+
type=click.BOOL,
747+
is_flag=True)
748+
@click.option('-s', '--sim',
749+
help='Use software NCP and connect to the OT simulator.',
750+
type=click.BOOL,
751+
is_flag=True)
752+
def thread(package, port, address, server_port, prefix, panid, channel, jlink_snr, flash_connectivity, sim):
753+
ble_driver_init('NRF52')
754+
from nordicsemi.thread import tncp
755+
from nordicsemi.thread.dfu_thread import create_dfu_server
756+
from nordicsemi.thread.tncp import NCPTransport
757+
from nordicsemi.thread.ncp_flasher import NCPFlasher
758+
759+
"""Perform a Device Firmware Update on a device with a bootloader that supports Thread DFU."""
760+
if address is None:
761+
address = ipaddress.ip_address(u"ff03::1")
762+
click.echo("Address not specified. Using ff03::1 (all Thread nodes)")
763+
else:
764+
try:
765+
address = ipaddress.ip_address(address)
766+
except:
767+
click.echo("Invalid IPv6 address")
768+
return
769+
770+
if (not sim):
771+
if port is None and jlink_snr is None:
772+
click.echo("Please specify serial port or Jlink serial number.")
773+
return
774+
775+
elif port is None:
776+
port = get_port_by_snr(jlink_snr)
777+
if port is None:
778+
click.echo("\nNo Segger USB CDC ports found, please connect your board.")
779+
return
780+
781+
stream_descriptor = 'u:' + port
782+
logger.info("Using connectivity board at serial port: {}".format(port))
783+
else:
784+
stream_descriptor = 'p:' + Flasher.which('ot-ncp') + ' 30'
785+
logger.info("Using ot-ncp binary: {}".format(stream_descriptor))
786+
787+
if flash_connectivity:
788+
flasher = NCPFlasher(serial_port=port, snr = jlink_snr)
789+
if flasher.fw_check():
790+
click.echo("Board already flashed with connectivity firmware.")
791+
else:
792+
click.echo("Flashing connectivity firmware...")
793+
flasher.fw_flash()
794+
click.echo("Connectivity firmware flashed.")
795+
796+
flasher.reset()
797+
798+
# Delay is required because NCP needs time to initialize.
799+
time.sleep(1.0)
800+
801+
config = tncp.NCPTransport.get_default_config()
802+
if (panid):
803+
config[tncp.NCPTransport.CFG_KEY_PANID] = panid
804+
if (channel):
805+
config[tncp.NCPTransport.CFG_KEY_CHANNEL] = channel
806+
if (flash_connectivity):
807+
config[tncp.NCPTransport.CFG_KEY_RESET] = False
808+
809+
transport = NCPTransport(server_port, stream_descriptor, config)
810+
dfu = create_dfu_server(transport, package, prefix)
811+
812+
try:
813+
sighandler = lambda signum, frame : dfu.stop
814+
signal.signal(signal.SIGINT, sighandler)
815+
signal.signal(signal.SIGTERM, sighandler)
816+
817+
dfu.start()
818+
dfu.trigger(address, 3)
819+
click.echo("Press <ENTER> terminate")
820+
pause()
821+
click.echo("Terminating")
822+
823+
except Exception as e:
824+
logger.exception(e)
825+
finally:
826+
dfu.stop()
700827

701828
if __name__ == '__main__':
702829
cli()

nordicsemi/thread/__init__.py

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
#
2+
# Copyright (c) 2016 Nordic Semiconductor ASA
3+
# All rights reserved.
4+
#
5+
# Redistribution and use in source and binary forms, with or without modification,
6+
# are permitted provided that the following conditions are met:
7+
#
8+
# 1. Redistributions of source code must retain the above copyright notice, this
9+
# list of conditions and the following disclaimer.
10+
#
11+
# 2. Redistributions in binary form must reproduce the above copyright notice, this
12+
# list of conditions and the following disclaimer in the documentation and/or
13+
# other materials provided with the distribution.
14+
#
15+
# 3. Neither the name of Nordic Semiconductor ASA nor the names of other
16+
# contributors to this software may be used to endorse or promote products
17+
# derived from this software without specific prior written permission.
18+
#
19+
# 4. This software must only be used in or with a processor manufactured by Nordic
20+
# Semiconductor ASA, or in or with a processor manufactured by a third party that
21+
# is used in combination with a processor manufactured by Nordic Semiconductor.
22+
#
23+
# 5. Any software provided in binary or object form under this license must not be
24+
# reverse engineered, decompiled, modified and/or disassembled.
25+
#
26+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
27+
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
28+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
29+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
30+
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
31+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
32+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
33+
# ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
35+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36+
#
37+
38+
"""Package marker file."""
39+
40+
import sys
41+
from os.path import dirname
42+
43+
sys.path.append(dirname(__file__))

0 commit comments

Comments
 (0)