1
1
# Copyright lowRISC contributors.
2
2
# Licensed under the Apache License, Version 2.0, see LICENSE for details.
3
3
# SPDX-License-Identifier: Apache-2.0
4
-
5
4
r"""CW305 utility functions. Used to configure FPGA with OpenTitan design."""
6
5
7
6
import inspect
@@ -19,6 +18,7 @@ class RuntimePatchFPGAProgram:
19
18
20
19
This class can be used to detect if the FPGA was actually programmed or not.
21
20
"""
21
+
22
22
def __init__ (self , fpga , callback ):
23
23
"""Inits a RuntimePatchFPGAProgram.
24
24
@@ -31,18 +31,21 @@ def __init__(self, fpga, callback):
31
31
self ._orig_fn = fpga .FPGAProgram
32
32
33
33
def __enter__ (self ):
34
+
34
35
def wrapped_fn (* args , ** kwargs ):
35
36
self ._callback ()
36
37
return self ._orig_fn (* args , ** kwargs )
38
+
37
39
self ._fpga .FPGAProgram = wrapped_fn
38
40
39
41
def __exit__ (self , exc_type , exc_value , traceback ):
40
42
self ._fpga .FPGAProgram = self ._orig_fn
41
43
42
44
43
45
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 ):
46
49
47
50
# Extract target board type from bitstream name.
48
51
m = re .search ('cw305|cw310' , bitstream )
@@ -54,11 +57,17 @@ def __init__(self, bitstream, firmware, pll_frequency, baudrate, scope_gain,
54
57
fpga = cw .capture .targets .CW310 ()
55
58
programmer = SpiProgrammer (fpga )
56
59
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' )
58
62
63
+ # Added `pll_frequency` to handle frequencies other than 100MHz.
64
+ # Needed this for OTBN ECDSA.
65
+ # TODO: Remove these comments after discussion
59
66
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 )
62
71
63
72
def initialize_fpga (self , fpga , bitstream , pll_frequency ):
64
73
"""Initializes FPGA bitstream and sets PLL frequency."""
@@ -97,6 +106,9 @@ def program_callback():
97
106
fpga .pll .pll_outenable_set (False , 0 )
98
107
fpga .pll .pll_outenable_set (True , 1 )
99
108
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
100
112
fpga .pll .pll_outfreq_set (pll_frequency , 1 )
101
113
102
114
# Disable USB clock to reduce noise in power traces.
@@ -107,18 +119,23 @@ def program_callback():
107
119
108
120
return fpga
109
121
110
- def initialize_scope (self , scope_gain , num_samples , offset ):
122
+ def initialize_scope (self , scope_gain , num_samples , offset , pll_frequency ):
111
123
"""Initializes chipwhisperer scope."""
112
124
scope = cw .scope ()
113
125
scope .gain .db = scope_gain
114
126
scope .adc .basic_mode = "rising_edge"
115
127
if hasattr (scope , '_is_husky' ) and scope ._is_husky :
116
128
# 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
117
133
scope .clock .adc_mul = 2
118
- scope .clock .clkgen_freq = 100000000
134
+ scope .clock .extclk_monitor_enabled = False
119
135
scope .adc .samples = num_samples
120
- scope . clock . clkgen_src = 'extclk'
136
+
121
137
husky = True
138
+ print (f"Husky? = { husky } " )
122
139
else :
123
140
# We sample using the target clock (100 MHz).
124
141
scope .clock .adc_mul = 1
@@ -144,12 +161,47 @@ def initialize_scope(self, scope_gain, num_samples, offset):
144
161
assert (scope .clock .adc_locked ), "ADC failed to lock"
145
162
return scope
146
163
147
- def initialize_target (self , programmer , firmware , baudrate , output_len ):
164
+ def initialize_target (self , programmer , firmware , baudrate , output_len ,
165
+ pll_frequency ):
148
166
"""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
+
149
173
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
+
150
180
time .sleep (0.5 )
151
181
target = cw .target (self .scope )
152
182
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 )
154
187
target .flush ()
188
+
155
189
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 )
0 commit comments