1c2f99d7cSDavid van Moolenbroek #include "inc.h"
2c2f99d7cSDavid van Moolenbroek
3c2f99d7cSDavid van Moolenbroek static char getdents_buf[GETDENTS_BUFSIZ];
4c2f99d7cSDavid van Moolenbroek
fs_read(ino_t ino_nr,struct fsdriver_data * data,size_t bytes,off_t pos,int __unused call)5c2f99d7cSDavid van Moolenbroek ssize_t fs_read(ino_t ino_nr, struct fsdriver_data *data, size_t bytes,
6c2f99d7cSDavid van Moolenbroek off_t pos, int __unused call)
7c2f99d7cSDavid van Moolenbroek {
8c2f99d7cSDavid van Moolenbroek size_t off, chunk, block_size, cum_io;
9c2f99d7cSDavid van Moolenbroek off_t f_size;
10c2f99d7cSDavid van Moolenbroek struct inode *i_node;
11c2f99d7cSDavid van Moolenbroek struct buf *bp;
12c2f99d7cSDavid van Moolenbroek int r;
13c2f99d7cSDavid van Moolenbroek
14c2f99d7cSDavid van Moolenbroek /* Try to get inode according to its index. */
15*b1d06847SJean-Baptiste Boric if ((i_node = get_inode(ino_nr)) == NULL)
16c2f99d7cSDavid van Moolenbroek return EINVAL; /* No inode found. */
17c2f99d7cSDavid van Moolenbroek
18c2f99d7cSDavid van Moolenbroek f_size = i_node->i_stat.st_size;
19c2f99d7cSDavid van Moolenbroek if (pos >= f_size)
20c2f99d7cSDavid van Moolenbroek return 0; /* EOF */
21c2f99d7cSDavid van Moolenbroek
22c2f99d7cSDavid van Moolenbroek /* Limit the request to the remainder of the file size. */
23c2f99d7cSDavid van Moolenbroek if ((off_t)bytes > f_size - pos)
24c2f99d7cSDavid van Moolenbroek bytes = (size_t)(f_size - pos);
25c2f99d7cSDavid van Moolenbroek
26c2f99d7cSDavid van Moolenbroek block_size = v_pri.logical_block_size_l;
27c2f99d7cSDavid van Moolenbroek cum_io = 0;
28c2f99d7cSDavid van Moolenbroek
29c2f99d7cSDavid van Moolenbroek r = OK;
30c2f99d7cSDavid van Moolenbroek
31c2f99d7cSDavid van Moolenbroek /* Split the transfer into chunks that don't span two blocks. */
32c2f99d7cSDavid van Moolenbroek while (bytes > 0) {
33c2f99d7cSDavid van Moolenbroek off = pos % block_size;
34c2f99d7cSDavid van Moolenbroek
35c2f99d7cSDavid van Moolenbroek chunk = block_size - off;
36c2f99d7cSDavid van Moolenbroek if (chunk > bytes)
37c2f99d7cSDavid van Moolenbroek chunk = bytes;
38c2f99d7cSDavid van Moolenbroek
39c2f99d7cSDavid van Moolenbroek /* Read 'chunk' bytes. */
40*b1d06847SJean-Baptiste Boric bp = read_extent_block(&i_node->extent, pos);
41c2f99d7cSDavid van Moolenbroek if (bp == NULL)
42c2f99d7cSDavid van Moolenbroek panic("bp not valid in rw_chunk; this can't happen");
43c2f99d7cSDavid van Moolenbroek
44c2f99d7cSDavid van Moolenbroek r = fsdriver_copyout(data, cum_io, b_data(bp)+off, chunk);
45c2f99d7cSDavid van Moolenbroek
460314acfbSDavid van Moolenbroek lmfs_put_block(bp);
47c2f99d7cSDavid van Moolenbroek
48c2f99d7cSDavid van Moolenbroek if (r != OK)
49c2f99d7cSDavid van Moolenbroek break;
50c2f99d7cSDavid van Moolenbroek
51c2f99d7cSDavid van Moolenbroek /* Update counters and pointers. */
52c2f99d7cSDavid van Moolenbroek bytes -= chunk; /* Bytes yet to be read. */
53c2f99d7cSDavid van Moolenbroek cum_io += chunk; /* Bytes read so far. */
54c2f99d7cSDavid van Moolenbroek pos += chunk; /* Position within the file. */
55c2f99d7cSDavid van Moolenbroek }
56c2f99d7cSDavid van Moolenbroek
57c2f99d7cSDavid van Moolenbroek return (r == OK) ? cum_io : r;
58c2f99d7cSDavid van Moolenbroek }
59c2f99d7cSDavid van Moolenbroek
fs_getdents(ino_t ino_nr,struct fsdriver_data * data,size_t bytes,off_t * pos)60c2f99d7cSDavid van Moolenbroek ssize_t fs_getdents(ino_t ino_nr, struct fsdriver_data *data, size_t bytes,
61c2f99d7cSDavid van Moolenbroek off_t *pos)
62c2f99d7cSDavid van Moolenbroek {
63c2f99d7cSDavid van Moolenbroek struct fsdriver_dentry fsdentry;
64*b1d06847SJean-Baptiste Boric struct inode *i_node;
65*b1d06847SJean-Baptiste Boric off_t cur_pos;
66c2f99d7cSDavid van Moolenbroek int r, len;
67c2f99d7cSDavid van Moolenbroek char *cp;
68c2f99d7cSDavid van Moolenbroek
69*b1d06847SJean-Baptiste Boric if ((i_node = get_inode(ino_nr)) == NULL)
70c2f99d7cSDavid van Moolenbroek return EINVAL;
71c2f99d7cSDavid van Moolenbroek
72c2f99d7cSDavid van Moolenbroek if (*pos < 0 || *pos > SSIZE_MAX)
73c2f99d7cSDavid van Moolenbroek return EINVAL;
74c2f99d7cSDavid van Moolenbroek
75*b1d06847SJean-Baptiste Boric r = read_directory(i_node);
76*b1d06847SJean-Baptiste Boric if (r != OK)
77*b1d06847SJean-Baptiste Boric return r;
78*b1d06847SJean-Baptiste Boric
79c2f99d7cSDavid van Moolenbroek fsdriver_dentry_init(&fsdentry, data, bytes, getdents_buf,
80c2f99d7cSDavid van Moolenbroek sizeof(getdents_buf));
81c2f99d7cSDavid van Moolenbroek
82c2f99d7cSDavid van Moolenbroek r = OK;
83c2f99d7cSDavid van Moolenbroek
84*b1d06847SJean-Baptiste Boric for (cur_pos = *pos; cur_pos < i_node->dir_size; cur_pos++) {
85c2f99d7cSDavid van Moolenbroek /* Compute the length of the name */
86*b1d06847SJean-Baptiste Boric cp = memchr(i_node->dir_contents[cur_pos].name, '\0', NAME_MAX);
87c2f99d7cSDavid van Moolenbroek if (cp == NULL)
88c2f99d7cSDavid van Moolenbroek len = NAME_MAX;
89c2f99d7cSDavid van Moolenbroek else
90*b1d06847SJean-Baptiste Boric len = cp - i_node->dir_contents[cur_pos].name;
91c2f99d7cSDavid van Moolenbroek
92*b1d06847SJean-Baptiste Boric r = fsdriver_dentry_add(&fsdentry,
93*b1d06847SJean-Baptiste Boric i_node->dir_contents[cur_pos].i_node->i_stat.st_ino,
94*b1d06847SJean-Baptiste Boric i_node->dir_contents[cur_pos].name, len,
95*b1d06847SJean-Baptiste Boric IFTODT(i_node->dir_contents[cur_pos].i_node->i_stat.st_mode));
96c2f99d7cSDavid van Moolenbroek
97c2f99d7cSDavid van Moolenbroek if (r <= 0)
98c2f99d7cSDavid van Moolenbroek break;
99c2f99d7cSDavid van Moolenbroek }
100c2f99d7cSDavid van Moolenbroek
101c2f99d7cSDavid van Moolenbroek if (r >= 0 && (r = fsdriver_dentry_finish(&fsdentry)) >= 0)
102c2f99d7cSDavid van Moolenbroek *pos = cur_pos;
103c2f99d7cSDavid van Moolenbroek
104c2f99d7cSDavid van Moolenbroek return r;
105c2f99d7cSDavid van Moolenbroek }
106