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
|
#include <init/stdlib.h>
#include <shared/flags.h>
#include <shared/syscalls.h>
#include <stdint.h>
#define BUF_SIZE 64
static int tar_open(const char *path, int len, void *base, size_t base_len);
static int tar_read(struct fs_wait_response *res, void *base, size_t base_len);
static int tar_size(void *sector);
static void *tar_find(const char *path, size_t path_len, void *base, size_t base_len);
static int oct_parse(char *str, size_t len);
void tar_driver(void *base) {
static char buf[BUF_SIZE];
struct fs_wait_response res;
for (;;) {
switch (_syscall_fs_wait(buf, BUF_SIZE, &res)) {
case VFSOP_OPEN:
_syscall_fs_respond(NULL, tar_open(buf, res.len, base, ~0));
break;
case VFSOP_READ:
tar_read(&res, base, ~0);
break;
default:
_syscall_fs_respond(NULL, -1); // unsupported
break;
}
}
}
static int tar_open(const char *path, int len, void *base, size_t base_len) {
void *ptr;
if (len <= 1) return -1;
path += 1; // skip the leading slash
len -= 1;
ptr = tar_find(path, len, base, ~0);
if (!ptr) return -1;
return (int)ptr;
}
static int tar_read(struct fs_wait_response *res, void *base, size_t base_len) {
void *meta = (void*)res->id;
char type = *(char*)(meta + 156);
size_t meta_len;
int size;
static char buf[BUF_SIZE]; // TODO reuse a single buffer
size_t buf_pos = 0;
switch (type) {
case '\0':
case '0': /* normal files */
size = tar_size(meta);
if (res->offset < 0 || res->offset > size) {
// TODO support negative offsets
_syscall_fs_respond(NULL, -1);
} else {
_syscall_fs_respond(meta + 512 + res->offset, size - res->offset);
}
break;
case '5': /* directory */
meta_len = strlen(meta);
/* find files in dir */
for (size_t off = 0; off < base_len;) {
if (0 != memcmp(base + off + 257, "ustar", 5))
break; // not a metadata sector
// TODO more meaningful variable names and clean code up
if (0 == memcmp(base + off, meta, meta_len) &&
*(char*)(base + off + meta_len) != '\0') {
char *suffix = base + off + meta_len;
size_t suffix_len = strlen(suffix);
memcpy(buf + buf_pos, suffix, suffix_len);
buf[buf_pos + suffix_len] = '\0';
buf_pos += suffix_len + 1;
// TODO no buffer overrun check
// TODO don't list files in subdirectories
}
size = tar_size(base + off);
off += 512; // skip this metadata sector
off += (size + 511) & ~511; // skip the data sectors
}
_syscall_fs_respond(buf, buf_pos);
break;
default:
_syscall_fs_respond(NULL, -1);
break;
}
}
static int tar_size(void *sector) {
return oct_parse(sector + 124, 11);
}
static void *tar_find(const char *path, size_t path_len, void *base, size_t base_len) {
int size;
if (path_len > 100) return NULL; // illegal path
for (size_t off = 0; off < base_len;) {
if (0 != memcmp(base + off + 257, "ustar", 5))
break; // not a metadata sector
if (0 == memcmp(base + off, path, path_len) &&
*(char*)(base + off + path_len) == '\0')
return base + off; // file found, quit
size = tar_size(base + off);
off += 512; // skip this metadata sector
off += (size + 511) & ~511; // skip the data sectors
}
return NULL;
}
static int oct_parse(char *str, size_t len) {
int res = 0;
for (size_t i = 0; i < len; i++) {
res *= 8;
res += str[i] - '0'; // no format checking
}
return res;
}
|