16
16
#ifdef CONFIG_FLASH_SIMULATOR
17
17
#include <zephyr/drivers/flash/flash_simulator.h>
18
18
#endif /* CONFIG_FLASH_SIMULATOR */
19
+ #include "soc_flash_nrf.h"
19
20
20
21
/* __ALIGNED macro is not defined on NATIVE POSIX. This platform uses __aligned macro. */
21
22
#ifndef __ALIGNED
@@ -118,8 +119,92 @@ struct ipuc_context {
118
119
119
120
#define DEFINE_NRF_IPUC_REF (x , _ ) &__device_flash_nrf_ipuc_##x
120
121
122
+ #define MAX_SINGLE_BYTE_WRITE_TIME_US 4
123
+ #define MAX_SSF_IPUC_WRITE_COMMUNICATION_TIME_US 1200
124
+
121
125
LOG_MODULE_REGISTER (flash_ipuc , CONFIG_FLASH_IPUC_LOG_LEVEL );
122
126
127
+ struct ipuc_write_chunk_data {
128
+ struct zcbor_string * component_id ;
129
+ size_t offset ;
130
+ uintptr_t buffer ;
131
+ size_t chunk_size ;
132
+ bool last_chunk ;
133
+ };
134
+
135
+ static int ipuc_write_op (void * context )
136
+ {
137
+ suit_plat_err_t plat_ret = SUIT_PLAT_SUCCESS ;
138
+ struct ipuc_write_chunk_data * data = (struct ipuc_write_chunk_data * )context ;
139
+
140
+ plat_ret = suit_ipuc_write (data -> component_id , data -> offset , data -> buffer ,
141
+ data -> chunk_size , data -> last_chunk );
142
+ if (plat_ret != SUIT_PLAT_SUCCESS ) {
143
+ LOG_ERR ("Write error: %d" , plat_ret );
144
+ return - EIO ;
145
+ }
146
+
147
+ return FLASH_OP_DONE ;
148
+ }
149
+
150
+ /* TODO: do not synchronyze if only app core */
151
+
152
+ static int ipuc_synchronous_write (struct zcbor_string * component_id , size_t offset ,
153
+ uintptr_t buffer , size_t chunk_size , bool last_chunk )
154
+ {
155
+ struct ipuc_write_chunk_data data = {
156
+ .component_id = component_id ,
157
+ .offset = offset ,
158
+ .buffer = buffer ,
159
+ .chunk_size = chunk_size ,
160
+ .last_chunk = last_chunk ,
161
+ };
162
+
163
+ struct flash_op_desc flash_op_desc = {
164
+ .handler = ipuc_write_op ,
165
+ /* Currently data is passed as context despite of the type mismatch,
166
+ as nrf_flash_sync_exe simply passes the context without any operations.
167
+
168
+ However, a better approach might be creating a global context variable for
169
+ the flash_ipuc module which would store all of the data.
170
+ This would not only remove the issue with the type mismatch, it would
171
+ also allow the driver to (for example) handle different addresses
172
+ in a different way, for example not synchronyze if the NVM memory is
173
+ not in the same bank as the memory used by the radio protocols. */
174
+ .context = (void * ) & data
175
+ };
176
+
177
+ /* We are only accounting for SSF communication time when sending the request,
178
+ the response time does not matter, as the flash write has already finished
179
+ by then. */
180
+ uint32_t requested_timeslot_duration = MAX_SINGLE_BYTE_WRITE_TIME_US * chunk_size +
181
+ MAX_SSF_IPUC_WRITE_COMMUNICATION_TIME_US ;
182
+
183
+
184
+ nrf_flash_sync_set_context (requested_timeslot_duration );
185
+ return nrf_flash_sync_exe (& flash_op_desc );
186
+ }
187
+
188
+ static int ipuc_write (struct zcbor_string * component_id , size_t offset ,
189
+ uintptr_t buffer , size_t chunk_size , bool last_chunk )
190
+ {
191
+ suit_plat_err_t plat_ret = SUIT_PLAT_SUCCESS ;
192
+
193
+ #if defined(CONFIG_SOC_FLASH_NRF_RADIO_SYNC_RPC_HOST )
194
+ if (nrf_flash_sync_is_required ()) {
195
+ return ipuc_synchronous_write (component_id , offset , buffer , chunk_size , last_chunk );
196
+ }
197
+ #endif
198
+ plat_ret = suit_ipuc_write (component_id , offset , buffer , chunk_size , last_chunk );
199
+ if (plat_ret != SUIT_PLAT_SUCCESS ) {
200
+ LOG_ERR ("Data write (%p, 0x%x) error: %d" , (void * ) buffer , chunk_size , plat_ret );
201
+ return - EIO ;
202
+ }
203
+
204
+ return 0 ;
205
+ }
206
+
207
+
123
208
static int nrf_ipuc_read (const struct device * dev , off_t offset , void * data , size_t len )
124
209
{
125
210
struct ipuc_context * ctx = NULL ;
@@ -219,12 +304,12 @@ static int nrf_ipuc_write(const struct device *dev, off_t offset, const void *da
219
304
return - EIO ;
220
305
}
221
306
222
- plat_ret = suit_ipuc_write (& ctx -> component_id , offset ,
223
- (uintptr_t )unaligned_data_buf , unaligned_len , false);
224
- if (plat_ret != SUIT_PLAT_SUCCESS ) {
307
+ ret = ipuc_write (& ctx -> component_id , offset ,
308
+ (uintptr_t )unaligned_data_buf , unaligned_len , false);
309
+ if (ret != 0 ) {
225
310
LOG_ERR ("Unaligned data write (%p, 0x%x) error: %d" ,
226
- (void * )unaligned_data_buf , unaligned_len , plat_ret );
227
- return - EIO ;
311
+ (void * )unaligned_data_buf , unaligned_len , ret );
312
+ return ret ;
228
313
}
229
314
230
315
offset += unaligned_len ;
@@ -244,10 +329,10 @@ static int nrf_ipuc_write(const struct device *dev, off_t offset, const void *da
244
329
return - EIO ;
245
330
}
246
331
247
- plat_ret = suit_ipuc_write (& ctx -> component_id , offset , (uintptr_t )data , len , false);
248
- if (plat_ret != SUIT_PLAT_SUCCESS ) {
249
- LOG_ERR ("Aligned data write (%p, 0x%x) error: %d" , data , len , plat_ret );
250
- return - EIO ;
332
+ ret = ipuc_write (& ctx -> component_id , offset , (uintptr_t )data , len , false);
333
+ if (ret != 0 ) {
334
+ LOG_ERR ("Aligned data write (%p, 0x%x) error: %d" , data , len , ret );
335
+ return ret ;
251
336
}
252
337
253
338
return 0 ;
@@ -308,11 +393,11 @@ static int nrf_ipuc_erase(const struct device *dev, off_t offset, size_t size)
308
393
309
394
while (size > 0 ) {
310
395
bytes_erased = MIN (sizeof (erase_block ), size );
311
- plat_ret = suit_ipuc_write (& ctx -> component_id , offset , (uintptr_t )erase_block ,
312
- bytes_erased , false);
313
- if (plat_ret != SUIT_PLAT_SUCCESS ) {
396
+ ret = ipuc_write (& ctx -> component_id , offset , (uintptr_t )erase_block ,
397
+ bytes_erased , false);
398
+ if (ret != 0 ) {
314
399
LOG_ERR ("Failed to erase IPUC (%p:%zu): %d" ,
315
- (void * )(ctx -> address + offset ), bytes_erased , plat_ret );
400
+ (void * )(ctx -> address + offset ), bytes_erased , ret );
316
401
return - EIO ;
317
402
}
318
403
@@ -560,6 +645,9 @@ static struct device *flash_cache_ipuc(uintptr_t min_address, uintptr_t *ipuc_ad
560
645
(uintptr_t )ipuc_size );
561
646
return NULL ;
562
647
}
648
+ #if defined(CONFIG_SOC_FLASH_NRF_RADIO_SYNC_RPC_HOST )
649
+ nrf_flash_sync_init ();
650
+ #endif
563
651
564
652
dev = get_free_dev ();
565
653
if (dev == NULL ) {
0 commit comments