yading@10
|
1 /*
|
yading@10
|
2 * Copyright (c) 2008 BBC, Anuradha Suraparaju <asuraparaju at gmail dot com >
|
yading@10
|
3 *
|
yading@10
|
4 * This file is part of FFmpeg.
|
yading@10
|
5 *
|
yading@10
|
6 * FFmpeg is free software; you can redistribute it and/or
|
yading@10
|
7 * modify it under the terms of the GNU Lesser General Public
|
yading@10
|
8 * License as published by the Free Software Foundation; either
|
yading@10
|
9 * version 2.1 of the License, or (at your option) any later version.
|
yading@10
|
10 *
|
yading@10
|
11 * FFmpeg is distributed in the hope that it will be useful,
|
yading@10
|
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
yading@10
|
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
yading@10
|
14 * Lesser General Public License for more details.
|
yading@10
|
15 *
|
yading@10
|
16 * You should have received a copy of the GNU Lesser General Public
|
yading@10
|
17 * License along with FFmpeg; if not, write to the Free Software
|
yading@10
|
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
yading@10
|
19 */
|
yading@10
|
20
|
yading@10
|
21 /**
|
yading@10
|
22 * @file
|
yading@10
|
23 * function definitions common to libschroedinger decoder and encoder
|
yading@10
|
24 */
|
yading@10
|
25
|
yading@10
|
26 #include "libschroedinger.h"
|
yading@10
|
27 #include "libavutil/mem.h"
|
yading@10
|
28
|
yading@10
|
29 static const SchroVideoFormatInfo ff_schro_video_format_info[] = {
|
yading@10
|
30 { 640, 480, 24000, 1001},
|
yading@10
|
31 { 176, 120, 15000, 1001},
|
yading@10
|
32 { 176, 144, 25, 2 },
|
yading@10
|
33 { 352, 240, 15000, 1001},
|
yading@10
|
34 { 352, 288, 25, 2 },
|
yading@10
|
35 { 704, 480, 15000, 1001},
|
yading@10
|
36 { 704, 576, 25, 2 },
|
yading@10
|
37 { 720, 480, 30000, 1001},
|
yading@10
|
38 { 720, 576, 25, 1 },
|
yading@10
|
39 { 1280, 720, 60000, 1001},
|
yading@10
|
40 { 1280, 720, 50, 1 },
|
yading@10
|
41 { 1920, 1080, 30000, 1001},
|
yading@10
|
42 { 1920, 1080, 25, 1 },
|
yading@10
|
43 { 1920, 1080, 60000, 1001},
|
yading@10
|
44 { 1920, 1080, 50, 1 },
|
yading@10
|
45 { 2048, 1080, 24, 1 },
|
yading@10
|
46 { 4096, 2160, 24, 1 },
|
yading@10
|
47 };
|
yading@10
|
48
|
yading@10
|
49 static unsigned int get_video_format_idx(AVCodecContext *avctx)
|
yading@10
|
50 {
|
yading@10
|
51 unsigned int ret_idx = 0;
|
yading@10
|
52 unsigned int idx;
|
yading@10
|
53 unsigned int num_formats = sizeof(ff_schro_video_format_info) /
|
yading@10
|
54 sizeof(ff_schro_video_format_info[0]);
|
yading@10
|
55
|
yading@10
|
56 for (idx = 1; idx < num_formats; ++idx) {
|
yading@10
|
57 const SchroVideoFormatInfo *vf = &ff_schro_video_format_info[idx];
|
yading@10
|
58 if (avctx->width == vf->width &&
|
yading@10
|
59 avctx->height == vf->height) {
|
yading@10
|
60 ret_idx = idx;
|
yading@10
|
61 if (avctx->time_base.den == vf->frame_rate_num &&
|
yading@10
|
62 avctx->time_base.num == vf->frame_rate_denom)
|
yading@10
|
63 return idx;
|
yading@10
|
64 }
|
yading@10
|
65 }
|
yading@10
|
66 return ret_idx;
|
yading@10
|
67 }
|
yading@10
|
68
|
yading@10
|
69 void ff_schro_queue_init(FFSchroQueue *queue)
|
yading@10
|
70 {
|
yading@10
|
71 queue->p_head = queue->p_tail = NULL;
|
yading@10
|
72 queue->size = 0;
|
yading@10
|
73 }
|
yading@10
|
74
|
yading@10
|
75 void ff_schro_queue_free(FFSchroQueue *queue, void (*free_func)(void *))
|
yading@10
|
76 {
|
yading@10
|
77 while (queue->p_head)
|
yading@10
|
78 free_func(ff_schro_queue_pop(queue));
|
yading@10
|
79 }
|
yading@10
|
80
|
yading@10
|
81 int ff_schro_queue_push_back(FFSchroQueue *queue, void *p_data)
|
yading@10
|
82 {
|
yading@10
|
83 FFSchroQueueElement *p_new = av_mallocz(sizeof(FFSchroQueueElement));
|
yading@10
|
84
|
yading@10
|
85 if (!p_new)
|
yading@10
|
86 return -1;
|
yading@10
|
87
|
yading@10
|
88 p_new->data = p_data;
|
yading@10
|
89
|
yading@10
|
90 if (!queue->p_head)
|
yading@10
|
91 queue->p_head = p_new;
|
yading@10
|
92 else
|
yading@10
|
93 queue->p_tail->next = p_new;
|
yading@10
|
94 queue->p_tail = p_new;
|
yading@10
|
95
|
yading@10
|
96 ++queue->size;
|
yading@10
|
97 return 0;
|
yading@10
|
98 }
|
yading@10
|
99
|
yading@10
|
100 void *ff_schro_queue_pop(FFSchroQueue *queue)
|
yading@10
|
101 {
|
yading@10
|
102 FFSchroQueueElement *top = queue->p_head;
|
yading@10
|
103
|
yading@10
|
104 if (top) {
|
yading@10
|
105 void *data = top->data;
|
yading@10
|
106 queue->p_head = queue->p_head->next;
|
yading@10
|
107 --queue->size;
|
yading@10
|
108 av_freep(&top);
|
yading@10
|
109 return data;
|
yading@10
|
110 }
|
yading@10
|
111
|
yading@10
|
112 return NULL;
|
yading@10
|
113 }
|
yading@10
|
114
|
yading@10
|
115 /**
|
yading@10
|
116 * Schroedinger video preset table. Ensure that this tables matches up correctly
|
yading@10
|
117 * with the ff_schro_video_format_info table.
|
yading@10
|
118 */
|
yading@10
|
119 static const SchroVideoFormatEnum ff_schro_video_formats[]={
|
yading@10
|
120 SCHRO_VIDEO_FORMAT_CUSTOM ,
|
yading@10
|
121 SCHRO_VIDEO_FORMAT_QSIF ,
|
yading@10
|
122 SCHRO_VIDEO_FORMAT_QCIF ,
|
yading@10
|
123 SCHRO_VIDEO_FORMAT_SIF ,
|
yading@10
|
124 SCHRO_VIDEO_FORMAT_CIF ,
|
yading@10
|
125 SCHRO_VIDEO_FORMAT_4SIF ,
|
yading@10
|
126 SCHRO_VIDEO_FORMAT_4CIF ,
|
yading@10
|
127 SCHRO_VIDEO_FORMAT_SD480I_60 ,
|
yading@10
|
128 SCHRO_VIDEO_FORMAT_SD576I_50 ,
|
yading@10
|
129 SCHRO_VIDEO_FORMAT_HD720P_60 ,
|
yading@10
|
130 SCHRO_VIDEO_FORMAT_HD720P_50 ,
|
yading@10
|
131 SCHRO_VIDEO_FORMAT_HD1080I_60 ,
|
yading@10
|
132 SCHRO_VIDEO_FORMAT_HD1080I_50 ,
|
yading@10
|
133 SCHRO_VIDEO_FORMAT_HD1080P_60 ,
|
yading@10
|
134 SCHRO_VIDEO_FORMAT_HD1080P_50 ,
|
yading@10
|
135 SCHRO_VIDEO_FORMAT_DC2K_24 ,
|
yading@10
|
136 SCHRO_VIDEO_FORMAT_DC4K_24 ,
|
yading@10
|
137 };
|
yading@10
|
138
|
yading@10
|
139 SchroVideoFormatEnum ff_get_schro_video_format_preset(AVCodecContext *avctx)
|
yading@10
|
140 {
|
yading@10
|
141 unsigned int num_formats = sizeof(ff_schro_video_formats) /
|
yading@10
|
142 sizeof(ff_schro_video_formats[0]);
|
yading@10
|
143
|
yading@10
|
144 unsigned int idx = get_video_format_idx(avctx);
|
yading@10
|
145
|
yading@10
|
146 return (idx < num_formats) ? ff_schro_video_formats[idx] :
|
yading@10
|
147 SCHRO_VIDEO_FORMAT_CUSTOM;
|
yading@10
|
148 }
|
yading@10
|
149
|
yading@10
|
150 int ff_get_schro_frame_format (SchroChromaFormat schro_pix_fmt,
|
yading@10
|
151 SchroFrameFormat *schro_frame_fmt)
|
yading@10
|
152 {
|
yading@10
|
153 unsigned int num_formats = sizeof(schro_pixel_format_map) /
|
yading@10
|
154 sizeof(schro_pixel_format_map[0]);
|
yading@10
|
155
|
yading@10
|
156 int idx;
|
yading@10
|
157
|
yading@10
|
158 for (idx = 0; idx < num_formats; ++idx) {
|
yading@10
|
159 if (schro_pixel_format_map[idx].schro_pix_fmt == schro_pix_fmt) {
|
yading@10
|
160 *schro_frame_fmt = schro_pixel_format_map[idx].schro_frame_fmt;
|
yading@10
|
161 return 0;
|
yading@10
|
162 }
|
yading@10
|
163 }
|
yading@10
|
164 return -1;
|
yading@10
|
165 }
|
yading@10
|
166
|
yading@10
|
167 static void free_schro_frame(SchroFrame *frame, void *priv)
|
yading@10
|
168 {
|
yading@10
|
169 AVPicture *p_pic = priv;
|
yading@10
|
170
|
yading@10
|
171 if (!p_pic)
|
yading@10
|
172 return;
|
yading@10
|
173
|
yading@10
|
174 avpicture_free(p_pic);
|
yading@10
|
175 av_freep(&p_pic);
|
yading@10
|
176 }
|
yading@10
|
177
|
yading@10
|
178 SchroFrame *ff_create_schro_frame(AVCodecContext *avctx,
|
yading@10
|
179 SchroFrameFormat schro_frame_fmt)
|
yading@10
|
180 {
|
yading@10
|
181 AVPicture *p_pic;
|
yading@10
|
182 SchroFrame *p_frame;
|
yading@10
|
183 int y_width, uv_width;
|
yading@10
|
184 int y_height, uv_height;
|
yading@10
|
185 int i;
|
yading@10
|
186
|
yading@10
|
187 y_width = avctx->width;
|
yading@10
|
188 y_height = avctx->height;
|
yading@10
|
189 uv_width = y_width >> (SCHRO_FRAME_FORMAT_H_SHIFT(schro_frame_fmt));
|
yading@10
|
190 uv_height = y_height >> (SCHRO_FRAME_FORMAT_V_SHIFT(schro_frame_fmt));
|
yading@10
|
191
|
yading@10
|
192 p_pic = av_mallocz(sizeof(AVPicture));
|
yading@10
|
193 if (!p_pic || avpicture_alloc(p_pic, avctx->pix_fmt, y_width, y_height) < 0) {
|
yading@10
|
194 av_free(p_pic);
|
yading@10
|
195 return NULL;
|
yading@10
|
196 }
|
yading@10
|
197
|
yading@10
|
198 p_frame = schro_frame_new();
|
yading@10
|
199 p_frame->format = schro_frame_fmt;
|
yading@10
|
200 p_frame->width = y_width;
|
yading@10
|
201 p_frame->height = y_height;
|
yading@10
|
202 schro_frame_set_free_callback(p_frame, free_schro_frame, (void *)p_pic);
|
yading@10
|
203
|
yading@10
|
204 for (i = 0; i < 3; ++i) {
|
yading@10
|
205 p_frame->components[i].width = i ? uv_width : y_width;
|
yading@10
|
206 p_frame->components[i].stride = p_pic->linesize[i];
|
yading@10
|
207 p_frame->components[i].height = i ? uv_height : y_height;
|
yading@10
|
208 p_frame->components[i].length =
|
yading@10
|
209 p_frame->components[i].stride * p_frame->components[i].height;
|
yading@10
|
210 p_frame->components[i].data = p_pic->data[i];
|
yading@10
|
211
|
yading@10
|
212 if (i) {
|
yading@10
|
213 p_frame->components[i].v_shift =
|
yading@10
|
214 SCHRO_FRAME_FORMAT_V_SHIFT(p_frame->format);
|
yading@10
|
215 p_frame->components[i].h_shift =
|
yading@10
|
216 SCHRO_FRAME_FORMAT_H_SHIFT(p_frame->format);
|
yading@10
|
217 }
|
yading@10
|
218 }
|
yading@10
|
219
|
yading@10
|
220 return p_frame;
|
yading@10
|
221 }
|