 6c46a77d95
			
		
	
	
		6c46a77d95
		
	
	
	
	
		
			
			- The lmfs_get_block*(3) API calls may now return an error. The idea is to encourage a next generation of file system services to do a better job at dealing with block read errors than the MFS-derived implementations do. These existing file systems have been changed to panic immediately upon getting a block read error, in order to let unchecked errors cause corruption. Note that libbdev already retries failing I/O operations a few times first. - The libminixfs block device I/O module (bio.c) now deals properly with end-of-file conditions on block devices. Since a device or partition size may not be a multiple of the root file system's block size, support for partial block retrival has been added, with a new internal lmfs_get_partial_block(3) call. A new test program, test85, tests the new handling of EOF conditions when reading, writing, and memory-mapping a block device. Change-Id: I05e35b6b8851488328a2679da635ebba0c6d08ce
		
			
				
	
	
		
			110 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			110 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "inc.h"
 | |
| 
 | |
| static char getdents_buf[GETDENTS_BUFSIZ];
 | |
| 
 | |
| ssize_t fs_read(ino_t ino_nr, struct fsdriver_data *data, size_t bytes,
 | |
| 	off_t pos, int __unused call)
 | |
| {
 | |
| 	size_t off, chunk, block_size, cum_io;
 | |
| 	off_t f_size;
 | |
| 	struct inode *i_node;
 | |
| 	struct buf *bp;
 | |
| 	int r;
 | |
| 
 | |
| 	/* Try to get inode according to its index. */
 | |
| 	if ((i_node = find_inode(ino_nr)) == NULL)
 | |
| 		return EINVAL; /* No inode found. */
 | |
| 
 | |
| 	f_size = i_node->i_stat.st_size;
 | |
| 	if (pos >= f_size)
 | |
| 		return 0; /* EOF */
 | |
| 
 | |
| 	/* Limit the request to the remainder of the file size. */
 | |
| 	if ((off_t)bytes > f_size - pos)
 | |
| 		bytes = (size_t)(f_size - pos);
 | |
| 
 | |
| 	block_size = v_pri.logical_block_size_l;
 | |
| 	cum_io = 0;
 | |
| 
 | |
| 	r = OK;
 | |
| 
 | |
| 	/* Split the transfer into chunks that don't span two blocks. */
 | |
| 	while (bytes > 0) {
 | |
| 		off = pos % block_size;
 | |
| 
 | |
| 		chunk = block_size - off;
 | |
| 		if (chunk > bytes)
 | |
| 			chunk = bytes;
 | |
| 
 | |
| 		/* Read 'chunk' bytes. */
 | |
| 		bp = read_extent_block(i_node->extent, pos / block_size);
 | |
| 		if (bp == NULL)
 | |
| 			panic("bp not valid in rw_chunk; this can't happen");
 | |
| 
 | |
| 		r = fsdriver_copyout(data, cum_io, b_data(bp)+off, chunk);
 | |
| 
 | |
| 		lmfs_put_block(bp);
 | |
| 
 | |
| 		if (r != OK)
 | |
| 			break;
 | |
| 
 | |
| 		/* Update counters and pointers. */
 | |
| 		bytes -= chunk;		/* Bytes yet to be read. */
 | |
| 		cum_io += chunk;	/* Bytes read so far. */
 | |
| 		pos += chunk;		/* Position within the file. */
 | |
| 	}
 | |
| 
 | |
| 	return (r == OK) ? cum_io : r;
 | |
| }
 | |
| 
 | |
| ssize_t fs_getdents(ino_t ino_nr, struct fsdriver_data *data, size_t bytes,
 | |
| 	off_t *pos)
 | |
| {
 | |
| 	struct fsdriver_dentry fsdentry;
 | |
| 	struct inode *i_node, *i_node_tmp;
 | |
| 	size_t cur_pos, new_pos;
 | |
| 	int r, len;
 | |
| 	char *cp;
 | |
| 
 | |
| 	if ((i_node = find_inode(ino_nr)) == NULL)
 | |
| 		return EINVAL;
 | |
| 
 | |
| 	if (*pos < 0 || *pos > SSIZE_MAX)
 | |
| 		return EINVAL;
 | |
| 
 | |
| 	fsdriver_dentry_init(&fsdentry, data, bytes, getdents_buf,
 | |
| 	    sizeof(getdents_buf));
 | |
| 
 | |
| 	r = OK;
 | |
| 
 | |
| 	for (cur_pos = (size_t)*pos; ; cur_pos = new_pos) {
 | |
| 		i_node_tmp = alloc_inode();
 | |
| 		r = read_inode(i_node_tmp, i_node->extent, cur_pos, &new_pos);
 | |
| 		if ((r != OK) || (new_pos >= i_node->i_stat.st_size)) {
 | |
| 			put_inode(i_node_tmp);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		/* Compute the length of the name */
 | |
| 		cp = memchr(i_node_tmp->i_name, '\0', NAME_MAX);
 | |
| 		if (cp == NULL)
 | |
| 			len = NAME_MAX;
 | |
| 		else
 | |
| 			len = cp - i_node_tmp->i_name;
 | |
| 
 | |
| 		r = fsdriver_dentry_add(&fsdentry, i_node_tmp->i_stat.st_ino,
 | |
| 		    i_node_tmp->i_name, len,
 | |
| 		    IFTODT(i_node_tmp->i_stat.st_mode));
 | |
| 
 | |
| 		put_inode(i_node_tmp);
 | |
| 
 | |
| 		if (r <= 0)
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	if (r >= 0 && (r = fsdriver_dentry_finish(&fsdentry)) >= 0)
 | |
| 		*pos = cur_pos;
 | |
| 
 | |
| 	return r;
 | |
| }
 |