]> git.draconx.ca Git - liblbx.git/blob - src/fops.c
liblbx: Reset EOF flag after a successful seek on a pipe.
[liblbx.git] / src / fops.c
1 /*
2  *  2ooM: The Master of Orion II Reverse Engineering Project
3  *  Default file operations structures for liblbx.
4  *  Copyright (C) 2010 Nick Bowler
5  *
6  *  This program is free software: you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation, either version 3 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19 #include <stdio.h>
20
21 #include "misc.h"
22 #include "lbx.h"
23
24 /* Default I/O operations for ordinary files. */
25 static size_t file_read(void *buf, size_t size, void *handle)
26 {
27         return fread(buf, 1, size, (FILE *)handle);
28 }
29
30 static int file_seek(void *handle, long offset, int whence)
31 {
32         return fseek((FILE *)handle, offset, whence);
33 }
34
35 static long file_tell(void *handle)
36 {
37         return ftell((FILE *)handle);
38 }
39
40 static int file_eof(void *handle)
41 {
42         return feof((FILE *)handle);
43 }
44
45 const struct lbx_file_ops lbx_default_fops = {
46         .read = file_read,
47         .seek = file_seek,
48         .tell = file_tell,
49         .eof  = file_eof,
50 };
51
52 /* Read function for pipes that tracks the number of bytes read. */
53 static size_t pipe_read(void *buf, size_t size, void *handle)
54 {
55         struct lbx_pipe_state *state = handle;
56         size_t rc;
57
58         rc = fread(buf, 1, size, state->f);
59         state->offset += rc;
60         return rc;
61 }
62
63 /* Seek function for pipes that reads data into the void. */
64 static int pipe_seek(void *handle, long offset, int whence)
65 {
66         struct lbx_pipe_state *state = handle;
67         long distance;
68
69         switch (whence) {
70         case SEEK_SET:
71                 distance = offset - state->offset;
72                 break;
73         case SEEK_CUR:
74                 distance = offset;
75                 break;
76         case SEEK_END:
77                 distance = -1;
78                 break;
79         }
80
81         if (distance < 0)
82                 return -1;
83
84         while (distance > 0) {
85                 static unsigned char oblivion[1024];
86                 size_t rc, amount = MIN(sizeof oblivion, distance);
87
88                 rc = pipe_read(oblivion, amount, handle);
89                 distance -= rc;
90
91                 if (rc < amount)
92                         return -1;
93         }
94
95         clearerr(state->f);
96         return 0;
97 }
98
99 static long pipe_tell(void *handle)
100 {
101         struct lbx_pipe_state *state = handle;
102
103         return state->offset;
104 }
105
106 static int pipe_eof(void *handle)
107 {
108         struct lbx_pipe_state *state = handle;
109
110         return feof(state->f);
111 }
112
113 const struct lbx_file_ops lbx_pipe_fops = {
114         .read = pipe_read,
115         .seek = pipe_seek,
116         .tell = pipe_tell,
117         .eof  = pipe_eof,
118 };
119
120 static size_t lbx_read(void *buf, size_t size, void *handle)
121 {
122         return lbx_file_read(handle, buf, size);
123 }
124
125 static int lbx_seek(void *handle, long offset, int whence)
126 {
127         return lbx_file_seek(handle, offset, whence);
128 }
129
130 static long lbx_tell(void *handle)
131 {
132         return lbx_file_tell(handle);
133 }
134
135 static int lbx_eof(void *handle)
136 {
137         return lbx_file_eof(handle);
138 }
139
140 const struct lbx_file_ops lbx_arch_fops = {
141         .read = lbx_read,
142         .seek = lbx_seek,
143         .tell = lbx_tell,
144         .eof  = lbx_eof,
145 };