19da7a653SHaojian Zhuang /*
2*06d223cbSYann Gautier * Copyright (c) 2016-2023, ARM Limited and Contributors. All rights reserved.
39da7a653SHaojian Zhuang *
482cb2c1aSdp-arm * SPDX-License-Identifier: BSD-3-Clause
59da7a653SHaojian Zhuang */
69da7a653SHaojian Zhuang
79da7a653SHaojian Zhuang #include <assert.h>
89da7a653SHaojian Zhuang #include <errno.h>
99da7a653SHaojian Zhuang #include <string.h>
1009d40e0eSAntonio Nino Diaz
1109d40e0eSAntonio Nino Diaz #include <platform_def.h>
1209d40e0eSAntonio Nino Diaz
1309d40e0eSAntonio Nino Diaz #include <common/debug.h>
1409d40e0eSAntonio Nino Diaz #include <drivers/io/io_block.h>
1509d40e0eSAntonio Nino Diaz #include <drivers/io/io_driver.h>
1609d40e0eSAntonio Nino Diaz #include <drivers/io/io_storage.h>
1709d40e0eSAntonio Nino Diaz #include <lib/utils.h>
189da7a653SHaojian Zhuang
199da7a653SHaojian Zhuang typedef struct {
209da7a653SHaojian Zhuang io_block_dev_spec_t *dev_spec;
219da7a653SHaojian Zhuang uintptr_t base;
2270cb0bffSYann Gautier unsigned long long file_pos;
2370cb0bffSYann Gautier unsigned long long size;
249da7a653SHaojian Zhuang } block_dev_state_t;
259da7a653SHaojian Zhuang
2670cb0bffSYann Gautier #define is_power_of_2(x) (((x) != 0U) && (((x) & ((x) - 1U)) == 0U))
279da7a653SHaojian Zhuang
289da7a653SHaojian Zhuang io_type_t device_type_block(void);
299da7a653SHaojian Zhuang
309da7a653SHaojian Zhuang static int block_open(io_dev_info_t *dev_info, const uintptr_t spec,
319da7a653SHaojian Zhuang io_entity_t *entity);
3270cb0bffSYann Gautier static int block_seek(io_entity_t *entity, int mode, signed long long offset);
339da7a653SHaojian Zhuang static int block_read(io_entity_t *entity, uintptr_t buffer, size_t length,
349da7a653SHaojian Zhuang size_t *length_read);
359da7a653SHaojian Zhuang static int block_write(io_entity_t *entity, const uintptr_t buffer,
369da7a653SHaojian Zhuang size_t length, size_t *length_written);
379da7a653SHaojian Zhuang static int block_close(io_entity_t *entity);
389da7a653SHaojian Zhuang static int block_dev_open(const uintptr_t dev_spec, io_dev_info_t **dev_info);
399da7a653SHaojian Zhuang static int block_dev_close(io_dev_info_t *dev_info);
409da7a653SHaojian Zhuang
419da7a653SHaojian Zhuang static const io_dev_connector_t block_dev_connector = {
429da7a653SHaojian Zhuang .dev_open = block_dev_open
439da7a653SHaojian Zhuang };
449da7a653SHaojian Zhuang
459da7a653SHaojian Zhuang static const io_dev_funcs_t block_dev_funcs = {
469da7a653SHaojian Zhuang .type = device_type_block,
479da7a653SHaojian Zhuang .open = block_open,
489da7a653SHaojian Zhuang .seek = block_seek,
499da7a653SHaojian Zhuang .size = NULL,
509da7a653SHaojian Zhuang .read = block_read,
519da7a653SHaojian Zhuang .write = block_write,
529da7a653SHaojian Zhuang .close = block_close,
539da7a653SHaojian Zhuang .dev_init = NULL,
549da7a653SHaojian Zhuang .dev_close = block_dev_close,
559da7a653SHaojian Zhuang };
569da7a653SHaojian Zhuang
579da7a653SHaojian Zhuang static block_dev_state_t state_pool[MAX_IO_BLOCK_DEVICES];
589da7a653SHaojian Zhuang static io_dev_info_t dev_info_pool[MAX_IO_BLOCK_DEVICES];
599da7a653SHaojian Zhuang
609da7a653SHaojian Zhuang /* Track number of allocated block state */
619da7a653SHaojian Zhuang static unsigned int block_dev_count;
629da7a653SHaojian Zhuang
device_type_block(void)639da7a653SHaojian Zhuang io_type_t device_type_block(void)
649da7a653SHaojian Zhuang {
659da7a653SHaojian Zhuang return IO_TYPE_BLOCK;
669da7a653SHaojian Zhuang }
679da7a653SHaojian Zhuang
689da7a653SHaojian Zhuang /* Locate a block state in the pool, specified by address */
find_first_block_state(const io_block_dev_spec_t * dev_spec,unsigned int * index_out)699da7a653SHaojian Zhuang static int find_first_block_state(const io_block_dev_spec_t *dev_spec,
709da7a653SHaojian Zhuang unsigned int *index_out)
719da7a653SHaojian Zhuang {
72b7c6529cSYann Gautier unsigned int index;
739da7a653SHaojian Zhuang int result = -ENOENT;
74b7c6529cSYann Gautier
75b7c6529cSYann Gautier for (index = 0U; index < MAX_IO_BLOCK_DEVICES; ++index) {
769da7a653SHaojian Zhuang /* dev_spec is used as identifier since it's unique */
779da7a653SHaojian Zhuang if (state_pool[index].dev_spec == dev_spec) {
789da7a653SHaojian Zhuang result = 0;
799da7a653SHaojian Zhuang *index_out = index;
809da7a653SHaojian Zhuang break;
819da7a653SHaojian Zhuang }
829da7a653SHaojian Zhuang }
839da7a653SHaojian Zhuang return result;
849da7a653SHaojian Zhuang }
859da7a653SHaojian Zhuang
869da7a653SHaojian Zhuang /* Allocate a device info from the pool and return a pointer to it */
allocate_dev_info(io_dev_info_t ** dev_info)879da7a653SHaojian Zhuang static int allocate_dev_info(io_dev_info_t **dev_info)
889da7a653SHaojian Zhuang {
899da7a653SHaojian Zhuang int result = -ENOMEM;
909da7a653SHaojian Zhuang assert(dev_info != NULL);
919da7a653SHaojian Zhuang
929da7a653SHaojian Zhuang if (block_dev_count < MAX_IO_BLOCK_DEVICES) {
939da7a653SHaojian Zhuang unsigned int index = 0;
949da7a653SHaojian Zhuang result = find_first_block_state(NULL, &index);
959da7a653SHaojian Zhuang assert(result == 0);
969da7a653SHaojian Zhuang /* initialize dev_info */
979da7a653SHaojian Zhuang dev_info_pool[index].funcs = &block_dev_funcs;
989da7a653SHaojian Zhuang dev_info_pool[index].info = (uintptr_t)&state_pool[index];
999da7a653SHaojian Zhuang *dev_info = &dev_info_pool[index];
1009da7a653SHaojian Zhuang ++block_dev_count;
1019da7a653SHaojian Zhuang }
1029da7a653SHaojian Zhuang
1039da7a653SHaojian Zhuang return result;
1049da7a653SHaojian Zhuang }
1059da7a653SHaojian Zhuang
1069da7a653SHaojian Zhuang
1079da7a653SHaojian Zhuang /* Release a device info to the pool */
free_dev_info(io_dev_info_t * dev_info)1089da7a653SHaojian Zhuang static int free_dev_info(io_dev_info_t *dev_info)
1099da7a653SHaojian Zhuang {
1109da7a653SHaojian Zhuang int result;
1119da7a653SHaojian Zhuang unsigned int index = 0;
1129da7a653SHaojian Zhuang block_dev_state_t *state;
1139da7a653SHaojian Zhuang assert(dev_info != NULL);
1149da7a653SHaojian Zhuang
1159da7a653SHaojian Zhuang state = (block_dev_state_t *)dev_info->info;
1169da7a653SHaojian Zhuang result = find_first_block_state(state->dev_spec, &index);
1179da7a653SHaojian Zhuang if (result == 0) {
1189da7a653SHaojian Zhuang /* free if device info is valid */
11932f0d3c6SDouglas Raillard zeromem(state, sizeof(block_dev_state_t));
12032f0d3c6SDouglas Raillard zeromem(dev_info, sizeof(io_dev_info_t));
1219da7a653SHaojian Zhuang --block_dev_count;
1229da7a653SHaojian Zhuang }
1239da7a653SHaojian Zhuang
1249da7a653SHaojian Zhuang return result;
1259da7a653SHaojian Zhuang }
1269da7a653SHaojian Zhuang
block_open(io_dev_info_t * dev_info,const uintptr_t spec,io_entity_t * entity)1279da7a653SHaojian Zhuang static int block_open(io_dev_info_t *dev_info, const uintptr_t spec,
1289da7a653SHaojian Zhuang io_entity_t *entity)
1299da7a653SHaojian Zhuang {
1309da7a653SHaojian Zhuang block_dev_state_t *cur;
1319da7a653SHaojian Zhuang io_block_spec_t *region;
1329da7a653SHaojian Zhuang
1339da7a653SHaojian Zhuang assert((dev_info->info != (uintptr_t)NULL) &&
1349da7a653SHaojian Zhuang (spec != (uintptr_t)NULL) &&
1359da7a653SHaojian Zhuang (entity->info == (uintptr_t)NULL));
1369da7a653SHaojian Zhuang
1379da7a653SHaojian Zhuang region = (io_block_spec_t *)spec;
1389da7a653SHaojian Zhuang cur = (block_dev_state_t *)dev_info->info;
1399da7a653SHaojian Zhuang assert(((region->offset % cur->dev_spec->block_size) == 0) &&
1409da7a653SHaojian Zhuang ((region->length % cur->dev_spec->block_size) == 0));
1419da7a653SHaojian Zhuang
1429da7a653SHaojian Zhuang cur->base = region->offset;
1439da7a653SHaojian Zhuang cur->size = region->length;
1449da7a653SHaojian Zhuang cur->file_pos = 0;
1459da7a653SHaojian Zhuang
1469da7a653SHaojian Zhuang entity->info = (uintptr_t)cur;
1479da7a653SHaojian Zhuang return 0;
1489da7a653SHaojian Zhuang }
1499da7a653SHaojian Zhuang
1509da7a653SHaojian Zhuang /* parameter offset is relative address at here */
block_seek(io_entity_t * entity,int mode,signed long long offset)15170cb0bffSYann Gautier static int block_seek(io_entity_t *entity, int mode, signed long long offset)
1529da7a653SHaojian Zhuang {
1539da7a653SHaojian Zhuang block_dev_state_t *cur;
1549da7a653SHaojian Zhuang
1559da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL);
1569da7a653SHaojian Zhuang
1579da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info;
15870cb0bffSYann Gautier assert((offset >= 0) && ((unsigned long long)offset < cur->size));
1599da7a653SHaojian Zhuang
1609da7a653SHaojian Zhuang switch (mode) {
1619da7a653SHaojian Zhuang case IO_SEEK_SET:
16270cb0bffSYann Gautier cur->file_pos = (unsigned long long)offset;
1639da7a653SHaojian Zhuang break;
1649da7a653SHaojian Zhuang case IO_SEEK_CUR:
16570cb0bffSYann Gautier cur->file_pos += (unsigned long long)offset;
1669da7a653SHaojian Zhuang break;
1679da7a653SHaojian Zhuang default:
1689da7a653SHaojian Zhuang return -EINVAL;
1699da7a653SHaojian Zhuang }
1709da7a653SHaojian Zhuang assert(cur->file_pos < cur->size);
1719da7a653SHaojian Zhuang return 0;
1729da7a653SHaojian Zhuang }
1739da7a653SHaojian Zhuang
174e19e40afSRoberto Vargas /*
175e19e40afSRoberto Vargas * This function allows the caller to read any number of bytes
176e19e40afSRoberto Vargas * from any position. It hides from the caller that the low level
177e19e40afSRoberto Vargas * driver only can read aligned blocks of data. For this reason
178e19e40afSRoberto Vargas * we need to handle the use case where the first byte to be read is not
179e19e40afSRoberto Vargas * aligned to start of the block, the last byte to be read is also not
180e19e40afSRoberto Vargas * aligned to the end of a block, and there are zero or more blocks-worth
181e19e40afSRoberto Vargas * of data in between.
182e19e40afSRoberto Vargas *
183e19e40afSRoberto Vargas * In such a case we need to read more bytes than requested (i.e. full
184e19e40afSRoberto Vargas * blocks) and strip-out the leading bytes (aka skip) and the trailing
185e19e40afSRoberto Vargas * bytes (aka padding). See diagram below
186e19e40afSRoberto Vargas *
187e19e40afSRoberto Vargas * cur->file_pos ------------
188e19e40afSRoberto Vargas * |
189e19e40afSRoberto Vargas * cur->base |
190e19e40afSRoberto Vargas * | |
191e19e40afSRoberto Vargas * v v<---- length ---->
192e19e40afSRoberto Vargas * --------------------------------------------------------------
193e19e40afSRoberto Vargas * | | block#1 | | block#n |
194e19e40afSRoberto Vargas * | block#0 | + | ... | + |
195e19e40afSRoberto Vargas * | | <- skip -> + | | + <- padding ->|
196e19e40afSRoberto Vargas * ------------------------+----------------------+--------------
197e19e40afSRoberto Vargas * ^ ^
198e19e40afSRoberto Vargas * | |
199e19e40afSRoberto Vargas * v iteration#1 iteration#n v
200e19e40afSRoberto Vargas * --------------------------------------------------
201e19e40afSRoberto Vargas * | | | |
202e19e40afSRoberto Vargas * |<---- request ---->| ... |<----- request ---->|
203e19e40afSRoberto Vargas * | | | |
204e19e40afSRoberto Vargas * --------------------------------------------------
205e19e40afSRoberto Vargas * / / | |
206e19e40afSRoberto Vargas * / / | |
207e19e40afSRoberto Vargas * / / | |
208e19e40afSRoberto Vargas * / / | |
209e19e40afSRoberto Vargas * / / | |
210e19e40afSRoberto Vargas * / / | |
211e19e40afSRoberto Vargas * / / | |
212e19e40afSRoberto Vargas * / / | |
213e19e40afSRoberto Vargas * / / | |
214e19e40afSRoberto Vargas * / / | |
215e19e40afSRoberto Vargas * <---- request ------> <------ request ----->
216e19e40afSRoberto Vargas * --------------------- -----------------------
217e19e40afSRoberto Vargas * | | | | | |
218e19e40afSRoberto Vargas * |<-skip->|<-nbytes->| -------->|<-nbytes->|<-padding->|
219e19e40afSRoberto Vargas * | | | | | | |
220e19e40afSRoberto Vargas * --------------------- | -----------------------
221e19e40afSRoberto Vargas * ^ \ \ | | |
222e19e40afSRoberto Vargas * | \ \ | | |
223e19e40afSRoberto Vargas * | \ \ | | |
224e19e40afSRoberto Vargas * buf->offset \ \ buf->offset | |
225e19e40afSRoberto Vargas * \ \ | |
226e19e40afSRoberto Vargas * \ \ | |
227e19e40afSRoberto Vargas * \ \ | |
228e19e40afSRoberto Vargas * \ \ | |
229e19e40afSRoberto Vargas * \ \ | |
230e19e40afSRoberto Vargas * \ \ | |
231e19e40afSRoberto Vargas * \ \ | |
232e19e40afSRoberto Vargas * --------------------------------
233e19e40afSRoberto Vargas * | | | |
234e19e40afSRoberto Vargas * buffer-------------->| | ... | |
235e19e40afSRoberto Vargas * | | | |
236e19e40afSRoberto Vargas * --------------------------------
237e19e40afSRoberto Vargas * <-count#1->| |
238e19e40afSRoberto Vargas * <---------- count#n -------->
239e19e40afSRoberto Vargas * <---------- length ---------->
240e19e40afSRoberto Vargas *
241e19e40afSRoberto Vargas * Additionally, the IO driver has an underlying buffer that is at least
242e19e40afSRoberto Vargas * one block-size and may be big enough to allow.
243e19e40afSRoberto Vargas */
block_read(io_entity_t * entity,uintptr_t buffer,size_t length,size_t * length_read)2449da7a653SHaojian Zhuang static int block_read(io_entity_t *entity, uintptr_t buffer, size_t length,
2459da7a653SHaojian Zhuang size_t *length_read)
2469da7a653SHaojian Zhuang {
2479da7a653SHaojian Zhuang block_dev_state_t *cur;
2489da7a653SHaojian Zhuang io_block_spec_t *buf;
2499da7a653SHaojian Zhuang io_block_ops_t *ops;
2509da7a653SHaojian Zhuang int lba;
251e19e40afSRoberto Vargas size_t block_size, left;
252e19e40afSRoberto Vargas size_t nbytes; /* number of bytes read in one iteration */
253e19e40afSRoberto Vargas size_t request; /* number of requested bytes in one iteration */
254e19e40afSRoberto Vargas size_t count; /* number of bytes already read */
255e19e40afSRoberto Vargas /*
256e19e40afSRoberto Vargas * number of leading bytes from start of the block
257e19e40afSRoberto Vargas * to the first byte to be read
258e19e40afSRoberto Vargas */
259e19e40afSRoberto Vargas size_t skip;
260e19e40afSRoberto Vargas
261e19e40afSRoberto Vargas /*
262e19e40afSRoberto Vargas * number of trailing bytes between the last byte
263e19e40afSRoberto Vargas * to be read and the end of the block
264e19e40afSRoberto Vargas */
265e19e40afSRoberto Vargas size_t padding;
2669da7a653SHaojian Zhuang
2679da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL);
2689da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info;
2699da7a653SHaojian Zhuang ops = &(cur->dev_spec->ops);
2709da7a653SHaojian Zhuang buf = &(cur->dev_spec->buffer);
2719da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size;
2729da7a653SHaojian Zhuang assert((length <= cur->size) &&
27370cb0bffSYann Gautier (length > 0U) &&
274*06d223cbSYann Gautier (ops->read != NULL));
2759da7a653SHaojian Zhuang
2769d063aa2SHaojian Zhuang /*
277e19e40afSRoberto Vargas * We don't know the number of bytes that we are going
278e19e40afSRoberto Vargas * to read in every iteration, because it will depend
279e19e40afSRoberto Vargas * on the low level driver.
2809d063aa2SHaojian Zhuang */
281e19e40afSRoberto Vargas count = 0;
28270cb0bffSYann Gautier for (left = length; left > 0U; left -= nbytes) {
283e19e40afSRoberto Vargas /*
284e19e40afSRoberto Vargas * We must only request operations aligned to the block
285e19e40afSRoberto Vargas * size. Therefore if file_pos is not block-aligned,
286e19e40afSRoberto Vargas * we have to request the operation to start at the
287e19e40afSRoberto Vargas * previous block boundary and skip the leading bytes. And
288e19e40afSRoberto Vargas * similarly, the number of bytes requested must be a
289e19e40afSRoberto Vargas * block size multiple
290e19e40afSRoberto Vargas */
29170cb0bffSYann Gautier skip = cur->file_pos & (block_size - 1U);
292e19e40afSRoberto Vargas
293e19e40afSRoberto Vargas /*
294e19e40afSRoberto Vargas * Calculate the block number containing file_pos
295e19e40afSRoberto Vargas * - e.g. block 3.
296e19e40afSRoberto Vargas */
297e19e40afSRoberto Vargas lba = (cur->file_pos + cur->base) / block_size;
298e19e40afSRoberto Vargas
29970cb0bffSYann Gautier if ((skip + left) > buf->length) {
300e19e40afSRoberto Vargas /*
301e19e40afSRoberto Vargas * The underlying read buffer is too small to
302e19e40afSRoberto Vargas * read all the required data - limit to just
303e19e40afSRoberto Vargas * fill the buffer, and then read again.
304e19e40afSRoberto Vargas */
305e19e40afSRoberto Vargas request = buf->length;
3069d063aa2SHaojian Zhuang } else {
307e19e40afSRoberto Vargas /*
308e19e40afSRoberto Vargas * The underlying read buffer is big enough to
309e19e40afSRoberto Vargas * read all the required data. Calculate the
310e19e40afSRoberto Vargas * number of bytes to read to align with the
311e19e40afSRoberto Vargas * block size.
312e19e40afSRoberto Vargas */
313e19e40afSRoberto Vargas request = skip + left;
31470cb0bffSYann Gautier request = (request + (block_size - 1U)) &
31570cb0bffSYann Gautier ~(block_size - 1U);
316e19e40afSRoberto Vargas }
317e19e40afSRoberto Vargas request = ops->read(lba, buf->offset, request);
318e19e40afSRoberto Vargas
319e19e40afSRoberto Vargas if (request <= skip) {
320e19e40afSRoberto Vargas /*
321e19e40afSRoberto Vargas * We couldn't read enough bytes to jump over
322e19e40afSRoberto Vargas * the skip bytes, so we should have to read
323e19e40afSRoberto Vargas * again the same block, thus generating
324e19e40afSRoberto Vargas * the same error.
325e19e40afSRoberto Vargas */
326e19e40afSRoberto Vargas return -EIO;
3279d063aa2SHaojian Zhuang }
3289d063aa2SHaojian Zhuang
3299d063aa2SHaojian Zhuang /*
330e19e40afSRoberto Vargas * Need to remove skip and padding bytes,if any, from
331e19e40afSRoberto Vargas * the read data when copying to the user buffer.
3329d063aa2SHaojian Zhuang */
333e19e40afSRoberto Vargas nbytes = request - skip;
33470cb0bffSYann Gautier padding = (nbytes > left) ? nbytes - left : 0U;
335e19e40afSRoberto Vargas nbytes -= padding;
336e19e40afSRoberto Vargas
337e19e40afSRoberto Vargas memcpy((void *)(buffer + count),
3389da7a653SHaojian Zhuang (void *)(buf->offset + skip),
339e19e40afSRoberto Vargas nbytes);
340e19e40afSRoberto Vargas
341e19e40afSRoberto Vargas cur->file_pos += nbytes;
342e19e40afSRoberto Vargas count += nbytes;
3439da7a653SHaojian Zhuang }
344e19e40afSRoberto Vargas assert(count == length);
345e19e40afSRoberto Vargas *length_read = count;
3469da7a653SHaojian Zhuang
3479da7a653SHaojian Zhuang return 0;
3489da7a653SHaojian Zhuang }
3499da7a653SHaojian Zhuang
350e19e40afSRoberto Vargas /*
351e19e40afSRoberto Vargas * This function allows the caller to write any number of bytes
352e19e40afSRoberto Vargas * from any position. It hides from the caller that the low level
353e19e40afSRoberto Vargas * driver only can write aligned blocks of data.
354e19e40afSRoberto Vargas * See comments for block_read for more details.
355e19e40afSRoberto Vargas */
block_write(io_entity_t * entity,const uintptr_t buffer,size_t length,size_t * length_written)3569da7a653SHaojian Zhuang static int block_write(io_entity_t *entity, const uintptr_t buffer,
3579da7a653SHaojian Zhuang size_t length, size_t *length_written)
3589da7a653SHaojian Zhuang {
3599da7a653SHaojian Zhuang block_dev_state_t *cur;
3609da7a653SHaojian Zhuang io_block_spec_t *buf;
3619da7a653SHaojian Zhuang io_block_ops_t *ops;
3629da7a653SHaojian Zhuang int lba;
363e19e40afSRoberto Vargas size_t block_size, left;
364e19e40afSRoberto Vargas size_t nbytes; /* number of bytes read in one iteration */
365e19e40afSRoberto Vargas size_t request; /* number of requested bytes in one iteration */
366e19e40afSRoberto Vargas size_t count; /* number of bytes already read */
367e19e40afSRoberto Vargas /*
368e19e40afSRoberto Vargas * number of leading bytes from start of the block
369e19e40afSRoberto Vargas * to the first byte to be read
370e19e40afSRoberto Vargas */
371e19e40afSRoberto Vargas size_t skip;
372e19e40afSRoberto Vargas
373e19e40afSRoberto Vargas /*
374e19e40afSRoberto Vargas * number of trailing bytes between the last byte
375e19e40afSRoberto Vargas * to be read and the end of the block
376e19e40afSRoberto Vargas */
377e19e40afSRoberto Vargas size_t padding;
3789da7a653SHaojian Zhuang
3799da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL);
3809da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info;
3819da7a653SHaojian Zhuang ops = &(cur->dev_spec->ops);
3829da7a653SHaojian Zhuang buf = &(cur->dev_spec->buffer);
3839da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size;
3849da7a653SHaojian Zhuang assert((length <= cur->size) &&
38570cb0bffSYann Gautier (length > 0U) &&
386*06d223cbSYann Gautier (ops->read != NULL) &&
387*06d223cbSYann Gautier (ops->write != NULL));
3889da7a653SHaojian Zhuang
3899d063aa2SHaojian Zhuang /*
390e19e40afSRoberto Vargas * We don't know the number of bytes that we are going
391e19e40afSRoberto Vargas * to write in every iteration, because it will depend
392e19e40afSRoberto Vargas * on the low level driver.
3939d063aa2SHaojian Zhuang */
394e19e40afSRoberto Vargas count = 0;
39570cb0bffSYann Gautier for (left = length; left > 0U; left -= nbytes) {
396e19e40afSRoberto Vargas /*
397e19e40afSRoberto Vargas * We must only request operations aligned to the block
398e19e40afSRoberto Vargas * size. Therefore if file_pos is not block-aligned,
399e19e40afSRoberto Vargas * we have to request the operation to start at the
400e19e40afSRoberto Vargas * previous block boundary and skip the leading bytes. And
401e19e40afSRoberto Vargas * similarly, the number of bytes requested must be a
402e19e40afSRoberto Vargas * block size multiple
403e19e40afSRoberto Vargas */
40470cb0bffSYann Gautier skip = cur->file_pos & (block_size - 1U);
405e19e40afSRoberto Vargas
406e19e40afSRoberto Vargas /*
407e19e40afSRoberto Vargas * Calculate the block number containing file_pos
408e19e40afSRoberto Vargas * - e.g. block 3.
409e19e40afSRoberto Vargas */
410e19e40afSRoberto Vargas lba = (cur->file_pos + cur->base) / block_size;
411e19e40afSRoberto Vargas
41270cb0bffSYann Gautier if ((skip + left) > buf->length) {
413e19e40afSRoberto Vargas /*
414e19e40afSRoberto Vargas * The underlying read buffer is too small to
415e19e40afSRoberto Vargas * read all the required data - limit to just
416e19e40afSRoberto Vargas * fill the buffer, and then read again.
417e19e40afSRoberto Vargas */
418e19e40afSRoberto Vargas request = buf->length;
4199d063aa2SHaojian Zhuang } else {
420e19e40afSRoberto Vargas /*
421e19e40afSRoberto Vargas * The underlying read buffer is big enough to
422e19e40afSRoberto Vargas * read all the required data. Calculate the
423e19e40afSRoberto Vargas * number of bytes to read to align with the
424e19e40afSRoberto Vargas * block size.
425e19e40afSRoberto Vargas */
426e19e40afSRoberto Vargas request = skip + left;
42770cb0bffSYann Gautier request = (request + (block_size - 1U)) &
42870cb0bffSYann Gautier ~(block_size - 1U);
4299d063aa2SHaojian Zhuang }
4309d063aa2SHaojian Zhuang
4319da7a653SHaojian Zhuang /*
432e19e40afSRoberto Vargas * The number of bytes that we are going to write
433e19e40afSRoberto Vargas * from the user buffer will depend of the size
434e19e40afSRoberto Vargas * of the current request.
4359da7a653SHaojian Zhuang */
436e19e40afSRoberto Vargas nbytes = request - skip;
43770cb0bffSYann Gautier padding = (nbytes > left) ? nbytes - left : 0U;
438e19e40afSRoberto Vargas nbytes -= padding;
439e19e40afSRoberto Vargas
4409da7a653SHaojian Zhuang /*
441e19e40afSRoberto Vargas * If we have skip or padding bytes then we have to preserve
442e19e40afSRoberto Vargas * some content and it means that we have to read before
443e19e40afSRoberto Vargas * writing
4449da7a653SHaojian Zhuang */
44570cb0bffSYann Gautier if ((skip > 0U) || (padding > 0U)) {
446e19e40afSRoberto Vargas request = ops->read(lba, buf->offset, request);
447e19e40afSRoberto Vargas /*
448e19e40afSRoberto Vargas * The read may return size less than
449e19e40afSRoberto Vargas * requested. Round down to the nearest block
450e19e40afSRoberto Vargas * boundary
451e19e40afSRoberto Vargas */
45270cb0bffSYann Gautier request &= ~(block_size - 1U);
453e19e40afSRoberto Vargas if (request <= skip) {
454e19e40afSRoberto Vargas /*
455e19e40afSRoberto Vargas * We couldn't read enough bytes to jump over
456e19e40afSRoberto Vargas * the skip bytes, so we should have to read
457e19e40afSRoberto Vargas * again the same block, thus generating
458e19e40afSRoberto Vargas * the same error.
459e19e40afSRoberto Vargas */
460e19e40afSRoberto Vargas return -EIO;
4619da7a653SHaojian Zhuang }
462e19e40afSRoberto Vargas nbytes = request - skip;
46370cb0bffSYann Gautier padding = (nbytes > left) ? nbytes - left : 0U;
464e19e40afSRoberto Vargas nbytes -= padding;
465e19e40afSRoberto Vargas }
466e19e40afSRoberto Vargas
467e19e40afSRoberto Vargas memcpy((void *)(buf->offset + skip),
468e19e40afSRoberto Vargas (void *)(buffer + count),
469e19e40afSRoberto Vargas nbytes);
470e19e40afSRoberto Vargas
471e19e40afSRoberto Vargas request = ops->write(lba, buf->offset, request);
472e19e40afSRoberto Vargas if (request <= skip)
473e19e40afSRoberto Vargas return -EIO;
474e19e40afSRoberto Vargas
475e19e40afSRoberto Vargas /*
476e19e40afSRoberto Vargas * And the previous write operation may modify the size
477e19e40afSRoberto Vargas * of the request, so again, we have to calculate the
478e19e40afSRoberto Vargas * number of bytes that we consumed from the user
479e19e40afSRoberto Vargas * buffer
480e19e40afSRoberto Vargas */
481e19e40afSRoberto Vargas nbytes = request - skip;
48270cb0bffSYann Gautier padding = (nbytes > left) ? nbytes - left : 0U;
483e19e40afSRoberto Vargas nbytes -= padding;
484e19e40afSRoberto Vargas
485e19e40afSRoberto Vargas cur->file_pos += nbytes;
486e19e40afSRoberto Vargas count += nbytes;
487e19e40afSRoberto Vargas }
488e19e40afSRoberto Vargas assert(count == length);
489e19e40afSRoberto Vargas *length_written = count;
490e19e40afSRoberto Vargas
4919da7a653SHaojian Zhuang return 0;
4929da7a653SHaojian Zhuang }
4939da7a653SHaojian Zhuang
block_close(io_entity_t * entity)4949da7a653SHaojian Zhuang static int block_close(io_entity_t *entity)
4959da7a653SHaojian Zhuang {
4969da7a653SHaojian Zhuang entity->info = (uintptr_t)NULL;
4979da7a653SHaojian Zhuang return 0;
4989da7a653SHaojian Zhuang }
4999da7a653SHaojian Zhuang
block_dev_open(const uintptr_t dev_spec,io_dev_info_t ** dev_info)5009da7a653SHaojian Zhuang static int block_dev_open(const uintptr_t dev_spec, io_dev_info_t **dev_info)
5019da7a653SHaojian Zhuang {
5029da7a653SHaojian Zhuang block_dev_state_t *cur;
5039da7a653SHaojian Zhuang io_block_spec_t *buffer;
5049da7a653SHaojian Zhuang io_dev_info_t *info;
5059da7a653SHaojian Zhuang size_t block_size;
5069da7a653SHaojian Zhuang int result;
5079da7a653SHaojian Zhuang
5089da7a653SHaojian Zhuang assert(dev_info != NULL);
5099da7a653SHaojian Zhuang result = allocate_dev_info(&info);
51070cb0bffSYann Gautier if (result != 0)
5119da7a653SHaojian Zhuang return -ENOENT;
5129da7a653SHaojian Zhuang
5139da7a653SHaojian Zhuang cur = (block_dev_state_t *)info->info;
5149da7a653SHaojian Zhuang /* dev_spec is type of io_block_dev_spec_t. */
5159da7a653SHaojian Zhuang cur->dev_spec = (io_block_dev_spec_t *)dev_spec;
5169da7a653SHaojian Zhuang buffer = &(cur->dev_spec->buffer);
5179da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size;
51870cb0bffSYann Gautier assert((block_size > 0U) &&
51970cb0bffSYann Gautier (is_power_of_2(block_size) != 0U) &&
52070cb0bffSYann Gautier ((buffer->offset % block_size) == 0U) &&
52170cb0bffSYann Gautier ((buffer->length % block_size) == 0U));
5229da7a653SHaojian Zhuang
5239da7a653SHaojian Zhuang *dev_info = info; /* cast away const */
5249da7a653SHaojian Zhuang (void)block_size;
5259da7a653SHaojian Zhuang (void)buffer;
5269da7a653SHaojian Zhuang return 0;
5279da7a653SHaojian Zhuang }
5289da7a653SHaojian Zhuang
block_dev_close(io_dev_info_t * dev_info)5299da7a653SHaojian Zhuang static int block_dev_close(io_dev_info_t *dev_info)
5309da7a653SHaojian Zhuang {
5319da7a653SHaojian Zhuang return free_dev_info(dev_info);
5329da7a653SHaojian Zhuang }
5339da7a653SHaojian Zhuang
5349da7a653SHaojian Zhuang /* Exported functions */
5359da7a653SHaojian Zhuang
5369da7a653SHaojian Zhuang /* Register the Block driver with the IO abstraction */
register_io_dev_block(const io_dev_connector_t ** dev_con)5379da7a653SHaojian Zhuang int register_io_dev_block(const io_dev_connector_t **dev_con)
5389da7a653SHaojian Zhuang {
5399da7a653SHaojian Zhuang int result;
5409da7a653SHaojian Zhuang
5419da7a653SHaojian Zhuang assert(dev_con != NULL);
5429da7a653SHaojian Zhuang
5439da7a653SHaojian Zhuang /*
5449da7a653SHaojian Zhuang * Since dev_info isn't really used in io_register_device, always
5459da7a653SHaojian Zhuang * use the same device info at here instead.
5469da7a653SHaojian Zhuang */
5479da7a653SHaojian Zhuang result = io_register_device(&dev_info_pool[0]);
5489da7a653SHaojian Zhuang if (result == 0)
5499da7a653SHaojian Zhuang *dev_con = &block_dev_connector;
5509da7a653SHaojian Zhuang return result;
5519da7a653SHaojian Zhuang }
552