xref: /rk3399_ARM-atf/drivers/io/io_block.c (revision e19e40af9717542ad70e8b9479bca7373b8609fe)
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