mirror of
https://github.com/reactos/reactos.git
synced 2025-03-10 18:24:02 +00:00
224 lines
3.6 KiB
C
224 lines
3.6 KiB
C
![]() |
/*
|
||
|
* Creation and destruction.
|
||
|
*/
|
||
|
|
||
|
#include "fitz-base.h"
|
||
|
#include "fitz-stream.h"
|
||
|
|
||
|
static fz_stream *
|
||
|
newstm(int kind, int mode)
|
||
|
{
|
||
|
fz_stream *stm;
|
||
|
|
||
|
stm = fz_malloc(sizeof(fz_stream));
|
||
|
if (!stm)
|
||
|
return nil;
|
||
|
|
||
|
stm->refs = 1;
|
||
|
stm->kind = kind;
|
||
|
stm->mode = mode;
|
||
|
stm->dead = 0;
|
||
|
stm->error = nil;
|
||
|
stm->buffer = nil;
|
||
|
|
||
|
stm->chain = nil;
|
||
|
stm->filter = nil;
|
||
|
stm->file = -1;
|
||
|
|
||
|
return stm;
|
||
|
}
|
||
|
|
||
|
fz_error *
|
||
|
fz_ioerror(fz_stream *stm)
|
||
|
{
|
||
|
fz_error *error;
|
||
|
if (stm->error)
|
||
|
{
|
||
|
error = stm->error;
|
||
|
stm->error = nil;
|
||
|
return error;
|
||
|
}
|
||
|
return fz_throw("ioerror: no error");
|
||
|
}
|
||
|
|
||
|
fz_stream *
|
||
|
fz_keepstream(fz_stream *stm)
|
||
|
{
|
||
|
stm->refs ++;
|
||
|
return stm;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
fz_dropstream(fz_stream *stm)
|
||
|
{
|
||
|
stm->refs --;
|
||
|
if (stm->refs == 0)
|
||
|
{
|
||
|
if (stm->error)
|
||
|
{
|
||
|
fz_warn("unhandled %s", stm->error->msg);
|
||
|
fz_droperror(stm->error);
|
||
|
}
|
||
|
|
||
|
if (stm->mode == FZ_SWRITE)
|
||
|
{
|
||
|
stm->buffer->eof = 1;
|
||
|
fz_flush(stm);
|
||
|
}
|
||
|
|
||
|
switch (stm->kind)
|
||
|
{
|
||
|
case FZ_SFILE:
|
||
|
close(stm->file);
|
||
|
break;
|
||
|
case FZ_SFILTER:
|
||
|
fz_dropfilter(stm->filter);
|
||
|
fz_dropstream(stm->chain);
|
||
|
break;
|
||
|
case FZ_SBUFFER:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
fz_dropbuffer(stm->buffer);
|
||
|
fz_free(stm);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static fz_error *
|
||
|
openfile(fz_stream **stmp, char *path, int mode, int realmode)
|
||
|
{
|
||
|
fz_error *error;
|
||
|
fz_stream *stm;
|
||
|
|
||
|
stm = newstm(FZ_SFILE, mode);
|
||
|
if (!stm)
|
||
|
return fz_outofmem;
|
||
|
|
||
|
error = fz_newbuffer(&stm->buffer, FZ_BUFSIZE);
|
||
|
if (error)
|
||
|
{
|
||
|
fz_free(stm);
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
stm->file = open(path, realmode, 0666);
|
||
|
if (stm->file < 0)
|
||
|
{
|
||
|
fz_dropbuffer(stm->buffer);
|
||
|
fz_free(stm);
|
||
|
return fz_throw("ioerror: open '%s' failed: %s", path, strerror(errno));
|
||
|
}
|
||
|
|
||
|
*stmp = stm;
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
static fz_error *
|
||
|
openfilter(fz_stream **stmp, fz_filter *flt, fz_stream *src, int mode)
|
||
|
{
|
||
|
fz_error *error;
|
||
|
fz_stream *stm;
|
||
|
|
||
|
stm = newstm(FZ_SFILTER, mode);
|
||
|
if (!stm)
|
||
|
return fz_outofmem;
|
||
|
|
||
|
error = fz_newbuffer(&stm->buffer, FZ_BUFSIZE);
|
||
|
if (error)
|
||
|
{
|
||
|
fz_free(stm);
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
stm->chain = fz_keepstream(src);
|
||
|
stm->filter = fz_keepfilter(flt);
|
||
|
|
||
|
*stmp = stm;
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
static fz_error *
|
||
|
openbuffer(fz_stream **stmp, fz_buffer *buf, int mode)
|
||
|
{
|
||
|
fz_stream *stm;
|
||
|
|
||
|
stm = newstm(FZ_SBUFFER, mode);
|
||
|
if (!stm)
|
||
|
return fz_outofmem;
|
||
|
|
||
|
stm->buffer = fz_keepbuffer(buf);
|
||
|
|
||
|
if (mode == FZ_SREAD)
|
||
|
stm->buffer->eof = 1;
|
||
|
|
||
|
*stmp = stm;
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openrfile(fz_stream **stmp, char *path)
|
||
|
{
|
||
|
return openfile(stmp, path, FZ_SREAD, O_BINARY | O_RDONLY);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openwfile(fz_stream **stmp, char *path)
|
||
|
{
|
||
|
return openfile(stmp, path, FZ_SWRITE,
|
||
|
O_BINARY | O_WRONLY | O_CREAT | O_TRUNC);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openafile(fz_stream **stmp, char *path)
|
||
|
{
|
||
|
fz_error *error;
|
||
|
int t;
|
||
|
|
||
|
error = openfile(stmp, path, FZ_SWRITE, O_BINARY | O_WRONLY);
|
||
|
if (error)
|
||
|
return error;
|
||
|
|
||
|
t = lseek((*stmp)->file, 0, 2);
|
||
|
if (t < 0)
|
||
|
{
|
||
|
(*stmp)->dead = 1;
|
||
|
return fz_throw("ioerror: lseek: %s", strerror(errno));
|
||
|
}
|
||
|
|
||
|
return nil;
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openrfilter(fz_stream **stmp, fz_filter *flt, fz_stream *src)
|
||
|
{
|
||
|
return openfilter(stmp, flt, src, FZ_SREAD);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openwfilter(fz_stream **stmp, fz_filter *flt, fz_stream *src)
|
||
|
{
|
||
|
return openfilter(stmp, flt, src, FZ_SWRITE);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openrbuffer(fz_stream **stmp, fz_buffer *buf)
|
||
|
{
|
||
|
return openbuffer(stmp, buf, FZ_SREAD);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openwbuffer(fz_stream **stmp, fz_buffer *buf)
|
||
|
{
|
||
|
return openbuffer(stmp, buf, FZ_SWRITE);
|
||
|
}
|
||
|
|
||
|
fz_error * fz_openrmemory(fz_stream **stmp, char *mem, int len)
|
||
|
{
|
||
|
fz_error *error;
|
||
|
fz_buffer *buf;
|
||
|
|
||
|
error = fz_newbufferwithmemory(&buf, mem, len);
|
||
|
if (error)
|
||
|
return error;
|
||
|
|
||
|
error = fz_openrbuffer(stmp, buf);
|
||
|
|
||
|
fz_dropbuffer(buf);
|
||
|
|
||
|
return error;
|
||
|
}
|
||
|
|