Commit 96d413a4 authored by bcoudurier's avatar bcoudurier

use lzw compression in gif encoder

git-svn-id: file:///var/local/repositories/ffmpeg/trunk@20723 9553f0bf-9b14-0410-a0b8-cfaf0461ba5b
parent 9753b3cd
...@@ -43,6 +43,7 @@ ...@@ -43,6 +43,7 @@
#include "avcodec.h" #include "avcodec.h"
#include "bytestream.h" #include "bytestream.h"
#include "lzw.h"
/* The GIF format uses reversed order for bitstreams... */ /* The GIF format uses reversed order for bitstreams... */
/* at least they don't use PDP_ENDIAN :) */ /* at least they don't use PDP_ENDIAN :) */
...@@ -50,11 +51,10 @@ ...@@ -50,11 +51,10 @@
#include "put_bits.h" #include "put_bits.h"
/* bitstream minipacket size */
#define GIF_CHUNKS 100
typedef struct { typedef struct {
AVFrame picture; AVFrame picture;
LZWState *lzw;
uint8_t *buf;
} GIFContext; } GIFContext;
/* GIF header */ /* GIF header */
...@@ -82,12 +82,12 @@ static int gif_image_write_header(AVCodecContext *avctx, ...@@ -82,12 +82,12 @@ static int gif_image_write_header(AVCodecContext *avctx,
return 0; return 0;
} }
static int gif_image_write_image(AVCodecContext *avctx, uint8_t **bytestream, static int gif_image_write_image(AVCodecContext *avctx,
uint8_t **bytestream, uint8_t *end,
const uint8_t *buf, int linesize) const uint8_t *buf, int linesize)
{ {
PutBitContext p; GIFContext *s = avctx->priv_data;
uint8_t buffer[200]; /* 100 * 9 / 8 = 113 */ int len, height;
int i, left, w;
const uint8_t *ptr; const uint8_t *ptr;
/* image block */ /* image block */
...@@ -101,39 +101,25 @@ static int gif_image_write_image(AVCodecContext *avctx, uint8_t **bytestream, ...@@ -101,39 +101,25 @@ static int gif_image_write_image(AVCodecContext *avctx, uint8_t **bytestream,
bytestream_put_byte(bytestream, 0x08); bytestream_put_byte(bytestream, 0x08);
left= avctx->width * avctx->height; ff_lzw_encode_init(s->lzw, s->buf, avctx->width*avctx->height,
12, FF_LZW_GIF, put_bits);
init_put_bits(&p, buffer, 130);
/*
* the thing here is the bitstream is written as little packets, with a size byte before
* but it's still the same bitstream between packets (no flush !)
*/
ptr = buf; ptr = buf;
w = avctx->width; for (height = avctx->height; height--;) {
while(left>0) { len += ff_lzw_encode(s->lzw, ptr, avctx->width);
ptr += linesize;
put_bits(&p, 9, 0x0100); /* clear code */
for(i=(left<GIF_CHUNKS)?left:GIF_CHUNKS;i;i--) {
put_bits(&p, 9, *ptr++);
if (--w == 0) {
w = avctx->width;
buf += linesize;
ptr = buf;
}
} }
len += ff_lzw_encode_flush(s->lzw, flush_put_bits);
if(left<=GIF_CHUNKS) {
put_bits(&p, 9, 0x101); /* end of stream */ ptr = s->buf;
flush_put_bits(&p); while (len > 0) {
} int size = FFMIN(255, len);
if(put_bits_ptr(&p) - p.buf > 0) { bytestream_put_byte(bytestream, size);
bytestream_put_byte(bytestream, put_bits_ptr(&p) - p.buf); /* byte count of the packet */ if (end - *bytestream < size)
bytestream_put_buffer(bytestream, p.buf, put_bits_ptr(&p) - p.buf); /* the actual buffer */ return -1;
p.buf_ptr = p.buf; /* dequeue the bytes off the bitstream */ bytestream_put_buffer(bytestream, ptr, size);
} ptr += size;
left-=GIF_CHUNKS; len -= size;
} }
bytestream_put_byte(bytestream, 0x00); /* end of image block */ bytestream_put_byte(bytestream, 0x00); /* end of image block */
bytestream_put_byte(bytestream, 0x3b); bytestream_put_byte(bytestream, 0x3b);
...@@ -145,6 +131,12 @@ static av_cold int gif_encode_init(AVCodecContext *avctx) ...@@ -145,6 +131,12 @@ static av_cold int gif_encode_init(AVCodecContext *avctx)
GIFContext *s = avctx->priv_data; GIFContext *s = avctx->priv_data;
avctx->coded_frame = &s->picture; avctx->coded_frame = &s->picture;
s->lzw = av_mallocz(ff_lzw_encode_state_size);
if (!s->lzw)
return AVERROR_NOMEM;
s->buf = av_malloc(avctx->width*avctx->height*2);
if (!s->buf)
return AVERROR_NOMEM;
return 0; return 0;
} }
...@@ -155,15 +147,25 @@ static int gif_encode_frame(AVCodecContext *avctx, unsigned char *outbuf, int bu ...@@ -155,15 +147,25 @@ static int gif_encode_frame(AVCodecContext *avctx, unsigned char *outbuf, int bu
AVFrame *pict = data; AVFrame *pict = data;
AVFrame *const p = (AVFrame *)&s->picture; AVFrame *const p = (AVFrame *)&s->picture;
uint8_t *outbuf_ptr = outbuf; uint8_t *outbuf_ptr = outbuf;
uint8_t *end = outbuf + buf_size;
*p = *pict; *p = *pict;
p->pict_type = FF_I_TYPE; p->pict_type = FF_I_TYPE;
p->key_frame = 1; p->key_frame = 1;
gif_image_write_header(avctx, &outbuf_ptr, (uint32_t *)pict->data[1]); gif_image_write_header(avctx, &outbuf_ptr, (uint32_t *)pict->data[1]);
gif_image_write_image(avctx, &outbuf_ptr, pict->data[0], pict->linesize[0]); gif_image_write_image(avctx, &outbuf_ptr, end, pict->data[0], pict->linesize[0]);
return outbuf_ptr - outbuf; return outbuf_ptr - outbuf;
} }
static int gif_encode_close(AVCodecContext *avctx)
{
GIFContext *s = avctx->priv_data;
av_freep(&s->lzw);
av_freep(&s->buf);
return 0;
}
AVCodec gif_encoder = { AVCodec gif_encoder = {
"gif", "gif",
CODEC_TYPE_VIDEO, CODEC_TYPE_VIDEO,
...@@ -171,7 +173,7 @@ AVCodec gif_encoder = { ...@@ -171,7 +173,7 @@ AVCodec gif_encoder = {
sizeof(GIFContext), sizeof(GIFContext),
gif_encode_init, gif_encode_init,
gif_encode_frame, gif_encode_frame,
NULL, //encode_end, gif_encode_close,
.pix_fmts= (const enum PixelFormat[]){PIX_FMT_RGB8, PIX_FMT_BGR8, PIX_FMT_RGB4_BYTE, PIX_FMT_BGR4_BYTE, PIX_FMT_GRAY8, PIX_FMT_PAL8, PIX_FMT_NONE}, .pix_fmts= (const enum PixelFormat[]){PIX_FMT_RGB8, PIX_FMT_BGR8, PIX_FMT_RGB4_BYTE, PIX_FMT_BGR4_BYTE, PIX_FMT_GRAY8, PIX_FMT_PAL8, PIX_FMT_NONE},
.long_name= NULL_IF_CONFIG_SMALL("GIF (Graphics Interchange Format)"), .long_name= NULL_IF_CONFIG_SMALL("GIF (Graphics Interchange Format)"),
}; };
...@@ -32,6 +32,8 @@ ...@@ -32,6 +32,8 @@
#include "get_bits.h" #include "get_bits.h"
struct PutBitContext;
enum FF_LZW_MODES{ enum FF_LZW_MODES{
FF_LZW_GIF, FF_LZW_GIF,
FF_LZW_TIFF FF_LZW_TIFF
...@@ -52,8 +54,11 @@ void ff_lzw_decode_tail(LZWState *lzw); ...@@ -52,8 +54,11 @@ void ff_lzw_decode_tail(LZWState *lzw);
struct LZWEncodeState; struct LZWEncodeState;
extern const int ff_lzw_encode_state_size; extern const int ff_lzw_encode_state_size;
void ff_lzw_encode_init(struct LZWEncodeState * s, uint8_t * outbuf, int outsize, int maxbits); void ff_lzw_encode_init(struct LZWEncodeState *s, uint8_t *outbuf, int outsize,
int maxbits, enum FF_LZW_MODES mode,
void (*lzw_put_bits)(struct PutBitContext *, int, unsigned int));
int ff_lzw_encode(struct LZWEncodeState * s, const uint8_t * inbuf, int insize); int ff_lzw_encode(struct LZWEncodeState * s, const uint8_t * inbuf, int insize);
int ff_lzw_encode_flush(struct LZWEncodeState * s); int ff_lzw_encode_flush(struct LZWEncodeState *s,
void (*lzw_flush_put_bits)(struct PutBitContext *));
#endif /* AVCODEC_LZW_H */ #endif /* AVCODEC_LZW_H */
...@@ -58,6 +58,8 @@ typedef struct LZWEncodeState { ...@@ -58,6 +58,8 @@ typedef struct LZWEncodeState {
int maxcode; ///< Max value of code int maxcode; ///< Max value of code
int output_bytes; ///< Number of written bytes int output_bytes; ///< Number of written bytes
int last_code; ///< Value of last output code or LZW_PREFIX_EMPTY int last_code; ///< Value of last output code or LZW_PREFIX_EMPTY
enum FF_LZW_MODES mode; ///< TIFF or GIF
void (*put_bits)(PutBitContext *, int, unsigned); ///< GIF is LE while TIFF is BE
}LZWEncodeState; }LZWEncodeState;
...@@ -110,7 +112,7 @@ static inline int hashOffset(const int head) ...@@ -110,7 +112,7 @@ static inline int hashOffset(const int head)
static inline void writeCode(LZWEncodeState * s, int c) static inline void writeCode(LZWEncodeState * s, int c)
{ {
assert(0 <= c && c < 1 << s->bits); assert(0 <= c && c < 1 << s->bits);
put_bits(&s->pb, s->bits, c); s->put_bits(&s->pb, s->bits, c);
} }
...@@ -151,7 +153,7 @@ static inline void addCode(LZWEncodeState * s, uint8_t c, int hash_prefix, int h ...@@ -151,7 +153,7 @@ static inline void addCode(LZWEncodeState * s, uint8_t c, int hash_prefix, int h
s->tabsize++; s->tabsize++;
if (s->tabsize >= 1 << s->bits) if (s->tabsize >= (1 << s->bits) + (s->mode == FF_LZW_GIF))
s->bits++; s->bits++;
} }
...@@ -196,7 +198,9 @@ static int writtenBytes(LZWEncodeState *s){ ...@@ -196,7 +198,9 @@ static int writtenBytes(LZWEncodeState *s){
* @param outsize Size of output buffer * @param outsize Size of output buffer
* @param maxbits Maximum length of code * @param maxbits Maximum length of code
*/ */
void ff_lzw_encode_init(LZWEncodeState * s, uint8_t * outbuf, int outsize, int maxbits) void ff_lzw_encode_init(LZWEncodeState *s, uint8_t *outbuf, int outsize,
int maxbits, enum FF_LZW_MODES mode,
void (*lzw_put_bits)(PutBitContext *, int, unsigned))
{ {
s->clear_code = 256; s->clear_code = 256;
s->end_code = 257; s->end_code = 257;
...@@ -208,6 +212,8 @@ void ff_lzw_encode_init(LZWEncodeState * s, uint8_t * outbuf, int outsize, int m ...@@ -208,6 +212,8 @@ void ff_lzw_encode_init(LZWEncodeState * s, uint8_t * outbuf, int outsize, int m
s->output_bytes = 0; s->output_bytes = 0;
s->last_code = LZW_PREFIX_EMPTY; s->last_code = LZW_PREFIX_EMPTY;
s->bits = 9; s->bits = 9;
s->mode = mode;
s->put_bits = lzw_put_bits;
} }
/** /**
...@@ -250,12 +256,13 @@ int ff_lzw_encode(LZWEncodeState * s, const uint8_t * inbuf, int insize) ...@@ -250,12 +256,13 @@ int ff_lzw_encode(LZWEncodeState * s, const uint8_t * inbuf, int insize)
* @param s LZW state * @param s LZW state
* @return Number of bytes written or -1 on error * @return Number of bytes written or -1 on error
*/ */
int ff_lzw_encode_flush(LZWEncodeState * s) int ff_lzw_encode_flush(LZWEncodeState *s,
void (*lzw_flush_put_bits)(PutBitContext *))
{ {
if (s->last_code != -1) if (s->last_code != -1)
writeCode(s, s->last_code); writeCode(s, s->last_code);
writeCode(s, s->end_code); writeCode(s, s->end_code);
flush_put_bits(&s->pb); lzw_flush_put_bits(&s->pb);
s->last_code = -1; s->last_code = -1;
return writtenBytes(s); return writtenBytes(s);
......
...@@ -32,6 +32,7 @@ ...@@ -32,6 +32,7 @@
#include "tiff.h" #include "tiff.h"
#include "rle.h" #include "rle.h"
#include "lzw.h" #include "lzw.h"
#include "put_bits.h"
#define TIFF_MAX_ENTRY 32 #define TIFF_MAX_ENTRY 32
...@@ -352,7 +353,8 @@ static int encode_frame(AVCodecContext * avctx, unsigned char *buf, ...@@ -352,7 +353,8 @@ static int encode_frame(AVCodecContext * avctx, unsigned char *buf,
for (i = 0; i < s->height; i++) { for (i = 0; i < s->height; i++) {
if (strip_sizes[i / s->rps] == 0) { if (strip_sizes[i / s->rps] == 0) {
if(s->compr == TIFF_LZW){ if(s->compr == TIFF_LZW){
ff_lzw_encode_init(s->lzws, ptr, s->buf_size - (*s->buf - s->buf_start), 12); ff_lzw_encode_init(s->lzws, ptr, s->buf_size - (*s->buf - s->buf_start),
12, FF_LZW_TIFF, put_bits);
} }
strip_offsets[i / s->rps] = ptr - buf; strip_offsets[i / s->rps] = ptr - buf;
} }
...@@ -372,7 +374,7 @@ static int encode_frame(AVCodecContext * avctx, unsigned char *buf, ...@@ -372,7 +374,7 @@ static int encode_frame(AVCodecContext * avctx, unsigned char *buf,
ptr += n; ptr += n;
if(s->compr == TIFF_LZW && (i==s->height-1 || i%s->rps == s->rps-1)){ if(s->compr == TIFF_LZW && (i==s->height-1 || i%s->rps == s->rps-1)){
int ret; int ret;
ret = ff_lzw_encode_flush(s->lzws); ret = ff_lzw_encode_flush(s->lzws, flush_put_bits);
strip_sizes[(i / s->rps )] += ret ; strip_sizes[(i / s->rps )] += ret ;
ptr += ret; ptr += ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment