13 #include <sys/types.h>
\r
21 static const bool UseDebug = false;
\r
22 static const bool UseCR = false;
\r
24 static const int StringSize = 4096;
\r
26 static const char LF = '\n';
\r
27 static const char CR = '\r';
\r
31 static int my_read (int fd, char string[], int size);
\r
32 static void my_write (int fd, const char string[], int size);
\r
38 bool io_is_ok(const io_t * io) {
\r
40 if (io == NULL) return false;
\r
42 if (io->name == NULL) return false;
\r
44 if (io->in_eof != true && io->in_eof != false) return false;
\r
46 if (io->in_size < 0 || io->in_size > BufferSize) return false;
\r
47 if (io->out_size < 0 || io->out_size > BufferSize) return false;
\r
54 void io_init(io_t * io) {
\r
63 ASSERT(io_is_ok(io));
\r
68 void io_close(io_t * io) {
\r
70 ASSERT(io_is_ok(io));
\r
72 ASSERT(io->out_fd>=0);
\r
74 my_log("> %s EOF\n",io->name);
\r
76 if (close(io->out_fd) == -1) {
\r
77 my_fatal("io_close(): close(): %s\n",strerror(errno));
\r
85 void io_get_update(io_t * io) {
\r
90 ASSERT(io_is_ok(io));
\r
92 ASSERT(io->in_fd>=0);
\r
93 ASSERT(!io->in_eof);
\r
99 size = BufferSize - pos;
\r
100 if (size <= 0) my_fatal("io_get_update(): buffer overflow\n");
\r
102 // read as many data as possible
\r
103 n = my_read(io->in_fd,&io->in_buffer[pos],size);
\r
104 if (UseDebug) my_log("POLYGLOT read %d byte%s from %s\n",n,(n>1)?"s":"",io->name);
\r
106 if (n > 0) { // at least one character was read
\r
108 // update buffer size
\r
110 ASSERT(n>=1&&n<=size);
\r
113 ASSERT(io->in_size>=0&&io->in_size<=BufferSize);
\r
125 bool io_line_ready(const io_t * io) {
\r
127 ASSERT(io_is_ok(io));
\r
129 if (io->in_eof) return true;
\r
131 if (memchr(io->in_buffer,LF,io->in_size) != NULL) return true; // buffer contains LF
\r
138 bool io_get_line(io_t * io, char string[], int size) {
\r
143 ASSERT(io_is_ok(io));
\r
144 ASSERT(string!=NULL);
\r
152 // test for end of buffer
\r
154 if (src >= io->in_size) {
\r
156 my_log("%s->Adapter: EOF\n",io->name);
\r
159 my_fatal("io_get_line(): no EOL in buffer\n");
\r
163 // test for end of string
\r
165 if (dst >= size) my_fatal("io_get_line(): buffer overflow\n");
\r
167 // copy the next character
\r
169 c = io->in_buffer[src++];
\r
171 if (c == LF) { // LF => line complete
\r
172 string[dst] = '\0';
\r
174 } else if (c != CR) { // skip CRs
\r
179 // shift the buffer
\r
183 io->in_size -= src;
\r
184 ASSERT(io->in_size>=0);
\r
186 if (io->in_size > 0) memmove(&io->in_buffer[0],&io->in_buffer[src],io->in_size);
\r
190 my_log("%s->Adapter: %s\n",io->name,string);
\r
197 void io_send(io_t * io, const char format[], ...) {
\r
200 char string[StringSize];
\r
203 ASSERT(io_is_ok(io));
\r
204 ASSERT(format!=NULL);
\r
206 ASSERT(io->out_fd>=0);
\r
210 va_start(arg_list,format);
\r
211 vsprintf(string,format,arg_list);
\r
214 // append string to buffer
\r
216 len = strlen(string);
\r
217 if (io->out_size + len > BufferSize-2) my_fatal("io_send(): buffer overflow\n");
\r
219 memcpy(&io->out_buffer[io->out_size],string,len);
\r
220 io->out_size += len;
\r
222 ASSERT(io->out_size>=0&&io->out_size<=BufferSize-2);
\r
226 io->out_buffer[io->out_size] = '\0';
\r
227 my_log("Adapter->%s: %s\n",io->name,io->out_buffer);
\r
228 // my_log("> %f %s %s\n",now_real(),io->name,io->out_buffer);
\r
229 // append EOL to buffer
\r
231 if (UseCR) io->out_buffer[io->out_size++] = CR;
\r
232 io->out_buffer[io->out_size++] = LF;
\r
234 ASSERT(io->out_size>=0&&io->out_size<=BufferSize);
\r
238 if (UseDebug) my_log("POLYGLOT writing %d byte%s to %s\n",io->out_size,(io->out_size>1)?"s":"",io->name);
\r
239 my_write(io->out_fd,io->out_buffer,io->out_size);
\r
246 void io_send_queue(io_t * io, const char format[], ...) {
\r
249 char string[StringSize];
\r
252 ASSERT(io_is_ok(io));
\r
253 ASSERT(format!=NULL);
\r
255 ASSERT(io->out_fd>=0);
\r
259 va_start(arg_list,format);
\r
260 vsprintf(string,format,arg_list);
\r
263 // append string to buffer
\r
265 len = strlen(string);
\r
266 if (io->out_size + len > BufferSize-2) my_fatal("io_send_queue(): buffer overflow\n");
\r
268 memcpy(&io->out_buffer[io->out_size],string,len);
\r
269 io->out_size += len;
\r
271 ASSERT(io->out_size>=0&&io->out_size<=BufferSize-2);
\r
276 static int my_read(int fd, char string[], int size) {
\r
281 ASSERT(string!=NULL);
\r
285 n = read(fd,string,size);
\r
286 } while (n == -1 && errno == EINTR);
\r
288 if (n == -1) my_fatal("my_read(): read(): %s\n",strerror(errno));
\r
297 static void my_write(int fd, const char string[], int size) {
\r
302 ASSERT(string!=NULL);
\r
307 n = write(fd,string,size);
\r
309 // if (n == -1 && errno != EINTR && errno != EPIPE) my_fatal("my_write(): write(): %s\n",strerror(errno));
\r
313 } else if (errno == EINTR) {
\r
314 n = 0; // nothing has been written
\r
315 } else if (errno == EPIPE) {
\r
316 n = size; // pretend everything has been written
\r
318 my_fatal("my_write(): write(): %s\n",strerror(errno));
\r
327 } while (size > 0);
\r