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 <debug.h> 99da7a653SHaojian Zhuang #include <errno.h> 109da7a653SHaojian Zhuang #include <io_block.h> 119da7a653SHaojian Zhuang #include <io_driver.h> 129da7a653SHaojian Zhuang #include <io_storage.h> 139da7a653SHaojian Zhuang #include <platform_def.h> 149da7a653SHaojian Zhuang #include <string.h> 1532f0d3c6SDouglas Raillard #include <utils.h> 169da7a653SHaojian Zhuang 179da7a653SHaojian Zhuang typedef struct { 189da7a653SHaojian Zhuang io_block_dev_spec_t *dev_spec; 199da7a653SHaojian Zhuang uintptr_t base; 209da7a653SHaojian Zhuang size_t file_pos; 219da7a653SHaojian Zhuang size_t size; 229da7a653SHaojian Zhuang } block_dev_state_t; 239da7a653SHaojian Zhuang 249da7a653SHaojian Zhuang #define is_power_of_2(x) ((x != 0) && ((x & (x - 1)) == 0)) 259da7a653SHaojian Zhuang 269da7a653SHaojian Zhuang io_type_t device_type_block(void); 279da7a653SHaojian Zhuang 289da7a653SHaojian Zhuang static int block_open(io_dev_info_t *dev_info, const uintptr_t spec, 299da7a653SHaojian Zhuang io_entity_t *entity); 309da7a653SHaojian Zhuang static int block_seek(io_entity_t *entity, int mode, ssize_t offset); 319da7a653SHaojian Zhuang static int block_read(io_entity_t *entity, uintptr_t buffer, size_t length, 329da7a653SHaojian Zhuang size_t *length_read); 339da7a653SHaojian Zhuang static int block_write(io_entity_t *entity, const uintptr_t buffer, 349da7a653SHaojian Zhuang size_t length, size_t *length_written); 359da7a653SHaojian Zhuang static int block_close(io_entity_t *entity); 369da7a653SHaojian Zhuang static int block_dev_open(const uintptr_t dev_spec, io_dev_info_t **dev_info); 379da7a653SHaojian Zhuang static int block_dev_close(io_dev_info_t *dev_info); 389da7a653SHaojian Zhuang 399da7a653SHaojian Zhuang static const io_dev_connector_t block_dev_connector = { 409da7a653SHaojian Zhuang .dev_open = block_dev_open 419da7a653SHaojian Zhuang }; 429da7a653SHaojian Zhuang 439da7a653SHaojian Zhuang static const io_dev_funcs_t block_dev_funcs = { 449da7a653SHaojian Zhuang .type = device_type_block, 459da7a653SHaojian Zhuang .open = block_open, 469da7a653SHaojian Zhuang .seek = block_seek, 479da7a653SHaojian Zhuang .size = NULL, 489da7a653SHaojian Zhuang .read = block_read, 499da7a653SHaojian Zhuang .write = block_write, 509da7a653SHaojian Zhuang .close = block_close, 519da7a653SHaojian Zhuang .dev_init = NULL, 529da7a653SHaojian Zhuang .dev_close = block_dev_close, 539da7a653SHaojian Zhuang }; 549da7a653SHaojian Zhuang 559da7a653SHaojian Zhuang static block_dev_state_t state_pool[MAX_IO_BLOCK_DEVICES]; 569da7a653SHaojian Zhuang static io_dev_info_t dev_info_pool[MAX_IO_BLOCK_DEVICES]; 579da7a653SHaojian Zhuang 589da7a653SHaojian Zhuang /* Track number of allocated block state */ 599da7a653SHaojian Zhuang static unsigned int block_dev_count; 609da7a653SHaojian Zhuang 619da7a653SHaojian Zhuang io_type_t device_type_block(void) 629da7a653SHaojian Zhuang { 639da7a653SHaojian Zhuang return IO_TYPE_BLOCK; 649da7a653SHaojian Zhuang } 659da7a653SHaojian Zhuang 669da7a653SHaojian Zhuang /* Locate a block state in the pool, specified by address */ 679da7a653SHaojian Zhuang static int find_first_block_state(const io_block_dev_spec_t *dev_spec, 689da7a653SHaojian Zhuang unsigned int *index_out) 699da7a653SHaojian Zhuang { 709da7a653SHaojian Zhuang int result = -ENOENT; 719da7a653SHaojian Zhuang for (int index = 0; index < MAX_IO_BLOCK_DEVICES; ++index) { 729da7a653SHaojian Zhuang /* dev_spec is used as identifier since it's unique */ 739da7a653SHaojian Zhuang if (state_pool[index].dev_spec == dev_spec) { 749da7a653SHaojian Zhuang result = 0; 759da7a653SHaojian Zhuang *index_out = index; 769da7a653SHaojian Zhuang break; 779da7a653SHaojian Zhuang } 789da7a653SHaojian Zhuang } 799da7a653SHaojian Zhuang return result; 809da7a653SHaojian Zhuang } 819da7a653SHaojian Zhuang 829da7a653SHaojian Zhuang /* Allocate a device info from the pool and return a pointer to it */ 839da7a653SHaojian Zhuang static int allocate_dev_info(io_dev_info_t **dev_info) 849da7a653SHaojian Zhuang { 859da7a653SHaojian Zhuang int result = -ENOMEM; 869da7a653SHaojian Zhuang assert(dev_info != NULL); 879da7a653SHaojian Zhuang 889da7a653SHaojian Zhuang if (block_dev_count < MAX_IO_BLOCK_DEVICES) { 899da7a653SHaojian Zhuang unsigned int index = 0; 909da7a653SHaojian Zhuang result = find_first_block_state(NULL, &index); 919da7a653SHaojian Zhuang assert(result == 0); 929da7a653SHaojian Zhuang /* initialize dev_info */ 939da7a653SHaojian Zhuang dev_info_pool[index].funcs = &block_dev_funcs; 949da7a653SHaojian Zhuang dev_info_pool[index].info = (uintptr_t)&state_pool[index]; 959da7a653SHaojian Zhuang *dev_info = &dev_info_pool[index]; 969da7a653SHaojian Zhuang ++block_dev_count; 979da7a653SHaojian Zhuang } 989da7a653SHaojian Zhuang 999da7a653SHaojian Zhuang return result; 1009da7a653SHaojian Zhuang } 1019da7a653SHaojian Zhuang 1029da7a653SHaojian Zhuang 1039da7a653SHaojian Zhuang /* Release a device info to the pool */ 1049da7a653SHaojian Zhuang static int free_dev_info(io_dev_info_t *dev_info) 1059da7a653SHaojian Zhuang { 1069da7a653SHaojian Zhuang int result; 1079da7a653SHaojian Zhuang unsigned int index = 0; 1089da7a653SHaojian Zhuang block_dev_state_t *state; 1099da7a653SHaojian Zhuang assert(dev_info != NULL); 1109da7a653SHaojian Zhuang 1119da7a653SHaojian Zhuang state = (block_dev_state_t *)dev_info->info; 1129da7a653SHaojian Zhuang result = find_first_block_state(state->dev_spec, &index); 1139da7a653SHaojian Zhuang if (result == 0) { 1149da7a653SHaojian Zhuang /* free if device info is valid */ 11532f0d3c6SDouglas Raillard zeromem(state, sizeof(block_dev_state_t)); 11632f0d3c6SDouglas Raillard zeromem(dev_info, sizeof(io_dev_info_t)); 1179da7a653SHaojian Zhuang --block_dev_count; 1189da7a653SHaojian Zhuang } 1199da7a653SHaojian Zhuang 1209da7a653SHaojian Zhuang return result; 1219da7a653SHaojian Zhuang } 1229da7a653SHaojian Zhuang 1239da7a653SHaojian Zhuang static int block_open(io_dev_info_t *dev_info, const uintptr_t spec, 1249da7a653SHaojian Zhuang io_entity_t *entity) 1259da7a653SHaojian Zhuang { 1269da7a653SHaojian Zhuang block_dev_state_t *cur; 1279da7a653SHaojian Zhuang io_block_spec_t *region; 1289da7a653SHaojian Zhuang 1299da7a653SHaojian Zhuang assert((dev_info->info != (uintptr_t)NULL) && 1309da7a653SHaojian Zhuang (spec != (uintptr_t)NULL) && 1319da7a653SHaojian Zhuang (entity->info == (uintptr_t)NULL)); 1329da7a653SHaojian Zhuang 1339da7a653SHaojian Zhuang region = (io_block_spec_t *)spec; 1349da7a653SHaojian Zhuang cur = (block_dev_state_t *)dev_info->info; 1359da7a653SHaojian Zhuang assert(((region->offset % cur->dev_spec->block_size) == 0) && 1369da7a653SHaojian Zhuang ((region->length % cur->dev_spec->block_size) == 0)); 1379da7a653SHaojian Zhuang 1389da7a653SHaojian Zhuang cur->base = region->offset; 1399da7a653SHaojian Zhuang cur->size = region->length; 1409da7a653SHaojian Zhuang cur->file_pos = 0; 1419da7a653SHaojian Zhuang 1429da7a653SHaojian Zhuang entity->info = (uintptr_t)cur; 1439da7a653SHaojian Zhuang return 0; 1449da7a653SHaojian Zhuang } 1459da7a653SHaojian Zhuang 1469da7a653SHaojian Zhuang /* parameter offset is relative address at here */ 1479da7a653SHaojian Zhuang static int block_seek(io_entity_t *entity, int mode, ssize_t offset) 1489da7a653SHaojian Zhuang { 1499da7a653SHaojian Zhuang block_dev_state_t *cur; 1509da7a653SHaojian Zhuang 1519da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL); 1529da7a653SHaojian Zhuang 1539da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info; 1549da7a653SHaojian Zhuang assert((offset >= 0) && (offset < cur->size)); 1559da7a653SHaojian Zhuang 1569da7a653SHaojian Zhuang switch (mode) { 1579da7a653SHaojian Zhuang case IO_SEEK_SET: 1589da7a653SHaojian Zhuang cur->file_pos = offset; 1599da7a653SHaojian Zhuang break; 1609da7a653SHaojian Zhuang case IO_SEEK_CUR: 1619da7a653SHaojian Zhuang cur->file_pos += offset; 1629da7a653SHaojian Zhuang break; 1639da7a653SHaojian Zhuang default: 1649da7a653SHaojian Zhuang return -EINVAL; 1659da7a653SHaojian Zhuang } 1669da7a653SHaojian Zhuang assert(cur->file_pos < cur->size); 1679da7a653SHaojian Zhuang return 0; 1689da7a653SHaojian Zhuang } 1699da7a653SHaojian Zhuang 170*e19e40afSRoberto Vargas /* 171*e19e40afSRoberto Vargas * This function allows the caller to read any number of bytes 172*e19e40afSRoberto Vargas * from any position. It hides from the caller that the low level 173*e19e40afSRoberto Vargas * driver only can read aligned blocks of data. For this reason 174*e19e40afSRoberto Vargas * we need to handle the use case where the first byte to be read is not 175*e19e40afSRoberto Vargas * aligned to start of the block, the last byte to be read is also not 176*e19e40afSRoberto Vargas * aligned to the end of a block, and there are zero or more blocks-worth 177*e19e40afSRoberto Vargas * of data in between. 178*e19e40afSRoberto Vargas * 179*e19e40afSRoberto Vargas * In such a case we need to read more bytes than requested (i.e. full 180*e19e40afSRoberto Vargas * blocks) and strip-out the leading bytes (aka skip) and the trailing 181*e19e40afSRoberto Vargas * bytes (aka padding). See diagram below 182*e19e40afSRoberto Vargas * 183*e19e40afSRoberto Vargas * cur->file_pos ------------ 184*e19e40afSRoberto Vargas * | 185*e19e40afSRoberto Vargas * cur->base | 186*e19e40afSRoberto Vargas * | | 187*e19e40afSRoberto Vargas * v v<---- length ----> 188*e19e40afSRoberto Vargas * -------------------------------------------------------------- 189*e19e40afSRoberto Vargas * | | block#1 | | block#n | 190*e19e40afSRoberto Vargas * | block#0 | + | ... | + | 191*e19e40afSRoberto Vargas * | | <- skip -> + | | + <- padding ->| 192*e19e40afSRoberto Vargas * ------------------------+----------------------+-------------- 193*e19e40afSRoberto Vargas * ^ ^ 194*e19e40afSRoberto Vargas * | | 195*e19e40afSRoberto Vargas * v iteration#1 iteration#n v 196*e19e40afSRoberto Vargas * -------------------------------------------------- 197*e19e40afSRoberto Vargas * | | | | 198*e19e40afSRoberto Vargas * |<---- request ---->| ... |<----- request ---->| 199*e19e40afSRoberto Vargas * | | | | 200*e19e40afSRoberto Vargas * -------------------------------------------------- 201*e19e40afSRoberto Vargas * / / | | 202*e19e40afSRoberto Vargas * / / | | 203*e19e40afSRoberto Vargas * / / | | 204*e19e40afSRoberto Vargas * / / | | 205*e19e40afSRoberto Vargas * / / | | 206*e19e40afSRoberto Vargas * / / | | 207*e19e40afSRoberto Vargas * / / | | 208*e19e40afSRoberto Vargas * / / | | 209*e19e40afSRoberto Vargas * / / | | 210*e19e40afSRoberto Vargas * / / | | 211*e19e40afSRoberto Vargas * <---- request ------> <------ request -----> 212*e19e40afSRoberto Vargas * --------------------- ----------------------- 213*e19e40afSRoberto Vargas * | | | | | | 214*e19e40afSRoberto Vargas * |<-skip->|<-nbytes->| -------->|<-nbytes->|<-padding->| 215*e19e40afSRoberto Vargas * | | | | | | | 216*e19e40afSRoberto Vargas * --------------------- | ----------------------- 217*e19e40afSRoberto Vargas * ^ \ \ | | | 218*e19e40afSRoberto Vargas * | \ \ | | | 219*e19e40afSRoberto Vargas * | \ \ | | | 220*e19e40afSRoberto Vargas * buf->offset \ \ buf->offset | | 221*e19e40afSRoberto Vargas * \ \ | | 222*e19e40afSRoberto Vargas * \ \ | | 223*e19e40afSRoberto Vargas * \ \ | | 224*e19e40afSRoberto Vargas * \ \ | | 225*e19e40afSRoberto Vargas * \ \ | | 226*e19e40afSRoberto Vargas * \ \ | | 227*e19e40afSRoberto Vargas * \ \ | | 228*e19e40afSRoberto Vargas * -------------------------------- 229*e19e40afSRoberto Vargas * | | | | 230*e19e40afSRoberto Vargas * buffer-------------->| | ... | | 231*e19e40afSRoberto Vargas * | | | | 232*e19e40afSRoberto Vargas * -------------------------------- 233*e19e40afSRoberto Vargas * <-count#1->| | 234*e19e40afSRoberto Vargas * <---------- count#n --------> 235*e19e40afSRoberto Vargas * <---------- length ----------> 236*e19e40afSRoberto Vargas * 237*e19e40afSRoberto Vargas * Additionally, the IO driver has an underlying buffer that is at least 238*e19e40afSRoberto Vargas * one block-size and may be big enough to allow. 239*e19e40afSRoberto Vargas */ 2409da7a653SHaojian Zhuang static int block_read(io_entity_t *entity, uintptr_t buffer, size_t length, 2419da7a653SHaojian Zhuang size_t *length_read) 2429da7a653SHaojian Zhuang { 2439da7a653SHaojian Zhuang block_dev_state_t *cur; 2449da7a653SHaojian Zhuang io_block_spec_t *buf; 2459da7a653SHaojian Zhuang io_block_ops_t *ops; 2469da7a653SHaojian Zhuang int lba; 247*e19e40afSRoberto Vargas size_t block_size, left; 248*e19e40afSRoberto Vargas size_t nbytes; /* number of bytes read in one iteration */ 249*e19e40afSRoberto Vargas size_t request; /* number of requested bytes in one iteration */ 250*e19e40afSRoberto Vargas size_t count; /* number of bytes already read */ 251*e19e40afSRoberto Vargas /* 252*e19e40afSRoberto Vargas * number of leading bytes from start of the block 253*e19e40afSRoberto Vargas * to the first byte to be read 254*e19e40afSRoberto Vargas */ 255*e19e40afSRoberto Vargas size_t skip; 256*e19e40afSRoberto Vargas 257*e19e40afSRoberto Vargas /* 258*e19e40afSRoberto Vargas * number of trailing bytes between the last byte 259*e19e40afSRoberto Vargas * to be read and the end of the block 260*e19e40afSRoberto Vargas */ 261*e19e40afSRoberto Vargas size_t padding; 2629da7a653SHaojian Zhuang 2639da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL); 2649da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info; 2659da7a653SHaojian Zhuang ops = &(cur->dev_spec->ops); 2669da7a653SHaojian Zhuang buf = &(cur->dev_spec->buffer); 2679da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size; 2689da7a653SHaojian Zhuang assert((length <= cur->size) && 2699da7a653SHaojian Zhuang (length > 0) && 2709da7a653SHaojian Zhuang (ops->read != 0)); 2719da7a653SHaojian Zhuang 2729d063aa2SHaojian Zhuang /* 273*e19e40afSRoberto Vargas * We don't know the number of bytes that we are going 274*e19e40afSRoberto Vargas * to read in every iteration, because it will depend 275*e19e40afSRoberto Vargas * on the low level driver. 2769d063aa2SHaojian Zhuang */ 277*e19e40afSRoberto Vargas count = 0; 278*e19e40afSRoberto Vargas for (left = length; left > 0; left -= nbytes) { 279*e19e40afSRoberto Vargas /* 280*e19e40afSRoberto Vargas * We must only request operations aligned to the block 281*e19e40afSRoberto Vargas * size. Therefore if file_pos is not block-aligned, 282*e19e40afSRoberto Vargas * we have to request the operation to start at the 283*e19e40afSRoberto Vargas * previous block boundary and skip the leading bytes. And 284*e19e40afSRoberto Vargas * similarly, the number of bytes requested must be a 285*e19e40afSRoberto Vargas * block size multiple 286*e19e40afSRoberto Vargas */ 287*e19e40afSRoberto Vargas skip = cur->file_pos & (block_size - 1); 288*e19e40afSRoberto Vargas 289*e19e40afSRoberto Vargas /* 290*e19e40afSRoberto Vargas * Calculate the block number containing file_pos 291*e19e40afSRoberto Vargas * - e.g. block 3. 292*e19e40afSRoberto Vargas */ 293*e19e40afSRoberto Vargas lba = (cur->file_pos + cur->base) / block_size; 294*e19e40afSRoberto Vargas 295*e19e40afSRoberto Vargas if (skip + left > buf->length) { 296*e19e40afSRoberto Vargas /* 297*e19e40afSRoberto Vargas * The underlying read buffer is too small to 298*e19e40afSRoberto Vargas * read all the required data - limit to just 299*e19e40afSRoberto Vargas * fill the buffer, and then read again. 300*e19e40afSRoberto Vargas */ 301*e19e40afSRoberto Vargas request = buf->length; 3029d063aa2SHaojian Zhuang } else { 303*e19e40afSRoberto Vargas /* 304*e19e40afSRoberto Vargas * The underlying read buffer is big enough to 305*e19e40afSRoberto Vargas * read all the required data. Calculate the 306*e19e40afSRoberto Vargas * number of bytes to read to align with the 307*e19e40afSRoberto Vargas * block size. 308*e19e40afSRoberto Vargas */ 309*e19e40afSRoberto Vargas request = skip + left; 310*e19e40afSRoberto Vargas request = (request + (block_size - 1)) & ~(block_size - 1); 311*e19e40afSRoberto Vargas } 312*e19e40afSRoberto Vargas request = ops->read(lba, buf->offset, request); 313*e19e40afSRoberto Vargas 314*e19e40afSRoberto Vargas if (request <= skip) { 315*e19e40afSRoberto Vargas /* 316*e19e40afSRoberto Vargas * We couldn't read enough bytes to jump over 317*e19e40afSRoberto Vargas * the skip bytes, so we should have to read 318*e19e40afSRoberto Vargas * again the same block, thus generating 319*e19e40afSRoberto Vargas * the same error. 320*e19e40afSRoberto Vargas */ 321*e19e40afSRoberto Vargas return -EIO; 3229d063aa2SHaojian Zhuang } 3239d063aa2SHaojian Zhuang 3249d063aa2SHaojian Zhuang /* 325*e19e40afSRoberto Vargas * Need to remove skip and padding bytes,if any, from 326*e19e40afSRoberto Vargas * the read data when copying to the user buffer. 3279d063aa2SHaojian Zhuang */ 328*e19e40afSRoberto Vargas nbytes = request - skip; 329*e19e40afSRoberto Vargas padding = (nbytes > left) ? nbytes - left : 0; 330*e19e40afSRoberto Vargas nbytes -= padding; 331*e19e40afSRoberto Vargas 332*e19e40afSRoberto Vargas memcpy((void *)(buffer + count), 3339da7a653SHaojian Zhuang (void *)(buf->offset + skip), 334*e19e40afSRoberto Vargas nbytes); 335*e19e40afSRoberto Vargas 336*e19e40afSRoberto Vargas cur->file_pos += nbytes; 337*e19e40afSRoberto Vargas count += nbytes; 3389da7a653SHaojian Zhuang } 339*e19e40afSRoberto Vargas assert(count == length); 340*e19e40afSRoberto Vargas *length_read = count; 3419da7a653SHaojian Zhuang 3429da7a653SHaojian Zhuang return 0; 3439da7a653SHaojian Zhuang } 3449da7a653SHaojian Zhuang 345*e19e40afSRoberto Vargas /* 346*e19e40afSRoberto Vargas * This function allows the caller to write any number of bytes 347*e19e40afSRoberto Vargas * from any position. It hides from the caller that the low level 348*e19e40afSRoberto Vargas * driver only can write aligned blocks of data. 349*e19e40afSRoberto Vargas * See comments for block_read for more details. 350*e19e40afSRoberto Vargas */ 3519da7a653SHaojian Zhuang static int block_write(io_entity_t *entity, const uintptr_t buffer, 3529da7a653SHaojian Zhuang size_t length, size_t *length_written) 3539da7a653SHaojian Zhuang { 3549da7a653SHaojian Zhuang block_dev_state_t *cur; 3559da7a653SHaojian Zhuang io_block_spec_t *buf; 3569da7a653SHaojian Zhuang io_block_ops_t *ops; 3579da7a653SHaojian Zhuang int lba; 358*e19e40afSRoberto Vargas size_t block_size, left; 359*e19e40afSRoberto Vargas size_t nbytes; /* number of bytes read in one iteration */ 360*e19e40afSRoberto Vargas size_t request; /* number of requested bytes in one iteration */ 361*e19e40afSRoberto Vargas size_t count; /* number of bytes already read */ 362*e19e40afSRoberto Vargas /* 363*e19e40afSRoberto Vargas * number of leading bytes from start of the block 364*e19e40afSRoberto Vargas * to the first byte to be read 365*e19e40afSRoberto Vargas */ 366*e19e40afSRoberto Vargas size_t skip; 367*e19e40afSRoberto Vargas 368*e19e40afSRoberto Vargas /* 369*e19e40afSRoberto Vargas * number of trailing bytes between the last byte 370*e19e40afSRoberto Vargas * to be read and the end of the block 371*e19e40afSRoberto Vargas */ 372*e19e40afSRoberto Vargas size_t padding; 3739da7a653SHaojian Zhuang 3749da7a653SHaojian Zhuang assert(entity->info != (uintptr_t)NULL); 3759da7a653SHaojian Zhuang cur = (block_dev_state_t *)entity->info; 3769da7a653SHaojian Zhuang ops = &(cur->dev_spec->ops); 3779da7a653SHaojian Zhuang buf = &(cur->dev_spec->buffer); 3789da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size; 3799da7a653SHaojian Zhuang assert((length <= cur->size) && 3809da7a653SHaojian Zhuang (length > 0) && 3819da7a653SHaojian Zhuang (ops->read != 0) && 3829da7a653SHaojian Zhuang (ops->write != 0)); 3839da7a653SHaojian Zhuang 3849d063aa2SHaojian Zhuang /* 385*e19e40afSRoberto Vargas * We don't know the number of bytes that we are going 386*e19e40afSRoberto Vargas * to write in every iteration, because it will depend 387*e19e40afSRoberto Vargas * on the low level driver. 3889d063aa2SHaojian Zhuang */ 389*e19e40afSRoberto Vargas count = 0; 390*e19e40afSRoberto Vargas for (left = length; left > 0; left -= nbytes) { 391*e19e40afSRoberto Vargas /* 392*e19e40afSRoberto Vargas * We must only request operations aligned to the block 393*e19e40afSRoberto Vargas * size. Therefore if file_pos is not block-aligned, 394*e19e40afSRoberto Vargas * we have to request the operation to start at the 395*e19e40afSRoberto Vargas * previous block boundary and skip the leading bytes. And 396*e19e40afSRoberto Vargas * similarly, the number of bytes requested must be a 397*e19e40afSRoberto Vargas * block size multiple 398*e19e40afSRoberto Vargas */ 399*e19e40afSRoberto Vargas skip = cur->file_pos & (block_size - 1); 400*e19e40afSRoberto Vargas 401*e19e40afSRoberto Vargas /* 402*e19e40afSRoberto Vargas * Calculate the block number containing file_pos 403*e19e40afSRoberto Vargas * - e.g. block 3. 404*e19e40afSRoberto Vargas */ 405*e19e40afSRoberto Vargas lba = (cur->file_pos + cur->base) / block_size; 406*e19e40afSRoberto Vargas 407*e19e40afSRoberto Vargas if (skip + left > buf->length) { 408*e19e40afSRoberto Vargas /* 409*e19e40afSRoberto Vargas * The underlying read buffer is too small to 410*e19e40afSRoberto Vargas * read all the required data - limit to just 411*e19e40afSRoberto Vargas * fill the buffer, and then read again. 412*e19e40afSRoberto Vargas */ 413*e19e40afSRoberto Vargas request = buf->length; 4149d063aa2SHaojian Zhuang } else { 415*e19e40afSRoberto Vargas /* 416*e19e40afSRoberto Vargas * The underlying read buffer is big enough to 417*e19e40afSRoberto Vargas * read all the required data. Calculate the 418*e19e40afSRoberto Vargas * number of bytes to read to align with the 419*e19e40afSRoberto Vargas * block size. 420*e19e40afSRoberto Vargas */ 421*e19e40afSRoberto Vargas request = skip + left; 422*e19e40afSRoberto Vargas request = (request + (block_size - 1)) & ~(block_size - 1); 4239d063aa2SHaojian Zhuang } 4249d063aa2SHaojian Zhuang 4259da7a653SHaojian Zhuang /* 426*e19e40afSRoberto Vargas * The number of bytes that we are going to write 427*e19e40afSRoberto Vargas * from the user buffer will depend of the size 428*e19e40afSRoberto Vargas * of the current request. 4299da7a653SHaojian Zhuang */ 430*e19e40afSRoberto Vargas nbytes = request - skip; 431*e19e40afSRoberto Vargas padding = (nbytes > left) ? nbytes - left : 0; 432*e19e40afSRoberto Vargas nbytes -= padding; 433*e19e40afSRoberto Vargas 4349da7a653SHaojian Zhuang /* 435*e19e40afSRoberto Vargas * If we have skip or padding bytes then we have to preserve 436*e19e40afSRoberto Vargas * some content and it means that we have to read before 437*e19e40afSRoberto Vargas * writing 4389da7a653SHaojian Zhuang */ 439*e19e40afSRoberto Vargas if (skip > 0 || padding > 0) { 440*e19e40afSRoberto Vargas request = ops->read(lba, buf->offset, request); 441*e19e40afSRoberto Vargas /* 442*e19e40afSRoberto Vargas * The read may return size less than 443*e19e40afSRoberto Vargas * requested. Round down to the nearest block 444*e19e40afSRoberto Vargas * boundary 445*e19e40afSRoberto Vargas */ 446*e19e40afSRoberto Vargas request &= ~(block_size-1); 447*e19e40afSRoberto Vargas if (request <= skip) { 448*e19e40afSRoberto Vargas /* 449*e19e40afSRoberto Vargas * We couldn't read enough bytes to jump over 450*e19e40afSRoberto Vargas * the skip bytes, so we should have to read 451*e19e40afSRoberto Vargas * again the same block, thus generating 452*e19e40afSRoberto Vargas * the same error. 453*e19e40afSRoberto Vargas */ 454*e19e40afSRoberto Vargas return -EIO; 4559da7a653SHaojian Zhuang } 456*e19e40afSRoberto Vargas nbytes = request - skip; 457*e19e40afSRoberto Vargas padding = (nbytes > left) ? nbytes - left : 0; 458*e19e40afSRoberto Vargas nbytes -= padding; 459*e19e40afSRoberto Vargas } 460*e19e40afSRoberto Vargas 461*e19e40afSRoberto Vargas memcpy((void *)(buf->offset + skip), 462*e19e40afSRoberto Vargas (void *)(buffer + count), 463*e19e40afSRoberto Vargas nbytes); 464*e19e40afSRoberto Vargas 465*e19e40afSRoberto Vargas request = ops->write(lba, buf->offset, request); 466*e19e40afSRoberto Vargas if (request <= skip) 467*e19e40afSRoberto Vargas return -EIO; 468*e19e40afSRoberto Vargas 469*e19e40afSRoberto Vargas /* 470*e19e40afSRoberto Vargas * And the previous write operation may modify the size 471*e19e40afSRoberto Vargas * of the request, so again, we have to calculate the 472*e19e40afSRoberto Vargas * number of bytes that we consumed from the user 473*e19e40afSRoberto Vargas * buffer 474*e19e40afSRoberto Vargas */ 475*e19e40afSRoberto Vargas nbytes = request - skip; 476*e19e40afSRoberto Vargas padding = (nbytes > left) ? nbytes - left : 0; 477*e19e40afSRoberto Vargas nbytes -= padding; 478*e19e40afSRoberto Vargas 479*e19e40afSRoberto Vargas cur->file_pos += nbytes; 480*e19e40afSRoberto Vargas count += nbytes; 481*e19e40afSRoberto Vargas } 482*e19e40afSRoberto Vargas assert(count == length); 483*e19e40afSRoberto Vargas *length_written = count; 484*e19e40afSRoberto Vargas 4859da7a653SHaojian Zhuang return 0; 4869da7a653SHaojian Zhuang } 4879da7a653SHaojian Zhuang 4889da7a653SHaojian Zhuang static int block_close(io_entity_t *entity) 4899da7a653SHaojian Zhuang { 4909da7a653SHaojian Zhuang entity->info = (uintptr_t)NULL; 4919da7a653SHaojian Zhuang return 0; 4929da7a653SHaojian Zhuang } 4939da7a653SHaojian Zhuang 4949da7a653SHaojian Zhuang static int block_dev_open(const uintptr_t dev_spec, io_dev_info_t **dev_info) 4959da7a653SHaojian Zhuang { 4969da7a653SHaojian Zhuang block_dev_state_t *cur; 4979da7a653SHaojian Zhuang io_block_spec_t *buffer; 4989da7a653SHaojian Zhuang io_dev_info_t *info; 4999da7a653SHaojian Zhuang size_t block_size; 5009da7a653SHaojian Zhuang int result; 5019da7a653SHaojian Zhuang 5029da7a653SHaojian Zhuang assert(dev_info != NULL); 5039da7a653SHaojian Zhuang result = allocate_dev_info(&info); 5049da7a653SHaojian Zhuang if (result) 5059da7a653SHaojian Zhuang return -ENOENT; 5069da7a653SHaojian Zhuang 5079da7a653SHaojian Zhuang cur = (block_dev_state_t *)info->info; 5089da7a653SHaojian Zhuang /* dev_spec is type of io_block_dev_spec_t. */ 5099da7a653SHaojian Zhuang cur->dev_spec = (io_block_dev_spec_t *)dev_spec; 5109da7a653SHaojian Zhuang buffer = &(cur->dev_spec->buffer); 5119da7a653SHaojian Zhuang block_size = cur->dev_spec->block_size; 5129da7a653SHaojian Zhuang assert((block_size > 0) && 5139da7a653SHaojian Zhuang (is_power_of_2(block_size) != 0) && 5149da7a653SHaojian Zhuang ((buffer->offset % block_size) == 0) && 5159da7a653SHaojian Zhuang ((buffer->length % block_size) == 0)); 5169da7a653SHaojian Zhuang 5179da7a653SHaojian Zhuang *dev_info = info; /* cast away const */ 5189da7a653SHaojian Zhuang (void)block_size; 5199da7a653SHaojian Zhuang (void)buffer; 5209da7a653SHaojian Zhuang return 0; 5219da7a653SHaojian Zhuang } 5229da7a653SHaojian Zhuang 5239da7a653SHaojian Zhuang static int block_dev_close(io_dev_info_t *dev_info) 5249da7a653SHaojian Zhuang { 5259da7a653SHaojian Zhuang return free_dev_info(dev_info); 5269da7a653SHaojian Zhuang } 5279da7a653SHaojian Zhuang 5289da7a653SHaojian Zhuang /* Exported functions */ 5299da7a653SHaojian Zhuang 5309da7a653SHaojian Zhuang /* Register the Block driver with the IO abstraction */ 5319da7a653SHaojian Zhuang int register_io_dev_block(const io_dev_connector_t **dev_con) 5329da7a653SHaojian Zhuang { 5339da7a653SHaojian Zhuang int result; 5349da7a653SHaojian Zhuang 5359da7a653SHaojian Zhuang assert(dev_con != NULL); 5369da7a653SHaojian Zhuang 5379da7a653SHaojian Zhuang /* 5389da7a653SHaojian Zhuang * Since dev_info isn't really used in io_register_device, always 5399da7a653SHaojian Zhuang * use the same device info at here instead. 5409da7a653SHaojian Zhuang */ 5419da7a653SHaojian Zhuang result = io_register_device(&dev_info_pool[0]); 5429da7a653SHaojian Zhuang if (result == 0) 5439da7a653SHaojian Zhuang *dev_con = &block_dev_connector; 5449da7a653SHaojian Zhuang return result; 5459da7a653SHaojian Zhuang } 546