Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 7ec32c2

Browse files
committedMar 28, 2025·
drivers: flash_ipuc: Integrate with flash synchronization
This commit integrates the Flash IPUC driver with the flash_sync_rpc driver. Signed-off-by: Artur Hadasz <artur.hadasz@nordicsemi.no>
1 parent ac41a39 commit 7ec32c2

File tree

1 file changed

+101
-13
lines changed

1 file changed

+101
-13
lines changed
 

‎drivers/flash/flash_ipuc/flash_ipuc.c

+101-13
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
#ifdef CONFIG_FLASH_SIMULATOR
1717
#include <zephyr/drivers/flash/flash_simulator.h>
1818
#endif /* CONFIG_FLASH_SIMULATOR */
19+
#include "soc_flash_nrf.h"
1920

2021
/* __ALIGNED macro is not defined on NATIVE POSIX. This platform uses __aligned macro. */
2122
#ifndef __ALIGNED
@@ -118,8 +119,92 @@ struct ipuc_context {
118119

119120
#define DEFINE_NRF_IPUC_REF(x, _) &__device_flash_nrf_ipuc_##x
120121

122+
#define MAX_SINGLE_BYTE_WRITE_TIME_US 4
123+
#define MAX_SSF_IPUC_WRITE_COMMUNICATION_TIME_US 1200
124+
121125
LOG_MODULE_REGISTER(flash_ipuc, CONFIG_FLASH_IPUC_LOG_LEVEL);
122126

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+
123208
static int nrf_ipuc_read(const struct device *dev, off_t offset, void *data, size_t len)
124209
{
125210
struct ipuc_context *ctx = NULL;
@@ -219,12 +304,12 @@ static int nrf_ipuc_write(const struct device *dev, off_t offset, const void *da
219304
return -EIO;
220305
}
221306

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) {
225310
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;
228313
}
229314

230315
offset += unaligned_len;
@@ -244,10 +329,10 @@ static int nrf_ipuc_write(const struct device *dev, off_t offset, const void *da
244329
return -EIO;
245330
}
246331

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;
251336
}
252337

253338
return 0;
@@ -308,11 +393,11 @@ static int nrf_ipuc_erase(const struct device *dev, off_t offset, size_t size)
308393

309394
while (size > 0) {
310395
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) {
314399
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);
316401
return -EIO;
317402
}
318403

@@ -560,6 +645,9 @@ static struct device *flash_cache_ipuc(uintptr_t min_address, uintptr_t *ipuc_ad
560645
(uintptr_t)ipuc_size);
561646
return NULL;
562647
}
648+
#if defined(CONFIG_SOC_FLASH_NRF_RADIO_SYNC_RPC_HOST)
649+
nrf_flash_sync_init();
650+
#endif
563651

564652
dev = get_free_dev();
565653
if (dev == NULL) {

0 commit comments

Comments
 (0)
Please sign in to comment.