1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
|
#include <init/fs/misc.h>
#include <init/stdlib.h>
#include <shared/flags.h>
#include <shared/mem.h>
#include <shared/syscalls.h>
bool fork2_n_mount(const char *path) {
handle_t h = _syscall_fs_fork2();
if (h) _syscall_mount(h, path, strlen(path));
return h;
}
static void fs_respond_delegate(struct fs_wait_response *res, handle_t delegate, const char *og_buf) {
/* The idea behind this function is that many fs implementations (e.g. for
* overlay fs) will want to forward received requests to the original fs
* implementation.
*
* Having some special support for this in the kernel makes sense - it would
* avoid a lot of unnecessary copies. However, it wouldn't be necessary for
* a working kernel - and since I want the "base" of this system to be a small
* as possible, I want to have an alternative implementation of this in the
* userland which works in exactly the same way. The libc would then choose
* the best available implementation.
*
* (note: this wouldn't be an issue if i treated everything like functions
* in the ruby prototype)
*
* This function was supposed to avoid unnecessary copies on delegated reads.
* It is being executed /after/ the write read, though. I don't currently see
* a good way to fix that. Here are a few loose ideas:
* - a fs_wait flag which requires a separate syscall to get the write copy
* - a way to "mark" returned handles
* - no write copy mark
* - mark with delegate handle
* every vfs op would go directly to that delegate handle
* probably the fastest way
* - treat open() just like in the ruby prototype
* Instead of just returning a handle, the driver would return a list
* of functions. Here's how it would work, in pseudocode:
* ```
* # for full delegation
* respond_new_handle({
* read: {DELEGATE, handle}, // extract the read function from `handle`
* write: {DELEGATE, handle},
* close: {DELEGATE, handle}
* });
* ```
*/
static char buf[1024];
int size;
int ret;
switch (res->op) {
case VFSOP_READ:
if (_syscall_fork()) {
// handle reads in a child
// this is a HORRIBLE workaround for making concurrent IO work without proper delegates
break;
}
// TODO instead of truncating the size, allocate a bigger buffer
size = res->capacity < sizeof(buf) ? res->capacity : sizeof(buf);
ret = _syscall_read(delegate, buf, size, res->offset);
_syscall_fs_respond(buf, ret);
_syscall_exit(0); // TODO unreapable - add a nonblocking reap syscall
break;
// TODO proper writing (see above)
case VFSOP_WRITE:
ret = _syscall_write(delegate, og_buf, res->len, res->offset);
_syscall_fs_respond(NULL, ret);
break;
default:
/* unsupported / unexpected */
_syscall_fs_respond(NULL, -1);
break;
}
}
void fs_passthru(const char *prefix) {
struct fs_wait_response res;
static char buf[1024];
int ret, prefix_len;
if (prefix) prefix_len = strlen(prefix);
while (!_syscall_fs_wait(buf, sizeof(buf), &res)) {
switch (res.op) {
case VFSOP_OPEN:
if (prefix) {
/* special case: rewriting the path */
if (prefix_len + res.len <= sizeof(buf)) {
// TODO memmove
char tmp[64];
memcpy(tmp, buf, res.len);
memcpy(buf, prefix, prefix_len);
memcpy(buf + prefix_len, tmp, res.len);
ret = _syscall_open(buf, res.len + prefix_len);
} else ret = -1;
} else {
ret = _syscall_open(buf, res.len);
}
_syscall_fs_respond(NULL, ret);
break;
default:
fs_respond_delegate(&res, res.id, buf);
break;
}
}
_syscall_exit(0);
}
void fs_dir_inject(const char *path) {
struct fs_dir_handle {
int delegate;
const char *inject;
};
const int path_len = strlen(path);
struct fs_wait_response res;
struct fs_dir_handle handles[16]; // TODO hardcoded FD_MAX - use malloc instead
int handle_next = 0;
static char buf[1024];
int ret;
while (!_syscall_fs_wait(buf, sizeof buf, &res)) {
switch (res.op) {
case VFSOP_OPEN:
if (handle_next > 15) _syscall_fs_respond(NULL, -2); // we ran out of handles, which is entirely our fault.
ret = _syscall_open(buf, res.len); /* errors handled in inject handler */
handles[handle_next].delegate = ret;
handles[handle_next].inject = NULL;
if (buf[res.len - 1] == '/' &&
res.len < path_len && !memcmp(path, buf, res.len)) {
handles[handle_next].inject = path + res.len;
} else {
/* not injecting, don't allow opening nonexistent stuff */
if (ret < 0) _syscall_fs_respond(NULL, ret);
}
_syscall_fs_respond(NULL, handle_next++);
break;
case VFSOP_READ:
if (handles[res.id].inject) {
// TODO check offset
struct fs_dir_handle h = handles[res.id];
int out_len = 0;
while (h.inject[out_len] != '/')
out_len++;
memcpy(buf, h.inject, out_len);
buf[out_len++] = '/';
buf[out_len++] = '\0';
if (h.delegate >= 0) {
int to_read = res.capacity < sizeof(buf) ? res.capacity : sizeof(buf);
to_read -= out_len;
ret = _syscall_read(h.delegate, buf + out_len, to_read, 0);
if (ret > 0) out_len += ret;
// TODO deduplicate entries
}
_syscall_fs_respond(buf, out_len);
break;
}
/* fallthrough */
default: {
struct fs_dir_handle h = handles[res.id];
if (h.delegate < 0)
_syscall_fs_respond(NULL, -1);
else
fs_respond_delegate(&res, h.delegate, buf);
break;
}
}
}
_syscall_exit(0);
}
|