19da7a653SHaojian Zhuang /* 232f0d3c6SDouglas Raillard * Copyright (c) 2016-2017, 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; 22*70cb0bffSYann Gautier unsigned long long file_pos; 23*70cb0bffSYann Gautier unsigned long long size; 249da7a653SHaojian Zhuang } block_dev_state_t; 259da7a653SHaojian Zhuang 26*70cb0bffSYann 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); 32*70cb0bffSYann 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 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 */ 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 */ 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 */ 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 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 */ 151*70cb0bffSYann 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; 158*70cb0bffSYann Gautier assert((offset >= 0) && ((unsigned long long)offset < cur->size)); 1599da7a653SHaojian Zhuang 1609da7a653SHaojian Zhuang switch (mode) { 1619da7a653SHaojian Zhuang case IO_SEEK_SET: 162*70cb0bffSYann Gautier cur->file_pos = (unsigned long long)offset; 1639da7a653SHaojian Zhuang break; 1649da7a653SHaojian Zhuang case IO_SEEK_CUR: 165*70cb0bffSYann 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 */ 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) && 273*70cb0bffSYann Gautier (length > 0U) && 2749da7a653SHaojian Zhuang (ops->read != 0)); 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; 282*70cb0bffSYann 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 */ 291*70cb0bffSYann 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 299*70cb0bffSYann 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; 314*70cb0bffSYann Gautier request = (request + (block_size - 1U)) & 315*70cb0bffSYann 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; 334*70cb0bffSYann 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 */ 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) && 385*70cb0bffSYann Gautier (length > 0U) && 3869da7a653SHaojian Zhuang (ops->read != 0) && 3879da7a653SHaojian Zhuang (ops->write != 0)); 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; 395*70cb0bffSYann 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 */ 404*70cb0bffSYann 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 412*70cb0bffSYann 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; 427*70cb0bffSYann Gautier request = (request + (block_size - 1U)) & 428*70cb0bffSYann 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; 437*70cb0bffSYann 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 */ 445*70cb0bffSYann 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 */ 452*70cb0bffSYann 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; 463*70cb0bffSYann 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; 482*70cb0bffSYann 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 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 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); 510*70cb0bffSYann 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; 518*70cb0bffSYann Gautier assert((block_size > 0U) && 519*70cb0bffSYann Gautier (is_power_of_2(block_size) != 0U) && 520*70cb0bffSYann Gautier ((buffer->offset % block_size) == 0U) && 521*70cb0bffSYann 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 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 */ 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