blob: 44901eae22d862caf59f13e453c3b2c584663c28 (
plain)
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
|
#include "handle.h"
#include <assert.h>
#include <fcntl.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <sys/random.h>
#include <unistd.h>
struct handle {
int fd;
char *key;
char *pathname;
struct handle *next;
struct handle *prev;
};
struct handle *head = NULL;
char *generate_random_key(int length) {
assert(length < 256);
char buf[256];
randomfill(buf, 256);
// assert(0 == getentropy(buf, 256));
char *r = malloc(length + 1);
if (!r) {
return NULL;
}
for (int i = 0; i < length; i++) {
r[i] = 'A' + (buf[i] & 0xF);
}
r[length] = '\0';
return r;
}
static struct handle *handle_get(struct sv key) {
struct handle *p = head;
for (; p; p = p->next) {
if (!sv_eq(C_TO_SV(p->key), key)) {
continue;
}
return p;
}
return NULL;
}
int handle_get_fd(struct sv key, int *fd, char **pathname) {
struct handle *h = handle_get(key);
if (!h) {
return 0;
}
if (fd) {
*fd = h->fd;
}
if (pathname) {
*pathname = h->pathname;
}
return 1;
}
char *handle_create(struct sv path, int is_dir, int flags, int mode) {
int f;
if (is_dir) {
// f = O_DIRECTORY | O_RDONLY;
f = O_RDONLY;
} else {
f = flags;
}
char *p = SV_TO_C(path);
int fd = open(p, f, mode);
if (-1 == fd) {
free(p);
return NULL;
}
free(p);
// 64 bits of entropy
char *key = generate_random_key(16);
if (!key) {
close(fd);
return NULL;
}
struct handle *n = malloc(sizeof(struct handle));
if (!n) {
free(key);
close(fd);
return NULL;
}
n->fd = fd;
n->key = key;
n->pathname = SV_TO_C(path);
n->next = head;
n->prev = NULL;
if (head) {
head->prev = n;
}
head = n;
return key;
}
int handle_close(struct sv key) {
struct handle *h = handle_get(key);
if (!h) {
return 0;
}
if (head == h) {
head = h->next;
}
if (h->prev) {
h->prev->next = h->next;
}
if (h->next) {
h->next->prev = h->prev;
}
free(h->key);
free(h->pathname);
free(h);
return 1;
}
|