a991b1fe |
/*
* Copyright (c) 2006 Konstantin Shishkov
*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/ |
ee5c8a9b |
/** |
ba87f080 |
* @file |
1e97700f |
* TIFF image decoder |
ee5c8a9b |
* @author Konstantin Shishkov
*/ |
f6f95d4e |
|
a991b1fe |
#include "avcodec.h" |
b250f9c6 |
#if CONFIG_ZLIB |
a991b1fe |
#include <zlib.h>
#endif |
ffc5430b |
#include "lzw.h" |
716222db |
#include "tiff.h" |
92799d32 |
#include "faxcompr.h" |
91cc5d37 |
#include "libavutil/common.h" |
bdcc13f2 |
#include "libavutil/intreadwrite.h" |
7ffe76e5 |
#include "libavutil/imgutils.h" |
339aabd5 |
|
a991b1fe |
typedef struct TiffContext {
AVCodecContext *avctx;
AVFrame picture;
int width, height; |
bbc572a2 |
unsigned int bpp, bppcount; |
7ca5338b |
uint32_t palette[256];
int palette_is_set; |
a991b1fe |
int le; |
aa909d30 |
enum TiffCompr compr; |
e2570297 |
int invert; |
92799d32 |
int fax_opts; |
108c96d5 |
int predictor; |
a58e96be |
int fill_order; |
a991b1fe |
|
108c96d5 |
int strips, rps, sstype; |
a991b1fe |
int sot; |
08e50529 |
const uint8_t* stripdata;
const uint8_t* stripsizes; |
a991b1fe |
int stripsize, stripoff; |
ffc5430b |
LZWState *lzw; |
a991b1fe |
} TiffContext;
|
ef673211 |
static unsigned tget_short(const uint8_t **p, int le) {
unsigned v = le ? AV_RL16(*p) : AV_RB16(*p); |
a991b1fe |
*p += 2;
return v;
}
|
ef673211 |
static unsigned tget_long(const uint8_t **p, int le) {
unsigned v = le ? AV_RL32(*p) : AV_RB32(*p); |
a991b1fe |
*p += 4;
return v;
}
|
ef673211 |
static unsigned tget(const uint8_t **p, int type, int le) { |
a991b1fe |
switch(type){
case TIFF_BYTE : return *(*p)++;
case TIFF_SHORT: return tget_short(p, le);
case TIFF_LONG : return tget_long (p, le); |
ef673211 |
default : return UINT_MAX; |
a991b1fe |
}
}
|
eec69b39 |
#if CONFIG_ZLIB |
e96f885a |
static int tiff_uncompress(uint8_t *dst, unsigned long *len, const uint8_t *src, int size)
{
z_stream zstream;
int zret;
memset(&zstream, 0, sizeof(zstream));
zstream.next_in = src;
zstream.avail_in = size;
zstream.next_out = dst;
zstream.avail_out = *len;
zret = inflateInit(&zstream);
if (zret != Z_OK) {
av_log(NULL, AV_LOG_ERROR, "Inflate init error: %d\n", zret);
return zret;
}
zret = inflate(&zstream, Z_SYNC_FLUSH);
inflateEnd(&zstream);
*len = zstream.total_out;
return zret == Z_STREAM_END ? Z_OK : zret;
} |
eec69b39 |
#endif |
e96f885a |
|
5f5e37e0 |
static void av_always_inline horizontal_fill(unsigned int bpp, uint8_t* dst,
int usePtr, const uint8_t *src,
uint8_t c, int width, int offset)
{ |
b46fb615 |
switch (bpp) { |
7ee0649f |
case 1:
while (--width >= 0) {
dst[(width+offset)*8+7] = (usePtr ? src[width] : c) & 0x1;
dst[(width+offset)*8+6] = (usePtr ? src[width] : c) >> 1 & 0x1;
dst[(width+offset)*8+5] = (usePtr ? src[width] : c) >> 2 & 0x1;
dst[(width+offset)*8+4] = (usePtr ? src[width] : c) >> 3 & 0x1;
dst[(width+offset)*8+3] = (usePtr ? src[width] : c) >> 4 & 0x1;
dst[(width+offset)*8+2] = (usePtr ? src[width] : c) >> 5 & 0x1;
dst[(width+offset)*8+1] = (usePtr ? src[width] : c) >> 6 & 0x1;
dst[(width+offset)*8+0] = (usePtr ? src[width] : c) >> 7;
}
break; |
b46fb615 |
case 2: |
e767968c |
while (--width >= 0) {
dst[(width+offset)*4+3] = (usePtr ? src[width] : c) & 0x3;
dst[(width+offset)*4+2] = (usePtr ? src[width] : c) >> 2 & 0x3;
dst[(width+offset)*4+1] = (usePtr ? src[width] : c) >> 4 & 0x3;
dst[(width+offset)*4+0] = (usePtr ? src[width] : c) >> 6; |
5f5e37e0 |
} |
b46fb615 |
break;
case 4: |
e767968c |
while (--width >= 0) {
dst[(width+offset)*2+1] = (usePtr ? src[width] : c) & 0xF;
dst[(width+offset)*2+0] = (usePtr ? src[width] : c) >> 4; |
5f5e37e0 |
} |
b46fb615 |
break;
default: |
5f5e37e0 |
if (usePtr) {
memcpy(dst + offset, src, width);
} else {
memset(dst + offset, c, width);
}
}
}
|
08e50529 |
static int tiff_unpack_strip(TiffContext *s, uint8_t* dst, int stride, const uint8_t *src, int size, int lines){ |
a991b1fe |
int c, line, pixels, code; |
08e50529 |
const uint8_t *ssrc = src; |
d8297ca4 |
int width = ((s->width * s->bpp) + 7) >> 3; |
b250f9c6 |
#if CONFIG_ZLIB |
a991b1fe |
uint8_t *zbuf; unsigned long outlen;
if(s->compr == TIFF_DEFLATE || s->compr == TIFF_ADOBE_DEFLATE){ |
e96f885a |
int ret; |
a991b1fe |
outlen = width * lines;
zbuf = av_malloc(outlen); |
e96f885a |
ret = tiff_uncompress(zbuf, &outlen, src, size);
if(ret != Z_OK){
av_log(s->avctx, AV_LOG_ERROR, "Uncompressing failed (%lu of %lu) with error %d\n", outlen, (unsigned long)width * lines, ret); |
a991b1fe |
av_free(zbuf);
return -1;
}
src = zbuf;
for(line = 0; line < lines; line++){ |
4392b3d1 |
if(s->bpp < 8 && s->avctx->pix_fmt == PIX_FMT_PAL8){
horizontal_fill(s->bpp, dst, 1, src, 0, width, 0);
}else{
memcpy(dst, src, width);
} |
a991b1fe |
dst += stride;
src += width;
}
av_free(zbuf);
return 0;
}
#endif |
ffc5430b |
if(s->compr == TIFF_LZW){
if(ff_lzw_decode_init(s->lzw, 8, src, size, FF_LZW_TIFF) < 0){
av_log(s->avctx, AV_LOG_ERROR, "Error initializing LZW decoder\n");
return -1;
}
} |
92799d32 |
if(s->compr == TIFF_CCITT_RLE || s->compr == TIFF_G3 || s->compr == TIFF_G4){
int i, ret = 0;
uint8_t *src2 = av_malloc(size + FF_INPUT_BUFFER_PADDING_SIZE);
if(!src2 || (unsigned)size + FF_INPUT_BUFFER_PADDING_SIZE < (unsigned)size){
av_log(s->avctx, AV_LOG_ERROR, "Error allocating temporary buffer\n");
return -1;
} |
a8b95021 |
if(s->fax_opts & 2){
av_log(s->avctx, AV_LOG_ERROR, "Uncompressed fax mode is not supported (yet)\n");
av_free(src2);
return -1;
} |
a58e96be |
if(!s->fill_order){
memcpy(src2, src, size);
}else{
for(i = 0; i < size; i++) |
91cc5d37 |
src2[i] = av_reverse[src[i]]; |
a58e96be |
} |
92799d32 |
memset(src2+size, 0, FF_INPUT_BUFFER_PADDING_SIZE);
switch(s->compr){
case TIFF_CCITT_RLE:
case TIFF_G3:
case TIFF_G4: |
8da5d3dc |
ret = ff_ccitt_unpack(s->avctx, src2, size, dst, lines, stride, s->compr, s->fax_opts); |
92799d32 |
break;
} |
4392b3d1 |
if (s->bpp < 8 && s->avctx->pix_fmt == PIX_FMT_PAL8) |
7ee0649f |
for (line = 0; line < lines; line++) {
horizontal_fill(s->bpp, dst, 1, dst, 0, width, 0);
dst += stride;
} |
92799d32 |
av_free(src2);
return ret;
} |
a991b1fe |
for(line = 0; line < lines; line++){
if(src - ssrc > size){
av_log(s->avctx, AV_LOG_ERROR, "Source data overread\n");
return -1;
}
switch(s->compr){
case TIFF_RAW: |
5ca5d432 |
if (ssrc + size - src < width)
return AVERROR_INVALIDDATA; |
b437f5b0 |
if (!s->fill_order) { |
714d2f97 |
horizontal_fill(s->bpp * (s->avctx->pix_fmt == PIX_FMT_PAL8),
dst, 1, src, 0, width, 0); |
b437f5b0 |
} else {
int i;
for (i = 0; i < width; i++)
dst[i] = av_reverse[src[i]];
} |
4386ed69 |
src += width; |
a991b1fe |
break;
case TIFF_PACKBITS:
for(pixels = 0; pixels < width;){ |
96de1c5e |
if (ssrc + size - src < 2)
return AVERROR_INVALIDDATA; |
a991b1fe |
code = (int8_t)*src++;
if(code >= 0){
code++; |
96de1c5e |
if (pixels + code > width ||
ssrc + size - src < code) { |
a991b1fe |
av_log(s->avctx, AV_LOG_ERROR, "Copy went out of bounds\n");
return -1;
} |
714d2f97 |
horizontal_fill(s->bpp * (s->avctx->pix_fmt == PIX_FMT_PAL8),
dst, 1, src, 0, code, pixels); |
a991b1fe |
src += code;
pixels += code;
}else if(code != -128){ // -127..-1
code = (-code) + 1;
if(pixels + code > width){
av_log(s->avctx, AV_LOG_ERROR, "Run went out of bounds\n");
return -1;
}
c = *src++; |
714d2f97 |
horizontal_fill(s->bpp * (s->avctx->pix_fmt == PIX_FMT_PAL8),
dst, 0, NULL, c, code, pixels); |
a991b1fe |
pixels += code;
}
}
break; |
ffc5430b |
case TIFF_LZW:
pixels = ff_lzw_decode(s->lzw, dst, width);
if(pixels < width){
av_log(s->avctx, AV_LOG_ERROR, "Decoded only %i bytes of %i\n", pixels, width);
return -1;
} |
4392b3d1 |
if (s->bpp < 8 && s->avctx->pix_fmt == PIX_FMT_PAL8) |
e767968c |
horizontal_fill(s->bpp, dst, 1, dst, 0, width, 0); |
ffc5430b |
break; |
a991b1fe |
}
dst += stride;
}
return 0;
}
|
bbc572a2 |
static int init_image(TiffContext *s)
{
int i, ret;
uint32_t *pal;
switch (s->bpp * 10 + s->bppcount) {
case 11: |
4392b3d1 |
if (!s->palette_is_set) {
s->avctx->pix_fmt = PIX_FMT_MONOBLACK;
break;
} |
5f5e37e0 |
case 21:
case 41: |
bbc572a2 |
case 81:
s->avctx->pix_fmt = PIX_FMT_PAL8;
break;
case 243:
s->avctx->pix_fmt = PIX_FMT_RGB24;
break;
case 161:
s->avctx->pix_fmt = PIX_FMT_GRAY16BE;
break; |
f746f379 |
case 162:
s->avctx->pix_fmt = PIX_FMT_GRAY8A;
break; |
bbc572a2 |
case 324:
s->avctx->pix_fmt = PIX_FMT_RGBA;
break;
case 483:
s->avctx->pix_fmt = s->le ? PIX_FMT_RGB48LE : PIX_FMT_RGB48BE;
break; |
c714cd3b |
case 644:
s->avctx->pix_fmt = s->le ? PIX_FMT_RGBA64LE : PIX_FMT_RGBA64BE;
break; |
bbc572a2 |
default:
av_log(s->avctx, AV_LOG_ERROR,
"This format is not supported (bpp=%d, bppcount=%d)\n",
s->bpp, s->bppcount);
return AVERROR_INVALIDDATA;
}
if (s->width != s->avctx->width || s->height != s->avctx->height) {
if ((ret = av_image_check_size(s->width, s->height, 0, s->avctx)) < 0)
return ret;
avcodec_set_dimensions(s->avctx, s->width, s->height);
}
if (s->picture.data[0])
s->avctx->release_buffer(s->avctx, &s->picture);
if ((ret = s->avctx->get_buffer(s->avctx, &s->picture)) < 0) {
av_log(s->avctx, AV_LOG_ERROR, "get_buffer() failed\n");
return ret;
} |
7ca5338b |
if (s->avctx->pix_fmt == PIX_FMT_PAL8) {
if (s->palette_is_set) {
memcpy(s->picture.data[1], s->palette, sizeof(s->palette));
} else {
/* make default grayscale pal */
pal = (uint32_t *) s->picture.data[1]; |
d898ab90 |
for (i = 0; i < 1<<s->bpp; i++) |
3b5733bc |
pal[i] = 0xFF << 24 | i * 255 / ((1<<s->bpp) - 1) * 0x010101; |
7ca5338b |
} |
bbc572a2 |
}
return 0;
} |
a991b1fe |
|
108c96d5 |
static int tiff_decode_tag(TiffContext *s, const uint8_t *start, const uint8_t *buf, const uint8_t *end_buf) |
a991b1fe |
{ |
ef673211 |
unsigned tag, type, count, off, value = 0; |
108c96d5 |
int i, j; |
640b73e1 |
uint32_t *pal; |
08e50529 |
const uint8_t *rp, *gp, *bp; |
a991b1fe |
|
5ca5d432 |
if (end_buf - buf < 12)
return -1; |
a991b1fe |
tag = tget_short(&buf, s->le);
type = tget_short(&buf, s->le);
count = tget_long(&buf, s->le);
off = tget_long(&buf, s->le);
|
424b6edd |
if (type == 0 || type >= FF_ARRAY_ELEMS(type_sizes)) {
av_log(s->avctx, AV_LOG_DEBUG, "Unknown tiff type (%u) encountered\n", type);
return 0;
}
|
a991b1fe |
if(count == 1){
switch(type){
case TIFF_BYTE:
case TIFF_SHORT:
buf -= 4;
value = tget(&buf, type, s->le);
buf = NULL;
break;
case TIFF_LONG:
value = off;
buf = NULL;
break; |
6a4583e9 |
case TIFF_STRING:
if(count <= 4){
buf -= 4;
break;
} |
a991b1fe |
default: |
ef673211 |
value = UINT_MAX; |
a991b1fe |
buf = start + off;
} |
424b6edd |
} else {
if (count <= 4 && type_sizes[type] * count <= 4) {
buf -= 4;
} else { |
a991b1fe |
buf = start + off;
}
}
if(buf && (buf < start || buf > end_buf)){
av_log(s->avctx, AV_LOG_ERROR, "Tag referencing position outside the image\n");
return -1;
}
switch(tag){
case TIFF_WIDTH:
s->width = value;
break;
case TIFF_HEIGHT:
s->height = value;
break;
case TIFF_BPP: |
bbc572a2 |
s->bppcount = count; |
d381249b |
if(count > 4){
av_log(s->avctx, AV_LOG_ERROR, "This format is not supported (bpp=%d, %d components)\n", s->bpp, count);
return -1;
} |
a991b1fe |
if(count == 1) s->bpp = value;
else{
switch(type){
case TIFF_BYTE: |
ec8a152b |
s->bpp = (off & 0xFF) + ((off >> 8) & 0xFF) + ((off >> 16) & 0xFF) + ((off >> 24) & 0xFF); |
a991b1fe |
break;
case TIFF_SHORT:
case TIFF_LONG: |
ec8a152b |
s->bpp = 0; |
5ca5d432 |
for(i = 0; i < count && buf < end_buf; i++) s->bpp += tget(&buf, type, s->le); |
a991b1fe |
break;
default:
s->bpp = -1;
}
} |
bbc572a2 |
break;
case TIFF_SAMPLES_PER_PIXEL:
if (count != 1) {
av_log(s->avctx, AV_LOG_ERROR,
"Samples per pixel requires a single value, many provided\n");
return AVERROR_INVALIDDATA; |
e2570297 |
} |
bbc572a2 |
if (s->bppcount == 1)
s->bpp *= value;
s->bppcount = value; |
a991b1fe |
break;
case TIFF_COMPR:
s->compr = value; |
108c96d5 |
s->predictor = 0; |
a991b1fe |
switch(s->compr){
case TIFF_RAW:
case TIFF_PACKBITS: |
ffc5430b |
case TIFF_LZW: |
92799d32 |
case TIFF_CCITT_RLE:
break;
case TIFF_G3:
case TIFF_G4:
s->fax_opts = 0; |
a991b1fe |
break;
case TIFF_DEFLATE:
case TIFF_ADOBE_DEFLATE: |
b250f9c6 |
#if CONFIG_ZLIB |
a991b1fe |
break;
#else
av_log(s->avctx, AV_LOG_ERROR, "Deflate: ZLib not compiled in\n");
return -1;
#endif |
27c18939 |
case TIFF_JPEG:
case TIFF_NEWJPEG:
av_log(s->avctx, AV_LOG_ERROR, "JPEG compression is not supported\n");
return -1; |
a991b1fe |
default:
av_log(s->avctx, AV_LOG_ERROR, "Unknown compression method %i\n", s->compr);
return -1;
}
break;
case TIFF_ROWSPERSTRIP: |
ef673211 |
if (type == TIFF_LONG && value == UINT_MAX) |
fe0e64ca |
value = s->height; |
380fd08f |
if(value < 1){ |
a991b1fe |
av_log(s->avctx, AV_LOG_ERROR, "Incorrect value of rows per strip\n");
return -1;
}
s->rps = value;
break;
case TIFF_STRIP_OFFS:
if(count == 1){
s->stripdata = NULL;
s->stripoff = value;
}else
s->stripdata = start + off;
s->strips = count; |
615259a3 |
if(s->strips == 1) s->rps = s->height; |
a991b1fe |
s->sot = type;
if(s->stripdata > end_buf){
av_log(s->avctx, AV_LOG_ERROR, "Tag referencing position outside the image\n");
return -1;
}
break;
case TIFF_STRIP_SIZE:
if(count == 1){
s->stripsizes = NULL;
s->stripsize = value;
s->strips = 1;
}else{
s->stripsizes = start + off;
}
s->strips = count; |
108c96d5 |
s->sstype = type; |
a991b1fe |
if(s->stripsizes > end_buf){
av_log(s->avctx, AV_LOG_ERROR, "Tag referencing position outside the image\n");
return -1;
}
break; |
98dfdfdd |
case TIFF_TILE_BYTE_COUNTS:
case TIFF_TILE_LENGTH:
case TIFF_TILE_OFFSETS:
case TIFF_TILE_WIDTH:
av_log(s->avctx, AV_LOG_ERROR, "Tiled images are not supported\n");
return AVERROR_PATCHWELCOME;
break; |
a991b1fe |
case TIFF_PREDICTOR: |
108c96d5 |
s->predictor = value; |
a991b1fe |
break; |
e2570297 |
case TIFF_INVERT:
switch(value){
case 0:
s->invert = 1;
break;
case 1:
s->invert = 0;
break; |
d00026d0 |
case 2:
case 3:
break;
default:
av_log(s->avctx, AV_LOG_ERROR, "Color mode %d is not supported\n", value);
return -1; |
e2570297 |
}
break; |
a58e96be |
case TIFF_FILL_ORDER:
if(value < 1 || value > 2){
av_log(s->avctx, AV_LOG_ERROR, "Unknown FillOrder value %d, trying default one\n", value);
value = 1;
}
s->fill_order = value - 1;
break; |
e2570297 |
case TIFF_PAL: |
7ca5338b |
pal = (uint32_t *) s->palette; |
339aabd5 |
off = type_sizes[type]; |
5ca5d432 |
if (count / 3 > 256 || end_buf - buf < count / 3 * off * 3)
return -1; |
e2570297 |
rp = buf;
gp = buf + count / 3 * off;
bp = buf + count / 3 * off * 2; |
339aabd5 |
off = (type_sizes[type] - 1) << 3; |
e2570297 |
for(i = 0; i < count / 3; i++){ |
dbfdb288 |
j = 0xff << 24;
j |= (tget(&rp, type, s->le) >> off) << 16; |
e2570297 |
j |= (tget(&gp, type, s->le) >> off) << 8;
j |= tget(&bp, type, s->le) >> off;
pal[i] = j;
} |
7ca5338b |
s->palette_is_set = 1; |
2878cecd |
break;
case TIFF_PLANAR:
if(value == 2){
av_log(s->avctx, AV_LOG_ERROR, "Planar format is not supported\n");
return -1;
}
break; |
92799d32 |
case TIFF_T4OPTIONS: |
7c6f5238 |
if(s->compr == TIFF_G3)
s->fax_opts = value;
break; |
92799d32 |
case TIFF_T6OPTIONS: |
7c6f5238 |
if(s->compr == TIFF_G4)
s->fax_opts = value; |
92799d32 |
break; |
762b4662 |
default:
av_log(s->avctx, AV_LOG_DEBUG, "Unknown or unsupported tag %d/0X%0X\n", tag, tag); |
a991b1fe |
}
return 0;
}
static int decode_frame(AVCodecContext *avctx,
void *data, int *data_size, |
7a00bbad |
AVPacket *avpkt) |
a991b1fe |
{ |
7a00bbad |
const uint8_t *buf = avpkt->data;
int buf_size = avpkt->size; |
a991b1fe |
TiffContext * const s = avctx->priv_data;
AVFrame *picture = data;
AVFrame * const p= (AVFrame*)&s->picture; |
08e50529 |
const uint8_t *orig_buf = buf, *end_buf = buf + buf_size; |
5ca5d432 |
unsigned off;
int id, le, ret; |
108c96d5 |
int i, j, entries; |
5ca5d432 |
int stride;
unsigned soff, ssize; |
108c96d5 |
uint8_t *dst; |
a991b1fe |
//parse image header |
5ca5d432 |
if (end_buf - buf < 8)
return AVERROR_INVALIDDATA; |
fead30d4 |
id = AV_RL16(buf); buf += 2; |
a991b1fe |
if(id == 0x4949) le = 1;
else if(id == 0x4D4D) le = 0;
else{
av_log(avctx, AV_LOG_ERROR, "TIFF header not found\n");
return -1;
}
s->le = le; |
e2570297 |
s->invert = 0; |
329851e0 |
s->compr = TIFF_RAW; |
a58e96be |
s->fill_order = 0; |
a991b1fe |
// As TIFF 6.0 specification puts it "An arbitrary but carefully chosen number
// that further identifies the file as a TIFF file"
if(tget_short(&buf, le) != 42){
av_log(avctx, AV_LOG_ERROR, "The answer to life, universe and everything is not correct!\n");
return -1;
} |
2744fdbd |
// Reset these pointers so we can tell if they were set this frame
s->stripsizes = s->stripdata = NULL; |
a991b1fe |
/* parse image file directory */
off = tget_long(&buf, le); |
5ca5d432 |
if (off >= UINT_MAX - 14 || end_buf - orig_buf < off + 14) { |
a991b1fe |
av_log(avctx, AV_LOG_ERROR, "IFD offset is greater than image size\n"); |
5ca5d432 |
return AVERROR_INVALIDDATA; |
a991b1fe |
}
buf = orig_buf + off;
entries = tget_short(&buf, le);
for(i = 0; i < entries; i++){ |
108c96d5 |
if(tiff_decode_tag(s, orig_buf, buf, end_buf) < 0) |
a991b1fe |
return -1;
buf += 12;
} |
108c96d5 |
if(!s->stripdata && !s->stripoff){
av_log(avctx, AV_LOG_ERROR, "Image data is missing\n");
return -1;
}
/* now we have the data and may start decoding */ |
bbc572a2 |
if ((ret = init_image(s)) < 0)
return ret;
|
108c96d5 |
if(s->strips == 1 && !s->stripsize){
av_log(avctx, AV_LOG_WARNING, "Image data size missing\n");
s->stripsize = buf_size - s->stripoff;
}
stride = p->linesize[0];
dst = p->data[0];
for(i = 0; i < s->height; i += s->rps){ |
5ca5d432 |
if(s->stripsizes) {
if (s->stripsizes >= end_buf)
return AVERROR_INVALIDDATA; |
108c96d5 |
ssize = tget(&s->stripsizes, s->sstype, s->le); |
5ca5d432 |
} else |
108c96d5 |
ssize = s->stripsize;
if(s->stripdata){ |
5ca5d432 |
if (s->stripdata >= end_buf)
return AVERROR_INVALIDDATA; |
108c96d5 |
soff = tget(&s->stripdata, s->sot, s->le);
}else
soff = s->stripoff; |
5ca5d432 |
if (soff > buf_size || ssize > buf_size - soff) {
av_log(avctx, AV_LOG_ERROR, "Invalid strip size/offset\n");
return -1; |
a4db272a |
} |
108c96d5 |
if(tiff_unpack_strip(s, dst, stride, orig_buf + soff, ssize, FFMIN(s->rps, s->height - i)) < 0)
break;
dst += s->rps * stride;
}
if(s->predictor == 2){
dst = p->data[0];
soff = s->bpp >> 3;
ssize = s->width * soff; |
c714cd3b |
if (s->avctx->pix_fmt == PIX_FMT_RGB48LE ||
s->avctx->pix_fmt == PIX_FMT_RGBA64LE) { |
72381b2b |
for (i = 0; i < s->height; i++) {
for (j = soff; j < ssize; j += 2)
AV_WL16(dst + j, AV_RL16(dst + j) + AV_RL16(dst + j - soff));
dst += stride;
} |
c714cd3b |
} else if (s->avctx->pix_fmt == PIX_FMT_RGB48BE ||
s->avctx->pix_fmt == PIX_FMT_RGBA64BE) { |
72381b2b |
for (i = 0; i < s->height; i++) {
for (j = soff; j < ssize; j += 2)
AV_WB16(dst + j, AV_RB16(dst + j) + AV_RB16(dst + j - soff));
dst += stride;
}
} else { |
251345a3 |
for(i = 0; i < s->height; i++) {
for(j = soff; j < ssize; j++)
dst[j] += dst[j - soff];
dst += stride;
} |
72381b2b |
} |
108c96d5 |
} |
a991b1fe |
|
e2570297 |
if(s->invert){ |
251345a3 |
dst = s->picture.data[0];
for(i = 0; i < s->height; i++){
for(j = 0; j < s->picture.linesize[0]; j++)
dst[j] = (s->avctx->pix_fmt == PIX_FMT_PAL8 ? (1<<s->bpp) - 1 : 255) - dst[j];
dst += s->picture.linesize[0]; |
e2570297 |
}
} |
a991b1fe |
*picture= *(AVFrame*)&s->picture;
*data_size = sizeof(AVPicture);
return buf_size;
}
|
98a6fff9 |
static av_cold int tiff_init(AVCodecContext *avctx){ |
a991b1fe |
TiffContext *s = avctx->priv_data;
s->width = 0;
s->height = 0;
s->avctx = avctx;
avcodec_get_frame_defaults((AVFrame*)&s->picture);
avctx->coded_frame= (AVFrame*)&s->picture; |
ffc5430b |
ff_lzw_decode_open(&s->lzw); |
92799d32 |
ff_ccitt_unpack_init(); |
a991b1fe |
return 0;
}
|
98a6fff9 |
static av_cold int tiff_end(AVCodecContext *avctx) |
a991b1fe |
{
TiffContext * const s = avctx->priv_data;
|
ffc5430b |
ff_lzw_decode_close(&s->lzw); |
a991b1fe |
if(s->picture.data[0])
avctx->release_buffer(avctx, &s->picture);
return 0;
}
|
e7e2df27 |
AVCodec ff_tiff_decoder = { |
ec6402b7 |
.name = "tiff",
.type = AVMEDIA_TYPE_VIDEO,
.id = CODEC_ID_TIFF,
.priv_data_size = sizeof(TiffContext),
.init = tiff_init,
.close = tiff_end,
.decode = decode_frame,
.capabilities = CODEC_CAP_DR1, |
fe4bf374 |
.long_name = NULL_IF_CONFIG_SMALL("TIFF image"), |
a991b1fe |
}; |