diff options
Diffstat (limited to 'fitz/stm_write.c')
-rw-r--r-- | fitz/stm_write.c | 313 |
1 files changed, 313 insertions, 0 deletions
diff --git a/fitz/stm_write.c b/fitz/stm_write.c new file mode 100644 index 00000000..ed928b50 --- /dev/null +++ b/fitz/stm_write.c @@ -0,0 +1,313 @@ +/* + * Output streams. + */ + +#include "fitz-base.h" +#include "fitz-stream.h" + +int +fz_wtell(fz_stream *stm) +{ + fz_buffer *buf = stm->buffer; + int t; + + if (stm->dead) + return EOF; + + if (stm->mode != FZ_SWRITE) + return EOF; + + switch (stm->kind) + { + case FZ_SFILE: + t = lseek(stm->file, 0, 1); + if (t < 0) + { + fz_warn("syserr: lseek: %s", strerror(errno)); + stm->dead = 1; + return EOF; + } + return t + (buf->wp - buf->rp); + + case FZ_SFILTER: + return stm->filter->count + (buf->wp - buf->rp); + + case FZ_SBUFFER: + return buf->wp - buf->bp; + + default: + return EOF; + } +} + +fz_error * +fz_wseek(fz_stream *stm, int offset, int whence) +{ + fz_buffer *buf = stm->buffer; + int t; + + if (stm->dead) + return fz_throw("assert: seek in dead stream"); + + if (stm->mode != FZ_SWRITE) + return fz_throw("assert: write operation on reading stream"); + + if (stm->kind != FZ_SFILE) + return fz_throw("assert: write seek on non-file stream"); + + t = lseek(stm->file, offset, whence); + if (t < 0) + { + stm->dead = 1; + return fz_throw("syserr: lseek: %s", strerror(errno)); + } + + buf->rp = buf->bp; + buf->wp = buf->bp; + buf->eof = 0; + + return fz_okay; +} + +static fz_error * +fz_flushfilterimp(fz_stream *stm) +{ + fz_buffer *buf = stm->buffer; + fz_error *error; + fz_error *reason; + +loop: + + reason = fz_process(stm->filter, stm->buffer, stm->chain->buffer); + + if (reason == fz_ioneedin) + { + if (buf->rp > buf->bp) + { + error = fz_rewindbuffer(buf); + if (error) + { + stm->dead = 1; + return fz_rethrow(error, "cannot rewind buffer"); + } + } + else + { + error = fz_growbuffer(buf); + if (error) + { + stm->dead = 1; + return fz_rethrow(error, "cannot grow buffer"); + } + } + } + + else if (reason == fz_ioneedout) + { + error = fz_flush(stm->chain); + if (error) + return fz_rethrow(error, "cannot flush chain buffer"); + } + + else if (reason == fz_iodone) + { + stm->dead = 2; /* special flag that we are dead because of eod */ + } + + else + { + stm->dead = 1; + return fz_rethrow(reason, "cannot process filter"); + } + + /* if we are at eof, repeat until other filter sets otherside to eof */ + if (buf->eof && !stm->chain->buffer->eof) + goto loop; + + return fz_okay; +} + +/* + * Empty the buffer into the sink. + * Promise to make more space available. + * Called by fz_write and fz_dropstream. + * If buffer is eof, then all data must be flushed. + */ +fz_error * +fz_flush(fz_stream *stm) +{ + fz_buffer *buf = stm->buffer; + fz_error *error; + int t; + + if (stm->dead == 2) /* eod flag */ + return fz_okay; + + if (stm->dead) + return fz_throw("assert: flush on dead stream"); + + if (stm->mode != FZ_SWRITE) + return fz_throw("assert: write operation on reading stream"); + + switch (stm->kind) + { + case FZ_SFILE: + while (buf->rp < buf->wp) + { + t = write(stm->file, buf->rp, buf->wp - buf->rp); + if (t < 0) + { + stm->dead = 1; + return fz_throw("syserr: write: %s", strerror(errno)); + } + + buf->rp += t; + } + + if (buf->rp > buf->bp) + { + error = fz_rewindbuffer(buf); + if (error) + { + stm->dead = 1; + return fz_rethrow(error, "cannot rewind buffer"); + } + } + + return fz_okay; + + case FZ_SFILTER: + error = fz_flushfilterimp(stm); + if (error) + return fz_rethrow(error, "cannot flush through filter"); + return fz_okay; + + case FZ_SBUFFER: + if (!buf->eof && buf->wp == buf->ep) + { + error = fz_growbuffer(buf); + if (error) + { + stm->dead = 1; + return fz_rethrow(error, "cannot grow buffer"); + } + } + return fz_okay; + + default: + return fz_throw("unknown stream type"); + } +} + +/* + * Write data to stream. + * Buffer until internal buffer is full. + * When full, call fz_flush to make more space available. + * Return error if all the data could not be written. + */ +fz_error * +fz_write(fz_stream *stm, unsigned char *mem, int n) +{ + fz_buffer *buf = stm->buffer; + fz_error *error; + int i = 0; + + if (stm->dead) + return fz_throw("assert: write on dead stream"); + + if (stm->mode != FZ_SWRITE) + return fz_throw("assert: write on reading stream"); + + while (i < n) + { + while (buf->wp < buf->ep && i < n) + *buf->wp++ = mem[i++]; + + if (buf->wp == buf->ep && i < n) + { + error = fz_flush(stm); + if (error) + return fz_rethrow(error, "cannot flush buffer"); + if (stm->dead) + return fz_throw("assert: write on dead stream"); + } + } + + return fz_okay; +} + +fz_error * +fz_printstr(fz_stream *stm, char *s) +{ + return fz_write(stm, (unsigned char *) s, strlen(s)); +} + +fz_error * +fz_printobj(fz_stream *file, fz_obj *obj, int tight) +{ + fz_error *error; + unsigned char buf[1024]; + char *ptr; + int n; + + n = fz_sprintobj(nil, 0, obj, tight); + if (n < sizeof buf) + { + fz_sprintobj(buf, sizeof buf, obj, tight); + error = fz_write(file, (unsigned char *) buf, n); + if (error) + return fz_rethrow(error, "cannot write buffer"); + return fz_okay; + } + else + { + ptr = fz_malloc(n); + if (!ptr) + return fz_throw("outofmem: scratch buffer"); + fz_sprintobj(ptr, n, obj, tight); + error = fz_write(file, (unsigned char *) ptr, n); + fz_free(ptr); + if (error) + return fz_rethrow(error, "cannot write buffer"); + return fz_okay; + } +} + +fz_error * +fz_print(fz_stream *stm, char *fmt, ...) +{ + fz_error *error; + va_list ap; + char buf[1024]; + char *p; + int n; + + va_start(ap, fmt); + n = vsnprintf(buf, sizeof buf, fmt, ap); + va_end(ap); + + if (n < sizeof buf) + { + error = fz_write(stm, (unsigned char *) buf, n); + if (error) + return fz_rethrow(error, "cannot write buffer"); + return fz_okay; + } + + p = fz_malloc(n); + if (!p) + return fz_throw("outofmem: scratch buffer"); + + va_start(ap, fmt); + vsnprintf(p, n, fmt, ap); + va_end(ap); + + error = fz_write(stm, (unsigned char *) p, n); + + fz_free(p); + + if (error) + return fz_rethrow(error, "cannot write buffer"); + return fz_okay; +} + |