mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-26 17:24:31 +03:00
Use file descriptor instead of chunks
This commit is contained in:
parent
c5565fc7f0
commit
4b755d43ec
|
@ -21,6 +21,7 @@ prune docs/_static
|
||||||
|
|
||||||
# build/src control detritus
|
# build/src control detritus
|
||||||
exclude .coveragerc
|
exclude .coveragerc
|
||||||
|
exclude codecov.yml
|
||||||
exclude .editorconfig
|
exclude .editorconfig
|
||||||
exclude .landscape.yaml
|
exclude .landscape.yaml
|
||||||
exclude .travis
|
exclude .travis
|
||||||
|
|
6
decode.c
6
decode.c
|
@ -695,15 +695,11 @@ PyImaging_SgiRleDecoderNew(PyObject* self, PyObject* args)
|
||||||
if (get_unpacker(decoder, mode, rawmode) < 0)
|
if (get_unpacker(decoder, mode, rawmode) < 0)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
decoder->pulls_fd = 1;
|
||||||
decoder->decode = ImagingSgiRleDecode;
|
decoder->decode = ImagingSgiRleDecode;
|
||||||
decoder->cleanup = ImagingSgiRleDecodeCleanup;
|
|
||||||
decoder->state.ystep = ystep;
|
decoder->state.ystep = ystep;
|
||||||
|
|
||||||
((SGISTATE*)decoder->state.context)->bpc = bpc;
|
((SGISTATE*)decoder->state.context)->bpc = bpc;
|
||||||
((SGISTATE*)decoder->state.context)->rowno = 0;
|
|
||||||
((SGISTATE*)decoder->state.context)->channo = 0;
|
|
||||||
((SGISTATE*)decoder->state.context)->starttabidx = 0;
|
|
||||||
((SGISTATE*)decoder->state.context)->lengthtabidx = 0;
|
|
||||||
|
|
||||||
return (PyObject*) decoder;
|
return (PyObject*) decoder;
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,28 +4,7 @@ typedef struct {
|
||||||
|
|
||||||
/* CONFIGURATION */
|
/* CONFIGURATION */
|
||||||
|
|
||||||
/* Number of bytes per pixel per channel */
|
/* Number of bytes per channel per pixel */
|
||||||
int bpc;
|
int bpc;
|
||||||
|
|
||||||
/* Number of UINT32 data in RLE tables */
|
|
||||||
int tablen;
|
|
||||||
|
|
||||||
/* Current row index */
|
|
||||||
int rowno;
|
|
||||||
|
|
||||||
/* Current channel index */
|
|
||||||
int channo;
|
|
||||||
|
|
||||||
/* Offsets table */
|
|
||||||
uint32_t* starttab;
|
|
||||||
|
|
||||||
/* Lengths table */
|
|
||||||
uint32_t* lengthtab;
|
|
||||||
|
|
||||||
/* Offsets table index */
|
|
||||||
int starttabidx;
|
|
||||||
|
|
||||||
/* Lengths table index */
|
|
||||||
int lengthtabidx;
|
|
||||||
|
|
||||||
} SGISTATE;
|
} SGISTATE;
|
|
@ -5,6 +5,7 @@
|
||||||
* decoder for Sgi RLE data.
|
* decoder for Sgi RLE data.
|
||||||
*
|
*
|
||||||
* history:
|
* history:
|
||||||
|
* 2017-07-28 mb fixed for images larger than 64KB
|
||||||
* 2017-07-20 mb created
|
* 2017-07-20 mb created
|
||||||
*
|
*
|
||||||
* Copyright (c) Mickael Bonfill 2017.
|
* Copyright (c) Mickael Bonfill 2017.
|
||||||
|
@ -28,7 +29,7 @@ static void read4B(uint32_t* dest, UINT8* buf)
|
||||||
*dest = (uint32_t)((buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3]);
|
*dest = (uint32_t)((buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int expandrow(UINT8* dest, UINT8* src, int n)
|
static int expandrow(UINT8* dest, UINT8* src, int n, int z)
|
||||||
{
|
{
|
||||||
UINT8 pixel, count;
|
UINT8 pixel, count;
|
||||||
|
|
||||||
|
@ -42,14 +43,16 @@ static int expandrow(UINT8* dest, UINT8* src, int n)
|
||||||
return count;
|
return count;
|
||||||
if (pixel & RLE_COPY_FLAG) {
|
if (pixel & RLE_COPY_FLAG) {
|
||||||
while(count--) {
|
while(count--) {
|
||||||
*dest++ = *src++;
|
*dest = *src++;
|
||||||
|
dest += z;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
pixel = *src++;
|
pixel = *src++;
|
||||||
while (count--) {
|
while (count--) {
|
||||||
*dest++ = pixel;
|
*dest = pixel;
|
||||||
|
dest += z;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,13 +60,13 @@ static int expandrow(UINT8* dest, UINT8* src, int n)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int expandrow2(UINT16* dest, UINT16* src, int n)
|
static int expandrow2(UINT16* dest, UINT16* src, int n, int z)
|
||||||
{
|
{
|
||||||
UINT8 pixel, count;
|
UINT8 pixel, count;
|
||||||
|
|
||||||
for (;n > 0; n--)
|
for (;n > 0; n--)
|
||||||
{
|
{
|
||||||
pixel = ((UINT8*)src)[1];
|
pixel = ((UINT8*)src)[0];
|
||||||
++src;
|
++src;
|
||||||
if (n == 1 && pixel != 0)
|
if (n == 1 && pixel != 0)
|
||||||
return n;
|
return n;
|
||||||
|
@ -72,12 +75,14 @@ static int expandrow2(UINT16* dest, UINT16* src, int n)
|
||||||
return count;
|
return count;
|
||||||
if (pixel & RLE_COPY_FLAG) {
|
if (pixel & RLE_COPY_FLAG) {
|
||||||
while(count--) {
|
while(count--) {
|
||||||
*dest++ = *src++;
|
*dest = *src++;
|
||||||
|
dest += z;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
while (count--) {
|
while (count--) {
|
||||||
*dest++ = *src;
|
*dest = *src;
|
||||||
|
dest += z;
|
||||||
}
|
}
|
||||||
++src;
|
++src;
|
||||||
}
|
}
|
||||||
|
@ -85,121 +90,76 @@ static int expandrow2(UINT16* dest, UINT16* src, int n)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
ImagingSgiRleDecode(Imaging im, ImagingCodecState state,
|
ImagingSgiRleDecode(Imaging im, ImagingCodecState state,
|
||||||
UINT8* buf, int bytes)
|
UINT8* buf, int bytes)
|
||||||
{
|
{
|
||||||
SGISTATE *context;
|
uint32_t *starttab, *lengthtab, rleoffset, rlelength;
|
||||||
|
int tablen, i, j, rowno, channo, bpc;
|
||||||
|
long bufsize;
|
||||||
UINT8 *ptr;
|
UINT8 *ptr;
|
||||||
uint32_t rleoffset, rlelength;
|
SGISTATE *context;
|
||||||
|
|
||||||
context = (SGISTATE*)state->context;
|
context = (SGISTATE*)state->context;
|
||||||
// oldcount = context->bytescount;
|
_imaging_seek_pyFd(state->fd, 0L, SEEK_END);
|
||||||
|
bufsize = _imaging_tell_pyFd(state->fd);
|
||||||
|
bufsize -= SGI_HEADER_SIZE;
|
||||||
|
ptr = malloc(sizeof(UINT8) * bufsize);
|
||||||
|
state->count = 0;
|
||||||
|
_imaging_seek_pyFd(state->fd, SGI_HEADER_SIZE, SEEK_SET);
|
||||||
|
_imaging_read_pyFd(state->fd, ptr, bufsize);
|
||||||
|
|
||||||
ptr = buf;
|
|
||||||
|
|
||||||
switch (state->state)
|
state->y = 0;
|
||||||
{
|
|
||||||
case 0:
|
|
||||||
/* decoder initialization */
|
|
||||||
if (state->ystep < 0)
|
if (state->ystep < 0)
|
||||||
state->y = im->ysize - 1;
|
state->y = im->ysize - 1;
|
||||||
else
|
else
|
||||||
state->ystep = 1;
|
state->ystep = 1;
|
||||||
|
|
||||||
context->tablen = im->ysize * im->bands;
|
tablen = im->bands * im->ysize;
|
||||||
context->starttab = calloc(context->tablen, sizeof(uint32_t));
|
starttab = calloc(tablen, sizeof(uint32_t));
|
||||||
context->lengthtab = calloc(context->tablen, sizeof(uint32_t));
|
lengthtab = calloc(tablen, sizeof(uint32_t));
|
||||||
|
|
||||||
state->state++;
|
for (i = 0, j = 0; i < tablen; i++, j+=4)
|
||||||
break;
|
read4B(&starttab[i], &ptr[j]);
|
||||||
case 1:
|
for (i = 0, j = tablen * sizeof(uint32_t); i < tablen; i++, j+=4)
|
||||||
/* read offsets table */
|
read4B(&lengthtab[i], &ptr[j]);
|
||||||
for (; context->starttabidx < context->tablen;
|
|
||||||
context->starttabidx++, ptr+=4, bytes-=4) {
|
|
||||||
|
|
||||||
/* check overflow */
|
state->count += tablen * sizeof(uint32_t) * 2;
|
||||||
if (bytes < 4)
|
|
||||||
return ptr - buf;
|
|
||||||
|
|
||||||
read4B(&context->starttab[context->starttabidx], ptr);
|
|
||||||
}
|
|
||||||
state->state++;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
/* read lengths table */
|
|
||||||
for (; context->lengthtabidx < context->tablen;
|
|
||||||
context->lengthtabidx++, ptr+=4, bytes-=4) {
|
|
||||||
|
|
||||||
/* check overflow */
|
for (rowno = 0; rowno < im->ysize; rowno++, state->y += state->ystep)
|
||||||
if (bytes < 4)
|
|
||||||
return ptr - buf;
|
|
||||||
|
|
||||||
read4B(&context->lengthtab[context->lengthtabidx], ptr);
|
|
||||||
}
|
|
||||||
state->state++;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
/* rows decompression */
|
|
||||||
for (; context->rowno < im->ysize * im->bands;
|
|
||||||
context->rowno++, state->y += state->ystep )
|
|
||||||
{
|
{
|
||||||
context->channo = (int)(context->rowno / im->ysize);
|
for (channo = 0; channo < im->bands; channo++)
|
||||||
rleoffset = context->starttab[context->rowno];
|
{
|
||||||
|
rleoffset = starttab[rowno + channo * im->ysize];
|
||||||
|
rlelength = lengthtab[rowno + channo * im->ysize];
|
||||||
rleoffset -= SGI_HEADER_SIZE;
|
rleoffset -= SGI_HEADER_SIZE;
|
||||||
rlelength = context->lengthtab[context->rowno];
|
|
||||||
|
|
||||||
/* check overflow */
|
|
||||||
if (rlelength > bytes)
|
|
||||||
return ptr - buf;
|
|
||||||
|
|
||||||
if (context->bpc == 1) {
|
|
||||||
if(expandrow(state->buffer, ptr, rlelength)) {
|
|
||||||
/* err: compressed row doesn't finish with 0 */
|
|
||||||
state->errcode = IMAGING_CODEC_OVERRUN;
|
|
||||||
return ptr - buf;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (context->bpc ==1) {
|
||||||
|
if(expandrow(&state->buffer[channo], &ptr[rleoffset], rlelength, im->bands))
|
||||||
|
goto sgi_finish_decode;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if(expandrow2((UINT16*)state->buffer, (UINT16*)ptr, rlelength)) {
|
if(expandrow2((UINT16*)&state->buffer[channo], (UINT16*)&ptr[rleoffset], rlelength, im->bands))
|
||||||
/* err: compressed row doesn't finish with 0 */
|
goto sgi_finish_decode;
|
||||||
state->errcode = IMAGING_CODEC_OVERRUN;
|
|
||||||
return ptr - buf;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* reset index */
|
state->count += rlelength;
|
||||||
if (state->y == -1)
|
|
||||||
state->y = im->ysize - 1;
|
|
||||||
if (state->y == im->ysize)
|
|
||||||
state->y = 0;
|
|
||||||
|
|
||||||
/* set image data */
|
|
||||||
for (state->x = context->channo; state->x < im->xsize * im->pixelsize; state->x+=im->pixelsize)
|
|
||||||
((UINT8*)im->image[state->y])[state->x] = *state->buffer++;
|
|
||||||
|
|
||||||
state->buffer -= im->xsize;
|
|
||||||
|
|
||||||
bytes -= rlelength;
|
|
||||||
ptr += rlelength;
|
|
||||||
}
|
|
||||||
return -1; /* no error */
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return ptr - buf;
|
state->shuffle((UINT8*)im->image[state->y], state->buffer, im->xsize);
|
||||||
}
|
|
||||||
|
}
|
||||||
int ImagingSgiRleDecodeCleanup(ImagingCodecState state) {
|
|
||||||
|
bufsize++;
|
||||||
// SGISTATE *context;
|
|
||||||
// context = (SGISTATE*)state->context;
|
sgi_finish_decode:
|
||||||
|
|
||||||
// free(context->starttab);
|
free(starttab);
|
||||||
// free(context->lengthtab);
|
free(lengthtab);
|
||||||
// // free(context);
|
free(ptr);
|
||||||
|
|
||||||
return -1;
|
return state->count - bufsize;
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user