Skip to content

Commit f51c6ec

Browse files
bilgidayvrozic
authored andcommittedAug 2, 2022
[otbn/ecdsa] Add capture support for ECDSA-256/384
Initial support to capture side-channel traces from OTBN ECDSA-256 and ECDSA-384 applications. New capture commands `ecdsa-simple` and `ecdsa-stream`. Supports only CW-Husky as the capture device. * New files: * `cw/capture_ecdsa_cw310.yaml`: config file for ECDSA-256 experiments * `cw/capture_ecdsa384_cw310.yaml`: config file for ECDSA-384 experiments * `cw/objs/lowrisc_systems_chip_earlgrey_cw310_0.1_ecdsa.bit`: example FPGA bitstream * `cw/objs/ecc384_serial_fpga_cw310.bin`: example ECDSA-384 driver * `cw/objs/ecc_serial_fpga_cw310.bin`: example ECDSA-256 driver * Modified files: * `cw/capture.py` * Add new command `ecdsa-simple` to collect long ECDSA traces without using CW-Husky's streaming mode * Add new command `ecdsa-stream` to collect long ECDSA traces using CW-Husky's streaming mode * `cw/device.py` * Changes to handle PLL frequencies other than 100MHZ * Add `program_target()` for programming the firmware during ECDSA experiments. * Notes: * Tested on CW-310 with a CW-Husky Signed-off-by: Bilgiday Yuce <bilgiday@opentitan.org>
1 parent de113cf commit f51c6ec

8 files changed

+466
-13
lines changed
 

‎cw/capture.py

+344-1
Large diffs are not rendered by default.

‎cw/capture_ecdsa384_cw310.yaml

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
device:
2+
fpga_bitstream: objs/lowrisc_systems_chip_earlgrey_cw310_0.1_ecdsa.bit
3+
fw_bin: objs/ecc384_serial_fpga_cw310.bin
4+
# Target operating frequency
5+
pll_frequency: 22000000
6+
baudrate: 115200
7+
capture:
8+
# Only ECDSA-256 (32) and ECDSA-384 (48) are supported at this moment.
9+
key_len_bytes: 48
10+
plain_text_len_bytes: 48
11+
output_len_bytes: 48
12+
# Samples per trace
13+
num_samples: 262140
14+
# Offset in samples
15+
offset: 0
16+
# scope gain in db - 13 for non-stream mode, 32.5 for stream mode
17+
scope_gain: 32.5
18+
num_traces: 1
19+
project_name: projects/opentitan_ecdsa_384
20+
waverunner_ip: 192.168.1.228
21+
batch_prng_seed: 0
22+
# As OTBN operations are long, we may need to
23+
# overwrite the folloing parameters
24+
adc_mul: 1
25+
decimate: 1
26+
plot_capture:
27+
show: true
28+
num_traces: 1
29+
trace_image_filename: projects/sample_traces_ecdsa384.html

‎cw/capture_ecdsa_cw310.yaml

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
device:
2+
fpga_bitstream: objs/lowrisc_systems_chip_earlgrey_cw310_0.1_ecdsa.bit
3+
fw_bin: objs/ecc_serial_fpga_cw310.bin
4+
# Target operating frequency
5+
pll_frequency: 22000000
6+
baudrate: 115200
7+
capture:
8+
# Only ECDSA-256 (32) and ECDSA-384 (48) are supported at this moment.
9+
key_len_bytes: 32
10+
plain_text_len_bytes: 32
11+
output_len_bytes: 32
12+
# Samples per trace
13+
num_samples: 262140
14+
# Offset in samples
15+
offset: 0
16+
# scope gain in db - 13 for non-stream mode, 32.5 for stream mode
17+
scope_gain: 32.5
18+
num_traces: 1
19+
project_name: projects/opentitan_ecdsa_256
20+
waverunner_ip: 192.168.1.228
21+
batch_prng_seed: 0
22+
# As OTBN operations are long, we may need to
23+
# overwrite the folloing parameters
24+
adc_mul: 1
25+
decimate: 1
26+
plot_capture:
27+
show: true
28+
num_traces: 1
29+
trace_image_filename: projects/sample_traces_ecdsa256.html

‎cw/objs/ecc384_serial_fpga_cw310.bin

25.4 KB
Binary file not shown.

‎cw/objs/ecc_serial_fpga_cw310.bin

23.5 KB
Binary file not shown.
Binary file not shown.

‎cw/util/device.py

+63-11
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
# Copyright lowRISC contributors.
22
# Licensed under the Apache License, Version 2.0, see LICENSE for details.
33
# SPDX-License-Identifier: Apache-2.0
4-
54
r"""CW305 utility functions. Used to configure FPGA with OpenTitan design."""
65

76
import inspect
@@ -19,6 +18,7 @@ class RuntimePatchFPGAProgram:
1918
2019
This class can be used to detect if the FPGA was actually programmed or not.
2120
"""
21+
2222
def __init__(self, fpga, callback):
2323
"""Inits a RuntimePatchFPGAProgram.
2424
@@ -31,18 +31,21 @@ def __init__(self, fpga, callback):
3131
self._orig_fn = fpga.FPGAProgram
3232

3333
def __enter__(self):
34+
3435
def wrapped_fn(*args, **kwargs):
3536
self._callback()
3637
return self._orig_fn(*args, **kwargs)
38+
3739
self._fpga.FPGAProgram = wrapped_fn
3840

3941
def __exit__(self, exc_type, exc_value, traceback):
4042
self._fpga.FPGAProgram = self._orig_fn
4143

4244

4345
class OpenTitan(object):
44-
def __init__(self, bitstream, firmware, pll_frequency, baudrate, scope_gain,
45-
num_samples, offset, output_len):
46+
47+
def __init__(self, bitstream, firmware, pll_frequency, baudrate,
48+
scope_gain, num_samples, offset, output_len):
4649

4750
# Extract target board type from bitstream name.
4851
m = re.search('cw305|cw310', bitstream)
@@ -54,11 +57,17 @@ def __init__(self, bitstream, firmware, pll_frequency, baudrate, scope_gain,
5457
fpga = cw.capture.targets.CW310()
5558
programmer = SpiProgrammer(fpga)
5659
else:
57-
raise ValueError('Could not infer target board type from bistream name')
60+
raise ValueError(
61+
'Could not infer target board type from bistream name')
5862

63+
# Added `pll_frequency` to handle frequencies other than 100MHz.
64+
# Needed this for OTBN ECDSA.
65+
# TODO: Remove these comments after discussion
5966
self.fpga = self.initialize_fpga(fpga, bitstream, pll_frequency)
60-
self.scope = self.initialize_scope(scope_gain, num_samples, offset)
61-
self.target = self.initialize_target(programmer, firmware, baudrate, output_len)
67+
self.scope = self.initialize_scope(scope_gain, num_samples, offset,
68+
pll_frequency)
69+
self.target = self.initialize_target(programmer, firmware, baudrate,
70+
output_len, pll_frequency)
6271

6372
def initialize_fpga(self, fpga, bitstream, pll_frequency):
6473
"""Initializes FPGA bitstream and sets PLL frequency."""
@@ -97,6 +106,9 @@ def program_callback():
97106
fpga.pll.pll_outenable_set(False, 0)
98107
fpga.pll.pll_outenable_set(True, 1)
99108
fpga.pll.pll_outenable_set(False, 2)
109+
# Added `pll_frequency` to handle frequencies other than 100MHz.
110+
# Needed this for OTBN ECDSA.
111+
# TODO: Remove these comments after discussion
100112
fpga.pll.pll_outfreq_set(pll_frequency, 1)
101113

102114
# Disable USB clock to reduce noise in power traces.
@@ -107,18 +119,23 @@ def program_callback():
107119

108120
return fpga
109121

110-
def initialize_scope(self, scope_gain, num_samples, offset):
122+
def initialize_scope(self, scope_gain, num_samples, offset, pll_frequency):
111123
"""Initializes chipwhisperer scope."""
112124
scope = cw.scope()
113125
scope.gain.db = scope_gain
114126
scope.adc.basic_mode = "rising_edge"
115127
if hasattr(scope, '_is_husky') and scope._is_husky:
116128
# We sample using the target clock * 2 (200 MHz).
129+
scope.clock.clkgen_src = 'extclk'
130+
# To fully capture the long OTBN applications,
131+
# we may need to use pll_frequencies other than 100 MHz.
132+
scope.clock.clkgen_freq = pll_frequency
117133
scope.clock.adc_mul = 2
118-
scope.clock.clkgen_freq = 100000000
134+
scope.clock.extclk_monitor_enabled = False
119135
scope.adc.samples = num_samples
120-
scope.clock.clkgen_src = 'extclk'
136+
121137
husky = True
138+
print(f"Husky? = {husky}")
122139
else:
123140
# We sample using the target clock (100 MHz).
124141
scope.clock.adc_mul = 1
@@ -144,12 +161,47 @@ def initialize_scope(self, scope_gain, num_samples, offset):
144161
assert (scope.clock.adc_locked), "ADC failed to lock"
145162
return scope
146163

147-
def initialize_target(self, programmer, firmware, baudrate, output_len):
164+
def initialize_target(self, programmer, firmware, baudrate, output_len,
165+
pll_frequency):
148166
"""Loads firmware image and initializes test target."""
167+
# To fully capture the long OTBN applications,
168+
# we may need to use pll_frequencies other than 100 MHz.
169+
# As the programming works at 100MHz, we set pll_frequency to 100MHz
170+
if pll_frequency != 100e6:
171+
self.fpga.pll.pll_outfreq_set(100e6, 1)
172+
149173
programmer.bootstrap(firmware)
174+
175+
# To handle the PLL frequencies other than 100e6,after programming is done,
176+
# we switch the pll frequency back to its original value
177+
if pll_frequency != 100e6:
178+
self.fpga.pll.pll_outfreq_set(pll_frequency, 1)
179+
150180
time.sleep(0.5)
151181
target = cw.target(self.scope)
152182
target.output_len = output_len
153-
target.baud = baudrate
183+
# Added `pll_frequency` to handle frequencies other than 100MHz.
184+
# Needed this for OTBN ECDSA.
185+
# TODO: Remove these comments after discussion
186+
target.baud = int(baudrate * pll_frequency / 100e6)
154187
target.flush()
188+
155189
return target
190+
191+
def program_target(self, fw, pll_frequency=100e6):
192+
"""Loads firmware image """
193+
programmer1 = SpiProgrammer(self.fpga)
194+
# To fully capture the long OTBN applications,
195+
# we may need to use pll_frequencies other than 100 MHz.
196+
# As the programming works at 100MHz, we set pll_frequency to 100MHz
197+
if self.scope.clock.clkgen_freq != 100e6:
198+
self.fpga.pll.pll_outfreq_set(100e6, 1)
199+
200+
programmer1.bootstrap(fw)
201+
202+
# To handle the PLL frequencies other than 100e6,after programming is done,
203+
# we switch the pll frequency back to its original value
204+
if self.scope.clock.clkgen_freq != 100e6:
205+
self.fpga.pll.pll_outfreq_set(pll_frequency, 1)
206+
207+
time.sleep(0.5)

‎test/tvla_test.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ def __init__(self, args: Args):
1818
def test_help():
1919
tvla = TvlaCmd(Args('--help')).run()
2020
# Assert that a message is printed on stdout or stderr.
21-
assert(len(tvla.stdout()) != 0 or len(tvla.stderr()) != 0)
21+
assert (len(tvla.stdout()) != 0 or len(tvla.stderr()) != 0)
2222

2323

2424
def ttest_significant(ttest_trace) -> bool:

0 commit comments

Comments
 (0)
Please sign in to comment.