forked from lowRISC/opentitan
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdif_spi_device.h
1264 lines (1171 loc) · 44.5 KB
/
dif_spi_device.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
#ifndef OPENTITAN_SW_DEVICE_LIB_DIF_DIF_SPI_DEVICE_H_
#define OPENTITAN_SW_DEVICE_LIB_DIF_DIF_SPI_DEVICE_H_
/**
* @file
* @brief <a href="/hw/ip/spi_device/doc/">SPI Device</a> Device Interface
* Functions
*/
#include <stddef.h>
#include <stdint.h>
#include "sw/device/lib/base/macros.h"
#include "sw/device/lib/base/mmio.h"
#include "sw/device/lib/dif/dif_base.h"
#include "sw/device/lib/dif/autogen/dif_spi_device_autogen.h"
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus
/**
* The mode that the spi device operates in.
*/
typedef enum dif_spi_device_mode {
/**
* In the generic firmware mode, the hardware dumps incoming data to SRAM and
* outgoing data from the SRAM.
*/
kDifSpiDeviceModeGeneric = 0,
/**
* In flash emulation mode, the hardware behaves like a SPI NOR flash device.
*/
kDifSpiDeviceModeFlashEmulation = 1,
/**
* In pass-through mode, the hardware forwards commands to another SPI flash
* device, with various tables providing rules for filtering and forwarding.
* The hardware may be configured to also behave like a SPI NOR flash device,
* with some commands and/or address regions targeting internal handling,
* instead of being passed through.
*/
kDifSpiDeviceModePassthrough = 2,
} dif_spi_device_mode_t;
/**
* A signal edge type: positive or negative.
*/
typedef enum dif_spi_device_edge {
/**
* Represents a positive edge (i.e., from low to high).
*/
kDifSpiDeviceEdgePositive,
/**
* Represents a negative edge (i.e., from high to low).
*/
kDifSpiDeviceEdgeNegative,
} dif_spi_device_edge_t;
/**
* A bit ordering within a byte.
*/
typedef enum dif_spi_device_bit_order {
/**
* Represents the most-significant-bit to least-significant-bit order.
*/
kDifSpiDeviceBitOrderMsbToLsb,
/**
* Represents the least-significant-bit to most-significant-bit order.
*/
kDifSpiDeviceBitOrderLsbToMsb,
} dif_spi_device_bit_order_t;
typedef struct dif_spi_device_generic_mode_config {
/**
* The length, in bytes, that should be reserved for the RX FIFO.
*
* `kDifSpiDeviceBufferLen / 2` is a good default for this value.
*/
uint16_t rx_fifo_len;
/**
* The length, in bytes, that should be reserved for the TX FIFO.
*
* `kDifSpiDeviceBufferLen / 2` is a good default for this value.
*/
uint16_t tx_fifo_len;
/**
* The number of bus clock cycles that the RX FIFO waits before committing a
* sub-word data item to the SRAM. Only used in Generic Mode.
*/
uint8_t rx_fifo_commit_wait;
} dif_spi_device_generic_mode_config_t;
/**
* Runtime configuration for SPI.
*
* This struct describes runtime information for one-time configuration of the
* hardware.
*/
typedef struct dif_spi_device_config {
dif_spi_device_edge_t clock_polarity;
dif_spi_device_edge_t data_phase;
dif_spi_device_bit_order_t tx_order;
dif_spi_device_bit_order_t rx_order;
dif_spi_device_mode_t device_mode;
union {
dif_spi_device_generic_mode_config_t generic;
} mode_cfg;
} dif_spi_device_config_t;
/**
* Struct containing the relevant run-time information for the DIF.
*/
typedef struct dif_spi_device_handle {
/**
* Device information of the hardware.
*/
dif_spi_device_t dev;
/**
* Configuration information of the hardware.
*/
dif_spi_device_config_t config;
} dif_spi_device_handle_t;
/**
* The length of the SPI device FIFO buffer, in bytes.
*
* Useful for initializing FIFO lengths: for example, for equally-sized FIFOs,
* `rx_fifo_len` and `tx_fifo_len` would be set to `kDifSpiDeviceBufferLen / 2`.
*/
extern const uint16_t kDifSpiDeviceBufferLen;
/**
* Initializes a SPI device handle for use.
*
* @param base_addr The MMIO base address of the spi_device peripheral.
* @param[out] Out param for the initialized handle.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_init_handle(mmio_region_t base_addr,
dif_spi_device_handle_t *spi);
/**
* Configures SPI with runtime information.
*
* This function should need to be called once for the lifetime of `handle`.
*
* @param spi A SPI handle.
* @param config Runtime configuration parameters.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_configure(dif_spi_device_handle_t *spi,
dif_spi_device_config_t config);
/**
* Turn on/off passthrough without changing any other configuration.
*
* This changes the mode to one of flash mode or passthrough mode. The current
* mode must be one of those, else this function will fail.
*
* @param spi A SPI handle.
* @param enable Whether to turn on passthrough mode.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_passthrough_mode(dif_spi_device_handle_t *spi,
dif_toggle_t enable);
/**
* Resets the asynchronous TX FIFO in generic mode.
*
* This function should only be called when the upstream spi host is held in
* reset or otherwise is inactive.
*
* @param spi A SPI handle.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_reset_generic_tx_fifo(dif_spi_device_handle_t *spi);
/**
* Resets the asynchronous RX FIFO in generic mode.
*
* This function should only be called when the upstream spi host is held in
* reset or otherwise is inactive.
*
* @param spi A SPI handle.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_reset_generic_rx_fifo(dif_spi_device_handle_t *spi);
/**
* Enable or disable the clock for the SRAM backing all the various memory and
* FIFO-related functions.
*
* This function should only be called when the upstream spi host is held in
* reset or otherwise is inactive.
*
* @param spi A SPI handle.
* @param enable Whether to enable or disable the clock.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_sram_clock_enable(dif_spi_device_handle_t *spi,
dif_toggle_t enable);
/**
* Get the current enablement state for the clock of the SRAM backing all the
* various memory and FIFO-related functions.
*
* This function should only be called when the upstream spi host is held in
* reset or otherwise is inactive.
*
* @param spi A SPI handle.
* @param[out] enabled Whether the clock is enabled or disabled.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_sram_clock_enable(dif_spi_device_handle_t *spi,
dif_toggle_t *enabled);
/**
* Issues an "abort" to the given SPI device, causing all in-progress IO to
* halt.
*
* Applies only to generic mode.
*
* @param spi A SPI handle.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_abort(dif_spi_device_handle_t *spi);
/**
* Sets up the "FIFO level" (that is, number of bytes present in a particular
* FIFO) at which the TxLevel and RxLevel IRQs will fire.
*
* For the reciept side, when the FIFO overflows `rx_level` (i.e., goes over
* the set level), an interrupt is fired. This can be used to detect that more
* data should be read from the RX FIFO. This is the
* `Spi Buffer -> Main Memory` case.
*
* Conversely, for the transmission side, when the FIFO underflows `tx_level`
* (i.e., goes below the set level), an interrupt is fired. This can be used
* to detect that there is free space to write more data to the TX FIFO.
* This is the `Main Memory -> Spi Buffer` case.
*
* Applies only to generic mode.
*
* @param spi A SPI handle.
* @param rx_level The new RX level, as described above.
* @param tx_level The new TX level, as described above.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_irq_levels(dif_spi_device_handle_t *spi,
uint16_t rx_level,
uint16_t tx_level);
/**
* Returns the number of bytes still pending receipt by software in the RX FIFO.
*
* Applies only to generic mode.
*
* @param spi A SPI handle.
* @param[out] bytes_pending The number of bytes pending
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_rx_pending(const dif_spi_device_handle_t *spi,
size_t *bytes_pending);
/**
* Returns the number of bytes still pending transmission by hardware in the TX
* FIFO.
*
* Applies only to generic mode.
*
* @param spi A SPI handle.
* @param[out] bytes_pending The number of bytes pending
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_tx_pending(const dif_spi_device_handle_t *spi,
size_t *bytes_pending);
/**
* Get the current async FIFO occupancy levels. Only used in generic mode.
*
* @param spi A SPI handle.
* @param[out] rx_fifo_level The occupancy level of the RX FIFO.
* @param[out] tx_fifo_level The occupancy level of the TX FIFO.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_async_fifo_levels(dif_spi_device_handle_t *spi,
uint16_t *rx_fifo_level,
uint16_t *tx_fifo_level);
/** Represents the status of the synchronous FIFOs in generic mode. */
typedef struct dif_spi_device_generic_fifo_status {
/** Whether the RX FIFO is full. */
bool rx_full;
/** Whether the RX FIFO is empty. */
bool rx_empty;
/** Whether the TX FIFO is full. */
bool tx_full;
/** Whether the TX FIFO is empty. */
bool tx_empty;
} dif_spi_device_generic_fifo_status_t;
/**
* Get the current empty/full status for generic mode's synchronous FIFOs.
*
* @param spi A SPI handle.
* @param[out] status The empty/full status for the FIFOs.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_generic_fifo_status(
dif_spi_device_handle_t *spi, dif_spi_device_generic_fifo_status_t *status);
/**
* Get the current level of the CSB pin.
*
* This is available for the esoteric case where CSB is used like a slow GPIO.
* Note that for ordinary SPI operation, software sampling of the CSB pin cannot
* be used to determine whether it is safe to perform an operation where the
* host must be inactive.
*
* @param spi A SPI handle.
* @param[out] csb The current value of the chip-select pin (false for asserted,
* true for deasserted).
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_csb_status(dif_spi_device_handle_t *spi,
bool *csb);
/**
* Reads at most `buf_len` bytes from the RX FIFO; the number of bytes read
* will be written to `bytes_received`.
*
* Applies only to generic mode.
*
* @param spi A SPI device.
* @param[out] buf A pointer to valid memory.
* @param buf_len The length of the buffer `buf` points to.
* @param[out] bytes_received The number of bytes successfully read; may be
* null.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_recv(dif_spi_device_handle_t *spi, void *buf,
size_t buf_len, size_t *bytes_received);
/**
* Writes at most `buf_len` bytes to the TX FIFO; the number of bytes actually
* written will be written to `bytes_sent`.
*
* Applies only to generic mode.
*
* @param spi A SPI device.
* @param buf A pointer to bytes to be written.
* @param buf_len The length of the buffer `buf` points to.
* @param[out] bytes_sent The number of bytes successfully written; may be null.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_send(dif_spi_device_handle_t *spi, const void *buf,
size_t buf_len, size_t *bytes_sent);
/**
* Writes `buf_len` bytes to the TX FIFO, blocking until all bytes can be
* written.
*
* Applies only to generic mode.
*
* @param spi A SPI device.
* @param buf A pointer to bytes to be written.
* @param buf_len The length of the buffer `buf` points to.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_send_polled(dif_spi_device_handle_t *spi,
const void *buf, size_t buf_len);
/**
* Enable the mailbox region for spi_device flash / passthrough modes.
*
* Allocate 1 KiB for the mailbox, starting from the provided base `address`.
*
* @param spi A SPI device.
* @param address The base address of the 1 KiB mailbox. The lower 10 bits are
* ignored.
* @return kDifBadArg if spi is null, else kDifOk.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_enable_mailbox(dif_spi_device_handle_t *spi,
uint32_t address);
/**
* Disable the mailbox region for spi_device flash / passthrough modes.
*
* @param spi A SPI device.
* @return kDifBadArg if spi is null, else kDifOk.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_disable_mailbox(dif_spi_device_handle_t *spi);
/**
* Get the active configuration for the mailbox region.
*
* @param spi A SPI device.
* @param[out] is_enabled Whether the mailbox region is enabled.
* @param[out] address If the mailbox is enabled, the base address of the
* mailbox region.
* @return kDifBadArg if any argument is null, else kDifOk.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_mailbox_configuration(
dif_spi_device_handle_t *spi, dif_toggle_t *is_enabled, uint32_t *address);
/**
* Set the address mode of the SPI device in flash/passthrough mode.
*
* @param spi A SPI device.
* @param addr_4b If kDifToggleEnabled, set the address mode to 4 Bytes.
* Otherwise, set the address mode to 3 Bytes.
* @return kDifBadArg if spi is NULL or addr_4b is not a valid toggle. kDifOk
* otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_4b_address_mode(dif_spi_device_handle_t *spi,
dif_toggle_t addr_4b);
/**
* Get the address mode of the SPI device in flash/passthrough mode.
*
* @param spi A SPI device.
* @param[out] addr_4b Points to toggle that will be set to `kDifToggleEnabled`
* if the device is in 4-byte address mode, else `kDifToggleDisabled`.
* @return kDifBadArg if spi or addr_4b are NULL. kDifOk otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_4b_address_mode(dif_spi_device_handle_t *spi,
dif_toggle_t *addr_4b);
typedef struct dif_spi_device_flash_id {
/** The device ID for the SPI flash. */
uint16_t device_id;
/** The JEDEC manufacturer ID. */
uint8_t manufacturer_id;
/** The continuation code used before the `manufacturer_id` byte. */
uint8_t continuation_code;
/** The number of continuation codes preceding the `manufacturer_id`. */
uint8_t num_continuation_code;
} dif_spi_device_flash_id_t;
/**
* Get the JEDEC ID presented when in flash / passthrough modes.
*
* @param spi A SPI device.
* @param[out] id Points to location that will be set to the current JEDEC ID.
* @return kDifBadArg if spi or id are NULL. kDifOk otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_id(dif_spi_device_handle_t *spi,
dif_spi_device_flash_id_t *id);
/**
* Set the JEDEC ID presented when in flash / passthrough modes.
*
* @param spi A SPI device.
* @param id The JEDEC ID to set.
* @return kDifBadArg if spi is NULL. kDifOk otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_flash_id(dif_spi_device_handle_t *spi,
dif_spi_device_flash_id_t id);
/**
* Represents which optional hardware features may intercept commands in
* passthrough mode. If selected, the function will be handled internally and
* not passed to the downstream SPI flash (sometimes with conditions, such as
* the address matching the mailbox region when mailbox interception is
* selected).
*/
typedef struct dif_spi_device_passthrough_intercept_config {
/**
* Whether to intercept commands to the status registers, such as WriteStatus
* and ReadStatus.
*/
bool status;
/**
* Whether to intercept the ReadID command and respond with the JEDEC ID
* programmed via `dif_spi_device_set_flash_id()`.
*/
bool jedec_id;
/**
* Whether to intercept the ReadSFDP command with data programmed to the
* internal SFDP region.
*/
bool sfdp;
/**
* Whether to intercept read memory commands with data programmed in the
* mailbox buffer, when the address is inside the mailbox region.
*/
bool mailbox;
} dif_spi_device_passthrough_intercept_config_t;
/**
* Configure the optional hardware functions to intercept passthrough commands.
*
* @param spi A SPI handle.
* @param config The passthrough interception configuration.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_passthrough_intercept_config(
dif_spi_device_handle_t *spi,
dif_spi_device_passthrough_intercept_config_t config);
/**
* Get the last address read from the flash memory that was not in the mailbox
* region.
*
* @param spi A SPI handle.
* @param[out] address The last address read.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_last_read_address(dif_spi_device_handle_t *spi,
uint32_t *address);
/**
* Set the read threshold watermark for reporting to the corresponding interrupt
* status bit.
*
* This function is intended to be used in flash mode for devices that
* sequentially read from address 0 to the end of some initial block. A supplied
* address of 0 disables the watermark status bit and interrupt.
*
* @param spi A SPI handle.
* @param address The watermark address that triggers reporting to the read
* threshold watermark's status bit, which can signal an interrupt.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_eflash_read_threshold(
dif_spi_device_handle_t *spi, uint32_t address);
typedef enum dif_spi_device_flash_address_type {
/** No address for this command */
kDifSpiDeviceFlashAddrDisabled = 0,
/** Address size for this command is determined by the current address mode */
kDifSpiDeviceFlashAddrCfg,
/** Address size for this command is fixed at 3 bytes */
kDifSpiDeviceFlashAddr3Byte,
/** Address size for this command is fixed at 4 bytes */
kDifSpiDeviceFlashAddr4Byte,
kDifSpiDeviceFlashAddrCount,
} dif_spi_device_flash_address_type_t;
/** An enum describing the type of I/O width used by a command's payload. */
typedef enum dif_spi_device_payload_io {
/** Command does not have a payload. */
kDifSpiDevicePayloadIoNone = 0x0,
/** Command's payload has an I/O channel width of 1. */
kDifSpiDevicePayloadIoSingle = 0x1,
/** Command's payload has an I/O channel width of 2. */
kDifSpiDevicePayloadIoDual = 0x3,
/** Command's payload has an I/O channel width of 4. */
kDifSpiDevicePayloadIoQuad = 0xf,
/** The command info slot had an invalid setting for the payload I/O width. */
kDifSpiDevicePayloadIoInvalid = 0x10,
} dif_spi_device_payload_io_t;
typedef struct dif_spi_device_flash_command {
/** The opcode for this command. */
uint8_t opcode;
/* The presence and type of the address phase of this command. */
dif_spi_device_flash_address_type_t address_type;
/** The number of dummy cycles between the address phase and the payload phase
* of a command. Note that if `payload_io_width` is
* `kDifSpiDevicePayloadIoNone`, a nonzero value may cause undefined behavior,
* possibly including the device assuming dummy cycles are part of this
* payload-less command.
*/
uint8_t dummy_cycles;
/** The I/O width for the payload phase of this command. */
dif_spi_device_payload_io_t payload_io_type;
/**
* Whether to translate the address using the address swap mask and value
* provided to `dif_spi_device_passthrough_set_swap_address()`. Incompatible
* with a command with `address_type` that is
* `kDifSpiDeviceFlashAddrDisabled`.
*/
bool passthrough_swap_address;
/** Whether the SPI host is receiving this command's payload phase. */
bool payload_dir_to_host;
/** Whether to swap up to the first 32 bits of the payload. */
bool payload_swap_enable;
/** Whether to upload the command to the payload FIFO. */
bool upload;
/** Whether to set the busy bit in the status register. */
bool set_busy_status;
} dif_spi_device_flash_command_t;
typedef enum dif_spi_device_flash_buffer_type {
/** eFlash region */
kDifSpiDeviceFlashBufferTypeEFlash = 0,
/** Mailbox region */
kDifSpiDeviceFlashBufferTypeMailbox,
/** SFDP region */
kDifSpiDeviceFlashBufferTypeSfdp,
/** Payload for uploaded commands */
kDifSpiDeviceFlashBufferTypePayload,
/** Count of buffer types */
kDifSpiDeviceFlashBufferTypes,
} dif_spi_device_flash_buffer_type_t;
/**
* Set up the indicated command info slot for flash / passthrough modes.
*
* @param spi A handle to a spi device.
* @param slot A command info slot ID.
* @param enable Whether to enable or disable the command slot.
* @param command_info If `enable` is set, provides the configuration for the
* command.
* @return `kDifBadArg` if `spi` is NULL or `slot` is larger than the number of
* command slots. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_flash_command_slot(
dif_spi_device_handle_t *spi, uint8_t slot, dif_toggle_t enable,
dif_spi_device_flash_command_t command_info);
/**
* Get the current configuration of the indicated command info slot.
*
* @param spi A handle to a spi device.
* @param slot A command info slot ID.
* @param[out] enabled A pointer to where the current slot state can be stored.
* @param[out] command_info If `enabled`, points to where the current command
* configuration can be stored.
* @return `kDifBadArg` if any pointers are NULL or `slot` is larger than the
* number of command slots. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_command_slot(
dif_spi_device_handle_t *spi, uint8_t slot, dif_toggle_t *enabled,
dif_spi_device_flash_command_t *command_info);
/**
* Configure the command properties of the hardware's EN4B function.
*
* EN4B is the command to activate 4-byte addressing for flash and passthrough
* modes.
*
* @param spi A handle to a spi_device.
* @param enable Whether to enable the function.
* @param opcode Which opcode activates the function.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_configure_flash_en4b_command(
dif_spi_device_handle_t *spi, dif_toggle_t enable, uint8_t opcode);
/**
* Configure the command properties of the hardware's EX4B function.
*
* EX4B is the command to deactivate 4-byte addressing for flash and passthrough
* modes. This would return the device to the 3-byte address mode.
*
* @param spi A handle to a spi_device.
* @param enable Whether to enable the function.
* @param opcode Which opcode activates the function.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_configure_flash_ex4b_command(
dif_spi_device_handle_t *spi, dif_toggle_t enable, uint8_t opcode);
/**
* Configure the command properties of the hardware's WREN function.
*
* WREN is the command to enable writes and set the WEL bit of the status
* register, for flash and passthroug modes.
*
* @param spi A handle to a spi_device.
* @param enable Whether to enable the function.
* @param opcode Which opcode activates the function.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_configure_flash_wren_command(
dif_spi_device_handle_t *spi, dif_toggle_t enable, uint8_t opcode);
/**
* Configure the command properties of the hardware's WRDI function.
*
* WRDI is the command to disable writes and clear the WEL bit of the status
* register, for flash and passthrough modes.
*
* @param spi A handle to a spi_device.
* @param enable Whether to enable the function.
* @param opcode Which opcode activates the function.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_configure_flash_wrdi_command(
dif_spi_device_handle_t *spi, dif_toggle_t enable, uint8_t opcode);
/**
* Set which address bits are swapped and their values for commands that have
* the address swap enabled.
*
* @param spi A handle to a spi device.
* @param mask A bitmask indicating which address bits should be replaced.
* @param replacement The values to swap in for the masked address bits.
* @return `kDifBadArg` if spi is NULL, else `kDifOk`.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_flash_address_swap(dif_spi_device_handle_t *spi,
uint32_t mask,
uint32_t replacement);
/**
* Set which bits are swapped and their values for commands that have
* the first-word payload swap function enabled.
*
* @param spi A handle to a spi device.
* @param mask A bitmask indicating which bits should be replaced.
* @param replacement The values to swap in for the masked bits.
* @return `kDifBadArg` if spi is NULL, else `kDifOk`.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_flash_payload_swap(dif_spi_device_handle_t *spi,
uint32_t mask,
uint32_t replacement);
/**
* Get the current occupancy level of the command FIFO. Used in flash and
* passthrough modes.
*
* @param spi A SPI handle.
* @param[out] occupancy The number of command entries present in the command
* FIFO.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_command_fifo_occupancy(
dif_spi_device_handle_t *spi, uint8_t *occupancy);
/**
* Get the current occupancy level of the address FIFO. Used in flash and
* passthrough modes.
*
* @param spi A SPI handle.
* @param[out] occupancy The number of address entries present in the command
* FIFO.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_address_fifo_occupancy(
dif_spi_device_handle_t *spi, uint8_t *occupancy);
/**
* Get the current occupancy level of the payload FIFO. Used in flash and
* passthrough modes.
*
* Also get the starting offset for the data that was captured. Only up to 256
* bytes of payload may be captured by the SPI device, and if the SPI host sends
* more data, the SPI device wraps around to the beginning of the payload buffer
* and writes the newest data from there. The starting offset points to the
* oldest item in the payload buffer.
*
* @param spi A SPI handle.
* @param[out] occupancy The number of bytes present in the command FIFO.
* @param[out] start_offset The starting offset in the payload buffer for the
* data, which may wrap around to the beginning.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_payload_fifo_occupancy(
dif_spi_device_handle_t *spi, uint16_t *occupancy, uint32_t *start_offset);
/**
* Pop the first command from the uploaded command FIFO.
*
* @param spi A handle to a spi device.
* @param[out] command A pointer to where the command can be stored.
* @return `kDifBadArg` if any pointers are NULL. `kDifUnavailable` if the FIFO
* was empty. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_pop_flash_command_fifo(dif_spi_device_handle_t *spi,
uint8_t *command);
/**
* Pop the first address from the uploaded address FIFO.
*
* @param spi A handle to a spi device.
* @param[out] address A pointer to where the address can be stored.
* @return `kDifBadArg` if any pointers are NULL. `kDifUnavailable` if the FIFO
* was empty. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_pop_flash_address_fifo(dif_spi_device_handle_t *spi,
uint32_t *address);
/**
* Read data from one of the memories associated with flash / passthrough modes.
*
* @param spi A handle to a spi device.
* @param buffer_type An identifier for which memory space to read from.
* @param offset The starting offset for read data in the memory.
* @param length The length, in bytes, of the data to be copied.
* @param[out] buf A pointer to the location where the data should be stored.
* @return `kDifBadArg` is any pointers are NULL or the `buffer_type` does not
* exist. `kDifOutOfRange` if the requested `offset` and `length` go beyond the
* indicated `buffer_type` region. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_read_flash_buffer(
dif_spi_device_handle_t *spi,
dif_spi_device_flash_buffer_type_t buffer_type, uint32_t offset,
size_t length, uint8_t *buf);
/**
* Write data to one of the memories associated with flash / passthrough modes.
*
* @param spi A handle to a spi device.
* @param buffer_type An identifier for which memory space to write to.
* @param offset The starting offset for the write location of the data.
* @param length The length, in bytes, of the data to be copied.
* @param buf A pointer to the location where the data can be copied from.
* @return `kDifBadArg` is any pointers are NULL or the `buffer_type` does not
* exist. `kDifOutOfRange` if the requested `offset` and `length` go beyond the
* indicated `buffer_type` region. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_write_flash_buffer(
dif_spi_device_handle_t *spi,
dif_spi_device_flash_buffer_type_t buffer_type, uint32_t offset,
size_t length, const uint8_t *buf);
/**
* Get whether the indicated command is filtered for passthrough.
*
* @param spi A handle to a spi device.
* @param command The command to be queried.
* @param[out] enabled A pointer to a location where the filter status can be
* stored. `kDifEnabled` means the command is filtered.
* @return `kDifBadArg` if any pointers are NULL. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_passthrough_command_filter(
dif_spi_device_handle_t *spi, uint8_t command, dif_toggle_t *enabled);
/**
* Set whether the indicated command is filtered for passthrough.
*
* @param spi A handle to a spi device.
* @param command The command to have its filter status updated.
* @param enable Whether to enable the command filter for the `command`.
* @return `kDifBadArg` if `spi` is NULL or `enable` is invalid. `kDifOk`
* otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_passthrough_command_filter(
dif_spi_device_handle_t *spi, uint8_t command, dif_toggle_t enable);
/**
* Set whether ALL commands are filtered for passthrough.
*
* Can be used for a faster initial state before using
* `dif_spi_device_set_passthrough_command_filter` to set the filter status for
* individual commands.
*
* @param spi A handle to a spi device.
* @param enable Whether to enable the command filter for all commands.
* @return `kDifBadArg` if `spi` is NULL or `enable` is invalid. `kDifOk`
* otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_all_passthrough_command_filters(
dif_spi_device_handle_t *spi, dif_toggle_t enable);
/**
* Clear the busy bit and write-enable bit for flash / passthrough modes.
*
* @param spi A handle to a spi device.
* @return `kDifBadArg` if `spi` is NULL. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_clear_flash_busy_bit(dif_spi_device_handle_t *spi);
/**
* Write the status registers for flash / passthrough modes.
*
* Note that the three registers are concatenated to make one 24-bit value, with
* the LSB being the busy bit.
*
* @param spi A handle to a spi device.
* @param value The value to write to the registers.
* @return `kDifBadArg` if `spi` is NULL. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_set_flash_status_registers(
dif_spi_device_handle_t *spi, uint32_t value);
/**
* Get the values of the status registers for flash / passthrough modes.
*
* Note that the three registers are concatenated to make one 24-bit value, with
* the LSB being the busy bit.
*
* @param spi A handle to a spi device.
* @param[out] value A pointer to where to write the values of the registesr.
* @return `kDifBadArg` if any pointer arguments are NULL. `kDifOk` otherwise.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_flash_status_registers(
dif_spi_device_handle_t *spi, uint32_t *value);
typedef struct dif_spi_device_tpm_caps {
/** Hardware revision of the TPM submodule. */
uint8_t revision;
/**
* True if the TPM supports 5 Locality. Otherwise, only 1 Locality is
* supported.
*/
bool multi_locality;
/**
* The maximum write size, in bytes, of the data phase of TPM
* transactions. This represents the maximum amount of data the FIFO can
* accept.
*/
uint8_t max_write_size;
/**
* The maximum read size, in bytes, of the data phase of TPM
* transactions. This represents the maximum amount of data the FIFO can
* accept.
*/
uint8_t max_read_size;
} dif_spi_device_tpm_caps_t;
/**
* Read the capabilities of the TPM hardware IP.
*
* @param spi A handle to a spi device.
* @param[out] caps The TPM's hardware capabilities.
* @return The result of the operation.
*/
OT_WARN_UNUSED_RESULT
dif_result_t dif_spi_device_get_tpm_capabilities(
dif_spi_device_handle_t *spi, dif_spi_device_tpm_caps_t *caps);
typedef enum dif_spi_device_tpm_interface {
/** Specifies the FIFO interface from the TPM specification. */
kDifSpiDeviceTpmInterfaceFifo = 0,
/** Specifies the CRB interface from the TPM specification. */
kDifSpiDeviceTpmInterfaceCrb,
kDifSpiDeviceTpmInterfaceCount,
} dif_spi_device_tpm_interface_t;
typedef struct dif_spi_device_tpm_config {
/**
* The register interface to implement for the TPM. Note that in CRB mode, the
* return-by-hardware logic is disabled.
*/
dif_spi_device_tpm_interface_t interface;
/**
* Upload the commands that would ordinarily be targeted towards registers
* that automatically return by hardware. This includes TPM_DID_VID_x,
* TPM_RID_x, TPM_STS_x, and TPM_ACCESS_x.
*/
bool disable_return_by_hardware;
/**
* Disable address prefix check that matches the pattern 0xD4_XXXX before
* using the return-by-hardware logic. When true, the upper 8 bits of the
* address are not required to be 0xD4 for the return-by-hardware logic to
* activate. When false, commands with non-matching addresses will be
* uploaded.
*/
bool disable_address_prefix_check;
/**
* Disable locality value check to be within the max value before using the
* return-by-hardware logic. When true, the TPM will return 0xFF-valued data
* on reads and discard writes to registers in the TPM region (matching the
* 0xD4_XXXX address pattern) if the locality value is above the maximum. When
* false, commands with invalid locality will be uploaded, and software would
* be responsible for returning 0xFF to read requests.