libavcodec/sunrast.c
d43df959
 /*
  * Sun Rasterfile (.sun/.ras/im{1,8,24}/.sunras) image decoder
  * Copyright (c) 2007, 2008 Ivo van Poorten
  *
  * 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
  */
 
1d9c2dc8
 #include "libavutil/common.h"
6a5d31ac
 #include "libavutil/intreadwrite.h"
7ffe76e5
 #include "libavutil/imgutils.h"
d43df959
 #include "avcodec.h"
594d4d5d
 #include "internal.h"
1f1fbcfa
 #include "sunrast.h"
d43df959
 
 static int sunrast_decode_frame(AVCodecContext *avctx, void *data,
df9b9567
                                 int *got_frame, AVPacket *avpkt)
 {
415f358a
     const uint8_t *buf       = avpkt->data;
     const uint8_t *buf_end   = avpkt->data + avpkt->size;
759001c5
     AVFrame * const p        = data;
d43df959
     unsigned int w, h, depth, type, maptype, maplength, stride, x, y, len, alen;
ceb0dd9f
     uint8_t *ptr, *ptr2 = NULL;
6ea28fc5
     const uint8_t *bufstart = buf;
41afdd91
     int ret;
d43df959
 
039f3c33
     if (avpkt->size < 32)
         return AVERROR_INVALIDDATA;
 
fef37717
     if (AV_RB32(buf) != RAS_MAGIC) {
d43df959
         av_log(avctx, AV_LOG_ERROR, "this is not sunras encoded data\n");
41afdd91
         return AVERROR_INVALIDDATA;
d43df959
     }
 
415f358a
     w         = AV_RB32(buf + 4);
     h         = AV_RB32(buf + 8);
     depth     = AV_RB32(buf + 12);
     type      = AV_RB32(buf + 20);
     maptype   = AV_RB32(buf + 24);
     maplength = AV_RB32(buf + 28);
2305742b
     buf      += 32;
d43df959
 
e37f161e
     if (type == RT_EXPERIMENTAL) {
6d97484d
         avpriv_request_sample(avctx, "TIFF/IFF/EXPERIMENTAL (compression) type");
f9708e9a
         return AVERROR_PATCHWELCOME;
d43df959
     }
8beeec88
     if (type > RT_FORMAT_IFF) {
b9596a50
         av_log(avctx, AV_LOG_ERROR, "invalid (compression) type\n");
41afdd91
         return AVERROR_INVALIDDATA;
d43df959
     }
79c7064c
     if (maptype == RMT_RAW) {
6d97484d
         avpriv_request_sample(avctx, "Unknown colormap type");
79c7064c
         return AVERROR_PATCHWELCOME;
     }
     if (maptype > RMT_RAW) {
d43df959
         av_log(avctx, AV_LOG_ERROR, "invalid colormap type\n");
41afdd91
         return AVERROR_INVALIDDATA;
d43df959
     }
 
b9596a50
     if (type == RT_FORMAT_TIFF || type == RT_FORMAT_IFF) {
         av_log(avctx, AV_LOG_ERROR, "unsupported (compression) type\n");
         return -1;
     }
d43df959
 
     switch (depth) {
         case 1:
ac627b3d
             avctx->pix_fmt = maplength ? AV_PIX_FMT_PAL8 : AV_PIX_FMT_MONOWHITE;
ceb0dd9f
             break;
         case 4:
ac627b3d
             avctx->pix_fmt = maplength ? AV_PIX_FMT_PAL8 : AV_PIX_FMT_NONE;
d43df959
             break;
         case 8:
716d413c
             avctx->pix_fmt = maplength ? AV_PIX_FMT_PAL8 : AV_PIX_FMT_GRAY8;
d43df959
             break;
         case 24:
716d413c
             avctx->pix_fmt = (type == RT_FORMAT_RGB) ? AV_PIX_FMT_RGB24 : AV_PIX_FMT_BGR24;
d43df959
             break;
2f21c507
         case 32:
ac627b3d
             avctx->pix_fmt = (type == RT_FORMAT_RGB) ? AV_PIX_FMT_0RGB : AV_PIX_FMT_0BGR;
2f21c507
             break;
d43df959
         default:
             av_log(avctx, AV_LOG_ERROR, "invalid depth\n");
41afdd91
             return AVERROR_INVALIDDATA;
d43df959
     }
 
9c0ece1b
     ret = ff_set_dimensions(avctx, w, h);
     if (ret < 0)
         return ret;
 
1ec94b0f
     if ((ret = ff_get_buffer(avctx, p, 0)) < 0)
41afdd91
         return ret;
d43df959
 
ce5e49b0
     p->pict_type = AV_PICTURE_TYPE_I;
d43df959
 
039f3c33
     if (buf_end - buf < maplength)
         return AVERROR_INVALIDDATA;
 
ceb0dd9f
     if (depth > 8 && maplength) {
d43df959
         av_log(avctx, AV_LOG_WARNING, "useless colormap found or file is corrupted, trying to recover\n");
 
912e7516
     } else if (maplength) {
d43df959
         unsigned int len = maplength / 3;
 
         if (maplength % 3 || maplength > 768) {
             av_log(avctx, AV_LOG_WARNING, "invalid colormap length\n");
41afdd91
             return AVERROR_INVALIDDATA;
d43df959
         }
 
         ptr = p->data[1];
415f358a
         for (x = 0; x < len; x++, ptr += 4)
2fed05f5
             *(uint32_t *)ptr = (0xFFU<<24) + (buf[x]<<16) + (buf[len+x]<<8) + buf[len+len+x];
d43df959
     }
 
     buf += maplength;
 
ceb0dd9f
     if (maplength && depth < 8) {
c4f1abec
         ptr = ptr2 = av_malloc_array((w + 15), h);
ceb0dd9f
         if (!ptr)
             return AVERROR(ENOMEM);
         stride = (w + 15 >> 3) * depth;
     } else {
d43df959
     ptr    = p->data[0];
     stride = p->linesize[0];
ceb0dd9f
     }
d43df959
 
     /* scanlines are aligned on 16 bit boundaries */
     len  = (depth * w + 7) >> 3;
415f358a
     alen = len + (len & 1);
d43df959
 
     if (type == RT_BYTE_ENCODED) {
         int value, run;
415f358a
         uint8_t *end = ptr + h * stride;
d43df959
 
         x = 0;
039f3c33
         while (ptr != end && buf < buf_end) {
d43df959
             run = 1;
039f3c33
             if (buf_end - buf < 1)
                 return AVERROR_INVALIDDATA;
 
566df2ee
             if ((value = *buf++) == RLE_TRIGGER) {
d43df959
                 run = *buf++ + 1;
                 if (run != 1)
                     value = *buf++;
             }
             while (run--) {
                 if (x < len)
                     ptr[x] = value;
                 if (++x >= alen) {
                     x = 0;
                     ptr += stride;
                     if (ptr == end)
                         break;
                 }
             }
         }
     } else {
415f358a
         for (y = 0; y < h; y++) {
039f3c33
             if (buf_end - buf < len)
                 break;
d43df959
             memcpy(ptr, buf, len);
             ptr += stride;
             buf += alen;
         }
     }
ac627b3d
     if (avctx->pix_fmt == AV_PIX_FMT_PAL8 && depth < 8) {
ceb0dd9f
         uint8_t *ptr_free = ptr2;
         ptr = p->data[0];
         for (y=0; y<h; y++) {
             for (x = 0; x < (w + 7 >> 3) * depth; x++) {
                 if (depth == 1) {
                     ptr[8*x]   = ptr2[x] >> 7;
                     ptr[8*x+1] = ptr2[x] >> 6 & 1;
                     ptr[8*x+2] = ptr2[x] >> 5 & 1;
                     ptr[8*x+3] = ptr2[x] >> 4 & 1;
                     ptr[8*x+4] = ptr2[x] >> 3 & 1;
                     ptr[8*x+5] = ptr2[x] >> 2 & 1;
                     ptr[8*x+6] = ptr2[x] >> 1 & 1;
                     ptr[8*x+7] = ptr2[x]      & 1;
                 } else {
                     ptr[2*x]   = ptr2[x] >> 4;
                     ptr[2*x+1] = ptr2[x] & 0xF;
                 }
             }
             ptr  += p->linesize[0];
             ptr2 += (w + 15 >> 3) * depth;
         }
         av_freep(&ptr_free);
     }
d43df959
 
df9b9567
     *got_frame = 1;
d43df959
 
     return buf - bufstart;
 }
 
e7e2df27
 AVCodec ff_sunrast_decoder = {
ec6402b7
     .name           = "sunrast",
b2bed932
     .long_name      = NULL_IF_CONFIG_SMALL("Sun Rasterfile image"),
ec6402b7
     .type           = AVMEDIA_TYPE_VIDEO,
36ef5369
     .id             = AV_CODEC_ID_SUNRAST,
ec6402b7
     .decode         = sunrast_decode_frame,
def97856
     .capabilities   = AV_CODEC_CAP_DR1,
d43df959
 };