span table
[visor] / libvisor / src / visor.c
1 #include "vilibc.h"
2 #include "visor.h"
3 #include "vimpl.h"
4
5 #define vi_malloc       vi->mm.malloc
6 #define vi_free         vi->mm.free
7 #define vi_realloc      vi->mm.realloc
8
9 #define vi_open         vi->fop.open
10 #define vi_size         vi->fop.size
11 #define vi_close        vi->fop.close
12 #define vi_map          vi->fop.map
13 #define vi_unmap        vi->fop.unmap
14 #define vi_read         vi->fop.read
15 #define vi_write        vi->fop.write
16 #define vi_seek         vi->fop.seek
17
18 #ifdef HAVE_LIBC
19 static const struct vi_alloc stdalloc = { malloc, free, realloc };
20 #endif
21
22 struct visor *vi_create(struct vi_alloc *mm)
23 {
24         struct visor *vi;
25
26 #ifdef HAVE_LIBC
27         if(!mm) mm = &stdalloc;
28 #else
29         if(!mm) return 0;
30 #endif
31
32         if(!(vi = mm->malloc(sizeof *vi))) {
33                 return 0;
34         }
35         memset(vi, 0, sizeof *vi);
36         vi->mm = *mm;
37
38         return vi;
39 }
40
41 void vi_destroy(struct visor *vi)
42 {
43         while(vi->buflist) {
44                 vi_delete_buf(vi, vi->buflist);
45         }
46         vi_free(vi);
47 }
48
49 void vi_set_fileops(struct visor *vi, struct vi_fileops *fop)
50 {
51         vi->fop = *fop;
52 }
53
54 void vi_set_ttyops(struct visor *vi, struct vi_ttyops *tty)
55 {
56         vi->tty = *tty;
57 }
58
59 struct vi_buffer *vi_new_buf(struct visor *vi, const char *path)
60 {
61         struct vi_buffer *nb;
62
63         if(!(nb = vi_malloc(sizeof *nb))) {
64                 vi_error(vi, "failed to allocate new buffer\n");
65                 return 0;
66         }
67         memset(nb, 0, sizeof *nb);
68         nb->vi = vi;
69
70         if(path) {
71                 if(vi_buf_read(nb, path) == -1) {
72                         vi_free(nb);
73                         return 0;
74                 }
75         }
76
77         if(vi->buflist) {
78                 struct vi_buffer *last = vi->buflist->prev;
79                 nb->prev = last;
80                 nb->next = vi->buflist;
81                 last->next = nb;
82                 vi->buflist->prev = nb;
83         } else {
84                 nb->next = nb->prev = nb;
85                 vi->buflist = nb;
86         }
87         return nb;
88 }
89
90 static int remove_buf(struct visor *vi, struct vi_buffer *vb)
91 {
92         if(!vi->buflist) {
93                 vi_error(vi, "failed to remove a buffer which doesn't exist\n");
94                 return -1;
95         }
96
97         if(vb->next == vb) {
98                 if(vi->buflist != vb) {
99                         vi_error(vi, "failed to remove buffer, buffer list inconsistency\n");
100                         return -1;
101                 }
102                 vi->buflist = 0;
103                 return 0;
104         }
105
106         if(vi->buflist == vb) {
107                 vi->buflist = vb->next;
108         }
109         vb->prev->next = vb->next;
110         vb->next->prev = vb->prev;
111         vb->next = vb->prev = vb;
112         return 0;
113 }
114
115 int vi_delete_buf(struct visor *vi, struct vi_buffer *vb)
116 {
117         if(remove_buf(vi, vb) == -1) {
118                 return -1;
119         }
120
121         vi_free(vb->path);
122         vi_free(vb->orig);
123         vi_free(vb->add);
124         vi_free(vb->spans);
125         return 0;
126 }
127
128 int vi_num_buf(struct visor *vi)
129 {
130         int count;
131         struct vi_buffer *vb;
132
133         if(!vi->buflist) return 0;
134
135         count = 1;
136         vb = vi->buflist->next;
137         while(vb != vi->buflist) {
138                 count++;
139                 vb = vb->next;
140         }
141         return count;
142 }
143
144 struct vi_buffer *vi_getcur_buf(struct visor *vi)
145 {
146         return vi->buflist;
147 }
148
149 void vi_setcur_buf(struct visor *vi, struct vi_buffer *vb)
150 {
151         vi->buflist = vb;
152 }
153
154 struct vi_buffer *vi_next_buf(struct visor *vi)
155 {
156         return vi->buflist ? vi->buflist->next : 0;
157 }
158
159 struct vi_buffer *vi_prev_buf(struct visor *vi)
160 {
161         return vi->buflist ? vi->buflist->prev : 0;
162 }
163
164 static int add_span(struct vi_buffer *vb, int src, vi_addr start, unsigned long size)
165 {
166         struct visor *vi = vb->vi;
167         struct vi_span *sp;
168
169         if(vb->num_spans >= vb->max_spans) {
170                 int newmax = vb->max_spans > 0 ? (vb->max_spans << 1) : 16;
171                 struct vi_span *tmp = vi_realloc(vb->spans, newmax * sizeof *tmp);
172                 if(!tmp) return -1;
173                 vb->spans = tmp;
174                 vb->max_spans = newmax;
175         }
176
177         sp = vb->spans + vb->num_spans++;
178         sp->beg = start;
179         sp->size = size;
180         return 0;
181 }
182
183 void vi_buf_reset(struct vi_buffer *vb)
184 {
185         struct visor *vi = vb->vi;
186         struct vi_buffer *prev, *next;
187
188         vi_free(vb->path);
189
190         if(vb->fp) {
191                 if(vb->file_mapped) vi_unmap(vb->fp);
192                 vi_close(vb->fp);
193         }
194         vi_free(vb->orig);
195         vi_free(vb->add);
196         vi_free(vb->spans);
197
198         prev = vb->prev;
199         next = vb->next;
200         memset(vb, 0, sizeof *vb);
201         vb->prev = prev;
202         vb->next = next;
203 }
204
205 int vi_buf_read(struct vi_buffer *vb, const char *path)
206 {
207         struct visor *vi = vb->vi;
208         vi_file *fp;
209         unsigned long fsz;
210         int plen;
211
212         vi_buf_reset(vb);
213
214         if(!(fp = vi_open(path))) {
215                 return -1;
216         }
217         plen = strlen(path);
218         if(!(vb->path = vi_malloc(plen + 1))) {
219                 vi_error(vi, "failed to allocate path name buffer\n");
220                 vi_buf_reset(vb);
221                 return -1;
222         }
223         memcpy(vb->path, path, plen + 1);
224
225         vb->num_spans = 0;
226
227         if((fsz = vi_size(fp))) {
228                 /* existing file, map it into memory, or failing that read it */
229                 if(!vi->fop.map || !(vb->orig = vi_map(fp))) {
230                         if(!(vb->orig = vi_malloc(fsz))) {
231                                 vi_buf_reset(vb);
232                                 return -1;
233                         }
234                 } else {
235                         vb->file_mapped = 1;
236                 }
237
238                 if(add_span(vb, SPAN_ORIG, 0, fsz) == -1) {
239                         vi_error(vi, "failed to allocate span\n");
240                         vi_buf_reset(vb);
241                         return -1;
242                 }
243         }
244         vb->orig_size = fsz;
245         return 0;
246 }