annotate bethsoftvideo.c @ 4813:a7f058c55491 libavcodec

indention cleanup
author michael
date Sat, 07 Apr 2007 23:16:05 +0000
parents d9cff0d54fdd
children d2fbd9c6df06
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
1 /*
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
2 * Bethesda VID video decoder
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
3 * Copyright (C) 2007 Nicholas Tung
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
4 *
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
5 * This file is part of FFmpeg.
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
6 *
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
7 * FFmpeg is free software; you can redistribute it and/or
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
8 * modify it under the terms of the GNU Lesser General Public
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
9 * License as published by the Free Software Foundation; either
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
10 * version 2.1 of the License, or (at your option) any later version.
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
11 *
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
12 * FFmpeg is distributed in the hope that it will be useful,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
15 * Lesser General Public License for more details.
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
16 *
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
17 * You should have received a copy of the GNU Lesser General Public
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
18 * License along with FFmpeg; if not, write to the Free Software
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
20 */
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
21
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
22 /**
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
23 * @file bethsoftvideo.c
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
24 * @brief Bethesda Softworks VID Video Decoder
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
25 * @author Nicholas Tung [ntung (at. ntung com] (2007-03)
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
26 * @sa http://wiki.multimedia.cx/index.php?title=Bethsoft_VID
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
27 * @sa http://www.svatopluk.com/andux/docs/dfvid.html
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
28 */
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
29
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
30 #include "common.h"
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
31 #include "dsputil.h"
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
32 #include "bethsoftvideo.h"
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
33 #include "bytestream.h"
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
34
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
35 typedef struct BethsoftvidContext {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
36 AVFrame frame;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
37 } BethsoftvidContext;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
38
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
39 static int bethsoftvid_decode_init(AVCodecContext *avctx)
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
40 {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
41 BethsoftvidContext *vid = avctx->priv_data;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
42 vid->frame.reference = 1;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
43 vid->frame.buffer_hints = FF_BUFFER_HINTS_VALID |
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
44 FF_BUFFER_HINTS_PRESERVE | FF_BUFFER_HINTS_REUSABLE;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
45 avctx->pix_fmt = PIX_FMT_PAL8; // palette in vid->frame.data[1]
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
46 av_log(avctx, AV_LOG_DEBUG, "[bethsoftvid video decoder] init\n");
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
47 return 0;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
48 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
49
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
50 static void set_palette(AVFrame * frame, uint8_t * palette_buffer)
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
51 {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
52 uint32_t * palette = (uint32_t *)frame->data[1];
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
53 int a;
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
54 for(a = 0; a < 256; a++){
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
55 palette[a] = AV_RB24(&palette_buffer[a * 3]) * 4; // multiply all colors by 4
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
56 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
57 frame->palette_has_changed = 1;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
58 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
59
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
60 static int bethsoftvid_decode_frame(AVCodecContext *avctx,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
61 void *data, int *data_size,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
62 uint8_t *buf, int buf_size)
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
63 {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
64 BethsoftvidContext * vid = avctx->priv_data;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
65 char block_type;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
66 uint8_t * destination;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
67 uint8_t * frame_end;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
68 int line_remaining = avctx->width; // number of bytes remaining on a line
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
69 const int wrap_to_next_line = vid->frame.linesize[0] - avctx->width;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
70 uint8_t rle_num_bytes;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
71 int yoffset;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
72
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
73 av_log(avctx, AV_LOG_DEBUG, "[bethsoftvid video decoder] decoding frame\n");
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
74
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
75 // reget buffer will copy old data, good for simple difference frames
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
76 if (avctx->reget_buffer(avctx, &vid->frame)) {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
77 av_log(avctx, AV_LOG_ERROR, "reget_buffer() failed\n");
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
78 return -1;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
79 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
80 destination = vid->frame.data[0];
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
81 frame_end = vid->frame.data[0] + vid->frame.linesize[0] * avctx->height;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
82
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
83 switch(block_type = *buf++){
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
84 case PALETTE_BLOCK:
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
85 set_palette(&vid->frame, buf);
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
86 return 0;
4812
d9cff0d54fdd use shorter names for the block type enum
michael
parents: 4803
diff changeset
87 case VIDEO_YOFF_P_FRAME:
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
88 yoffset = bytestream_get_le16(&buf);
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
89 if(yoffset >= avctx->height)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
90 return -1;
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
91 destination += vid->frame.linesize[0] * yoffset;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
92 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
93
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
94 // main code
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
95 while((rle_num_bytes = *buf++)){
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
96 int length = rle_num_bytes & 0x7f;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
97
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
98 // copy any bytes starting at the current position, and ending at the frame width
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
99 while(length > line_remaining){
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
100 if(rle_num_bytes < 0x80)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
101 bytestream_get_buffer(&buf, destination, line_remaining);
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
102 else if(block_type == VIDEO_I_FRAME)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
103 memset(destination, buf[0], line_remaining);
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
104 length -= line_remaining; // decrement the number of bytes to be copied
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
105 destination += line_remaining + wrap_to_next_line; // skip over extra bytes at end of frame
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
106 line_remaining = avctx->width;
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
107 if(destination == frame_end)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
108 goto end;
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
109 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
110
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
111 // copy any remaining bytes after / if line overflows
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
112 if(rle_num_bytes < 0x80)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
113 bytestream_get_buffer(&buf, destination, length);
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
114 else if(block_type == VIDEO_I_FRAME)
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
115 memset(destination, *buf++, length);
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
116 line_remaining -= length;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
117 destination += length;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
118 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
119 end:
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
120
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
121 *data_size = sizeof(AVFrame);
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
122 *(AVFrame*)data = vid->frame;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
123
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
124 return buf_size;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
125 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
126
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
127 static int bethsoftvid_decode_end(AVCodecContext *avctx)
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
128 {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
129 BethsoftvidContext * vid = avctx->priv_data;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
130 av_log(avctx, AV_LOG_DEBUG, "[bethsoftvid video decoder] closing\n");
4813
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
131 if(vid->frame.data[0])
a7f058c55491 indention cleanup
michael
parents: 4812
diff changeset
132 avctx->release_buffer(avctx, &vid->frame);
4803
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
133 return 0;
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
134 }
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
135
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
136 AVCodec bethsoftvid_decoder = {
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
137 .name = "bethsoftvid",
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
138 .type = CODEC_TYPE_VIDEO,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
139 .id = CODEC_ID_BETHSOFTVID,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
140 .priv_data_size = sizeof(BethsoftvidContext),
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
141 .init = bethsoftvid_decode_init,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
142 .close = bethsoftvid_decode_end,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
143 .decode = bethsoftvid_decode_frame,
35e47a6e01e2 Bethsoft VID demuxer and video decoder
diego
parents:
diff changeset
144 };