mirror of
https://github.com/outbackdingo/UltraGrid.git
synced 2026-03-22 03:40:30 +00:00
1098 lines
39 KiB
C
1098 lines
39 KiB
C
/*
|
|
* FILE: video_codec.c
|
|
* AUTHORS: Martin Benes <martinbenesh@gmail.com>
|
|
* Lukas Hejtmanek <xhejtman@ics.muni.cz>
|
|
* Petr Holub <hopet@ics.muni.cz>
|
|
* Milos Liska <xliska@fi.muni.cz>
|
|
* Jiri Matela <matela@ics.muni.cz>
|
|
* Dalibor Matura <255899@mail.muni.cz>
|
|
* Ian Wesley-Smith <iwsmith@cct.lsu.edu>
|
|
*
|
|
* Copyright (c) 2005-2010 CESNET z.s.p.o.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, is permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
*
|
|
* This product includes software developed by CESNET z.s.p.o.
|
|
*
|
|
* 4. Neither the name of the CESNET nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING,
|
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
* EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#include "config_unix.h"
|
|
#include "config_win32.h"
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
#include "debug.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "video_codec.h"
|
|
|
|
#define to_fourcc(a,b,c,d) (((uint32_t)(a)) | ((uint32_t)(b)<<8) | ((uint32_t)(c)<<16) | ((uint32_t)(d)<<24))
|
|
|
|
static int get_halign(codec_t codec);
|
|
static void vc_deinterlace_aligned(unsigned char *src, long src_linesize, int lines);
|
|
static void vc_deinterlace_unaligned(unsigned char *src, long src_linesize, int lines);
|
|
static void vc_copylineToUYVY(unsigned char *dst, const unsigned char *src, int dst_len,
|
|
int rshift, int gshift, int bshift, int pix_size);
|
|
|
|
const struct codec_info_t codec_info[] = {
|
|
[RGBA] = {RGBA, "RGBA", to_fourcc('R','G','B','A'), 1, 4.0, TRUE, FALSE, FALSE, "rgba"},
|
|
[UYVY] = {UYVY, "UYVY", to_fourcc('2','v','u','y'), 1, 2, FALSE, FALSE, FALSE, "yuv"},
|
|
[YUYV] = {YUYV, "YUYV", to_fourcc('Y','U','Y','V'), 1, 2, FALSE, FALSE, FALSE, "yuv"},
|
|
[Vuy2] = {Vuy2, "2vuy", to_fourcc('2','V','u','y'), 1, 2, FALSE, FALSE, FALSE, "yuv"},
|
|
[DVS8] = {DVS8, "DVS8", to_fourcc('d','v','s','8'), 1, 2, FALSE, FALSE, FALSE, "yuv"},
|
|
[R10k] = {R10k, "R10k", to_fourcc('R','1','0','k'), 1, 4, TRUE, FALSE, FALSE, "r10k"},
|
|
[v210] = {v210, "v210", to_fourcc('v','2','1','0'), 48, 8.0 / 3.0, FALSE, FALSE, FALSE, "v210"},
|
|
[DVS10] = {DVS10, "DVS10", to_fourcc('D','S','1','0'), 48, 8.0 / 3.0, FALSE, FALSE, FALSE, "dvs10"},
|
|
[DXT1] = {DXT1, "DXT1", to_fourcc('D','X','T','1'), 1, 0.5, TRUE, TRUE, FALSE, "dxt1"},
|
|
[DXT1_YUV] = {DXT1_YUV, "DXT1 YUV", to_fourcc('D','X','T','Y'), 1, 0.5, FALSE, TRUE, FALSE, "dxt1y"}, /* packet YCbCr inside DXT1 channels */
|
|
[DXT5] = {DXT5, "DXT5", to_fourcc('D','X','T','5'), 1, 1.0, FALSE, TRUE, FALSE, "yog"},/* DXT5 YCoCg */
|
|
[RGB] = {RGB, "RGB", to_fourcc('R','G','B','2'), 1, 3.0, TRUE, FALSE, FALSE, "rgb"},
|
|
[DPX10] = {DPX10, "DPX10", to_fourcc('D','P','1','0'), 1, 4.0, TRUE, FALSE, FALSE, "dpx"},
|
|
[JPEG] = {JPEG, "JPEG", to_fourcc('J','P','E','G'), 0, 0.0, FALSE, TRUE, FALSE, "jpg"},
|
|
[RAW] = {RAW, "raw", to_fourcc('r','a','w','s'), 0, 1.0, FALSE, TRUE, FALSE, "raw"}, /* raw SDI */
|
|
[H264] = {H264, "H.264", to_fourcc('A','V','C','1'), 0, 1.0, FALSE, TRUE, TRUE, "h264"},
|
|
[MJPG] = {MJPG, "MJPEG", to_fourcc('M','J','P','G'), 0, 1.0, FALSE, TRUE, FALSE, "jpg"},
|
|
[VP8] = {VP8, "VP8", to_fourcc('V','P','8','0'), 0, 1.0, FALSE, TRUE, TRUE, "vp8"},
|
|
[BGR] = {BGR, "BGR", to_fourcc('B','G','R','2'), 1, 3.0, TRUE, FALSE, FALSE, "bgr"},
|
|
{(codec_t) 0, NULL, 0, 0, 0.0, FALSE, FALSE, FALSE, NULL}
|
|
};
|
|
|
|
/* take care that UYVY is alias for both 2vuy and dvs8, do not use
|
|
* the further two and refer only to UYVY!! */
|
|
|
|
/* Also note that this is a priority list - is choosen first one that
|
|
* matches input codec and one of the supported output codec, so eg.
|
|
* list 10b->10b earlier to 10b->8b etc. */
|
|
const struct line_decode_from_to line_decoders[] = {
|
|
{ RGBA, RGBA, vc_copylineRGBA},
|
|
{ RGB, RGB, vc_copylineRGB},
|
|
{ DVS10, v210, (decoder_t) vc_copylineDVS10toV210},
|
|
{ DVS10, UYVY, (decoder_t) vc_copylineDVS10},
|
|
{ R10k, RGBA, vc_copyliner10k},
|
|
{ v210, UYVY, (decoder_t) vc_copylinev210},
|
|
{ YUYV, UYVY, (decoder_t) vc_copylineYUYV},
|
|
{ RGBA, RGB, (decoder_t) vc_copylineRGBAtoRGB},
|
|
{ RGB, RGBA, vc_copylineRGBtoRGBA},
|
|
{ DPX10, RGBA, vc_copylineDPX10toRGBA},
|
|
{ DPX10, RGB, (decoder_t) vc_copylineDPX10toRGB},
|
|
{ RGB, UYVY, (decoder_t) vc_copylineRGBtoUYVY},
|
|
{ BGR, RGB, (decoder_t) vc_copylineBGRtoRGB},
|
|
{ (codec_t) 0, (codec_t) 0, NULL }
|
|
};
|
|
|
|
struct alternate_fourcc {
|
|
uint32_t alias;
|
|
uint32_t primary_fcc;
|
|
};
|
|
|
|
const struct alternate_fourcc fourcc_aliases[] = {
|
|
// the following two are here because it was sent with wrong endiannes in past
|
|
{to_fourcc('A', 'B', 'G', 'R'), to_fourcc('R', 'G', 'B', 'A')},
|
|
{to_fourcc('2', 'B', 'G', 'R'), to_fourcc('R', 'G', 'B', '2')},
|
|
// following ones are rather for further compatibility (proposed codecs rename)
|
|
{to_fourcc('U', 'Y', 'V', 'Y'), to_fourcc('2', 'v', 'u', 'y')},
|
|
{to_fourcc('M', 'J', 'P', 'G'), to_fourcc('J', 'P', 'E', 'G')},
|
|
{to_fourcc('D', 'V', 'S', '8'), to_fourcc('d', 'v', 's', '8')},
|
|
{0,0}
|
|
};
|
|
|
|
void show_codec_help(char *module)
|
|
{
|
|
printf("\tSupported codecs (%s):\n", module);
|
|
|
|
printf("\t\t8bits\n");
|
|
|
|
printf("\t\t\t'RGBA' - Red Green Blue Alpha 32bit\n");
|
|
printf("\t\t\t'RGB' - Red Green Blue 24bit\n");
|
|
printf("\t\t\t'UYVY' - YUV 4:2:2\n");
|
|
printf("\t\t\t'2vuy' - YUV 4:2:2\n");
|
|
printf("\t\t\t'DVS8' - Centaurus 8bit YUV 4:2:2\n");
|
|
|
|
printf("\t\t10bits\n");
|
|
if (strcmp(module, "dvs") != 0) {
|
|
printf("\t\t\t'R10k' - RGB 4:4:4\n");
|
|
printf("\t\t\t'v210' - YUV 4:2:2\n");
|
|
}
|
|
printf("\t\t\t'DVS10' - Centaurus 10bit YUV 4:2:2\n");
|
|
}
|
|
|
|
double get_bpp(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].bpp;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
uint32_t get_fourcc(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].fcc;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
const char * get_codec_name(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].name;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
uint32_t get_fcc_from_codec(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].fcc;
|
|
i++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
codec_t get_codec_from_fcc(uint32_t fourcc)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (fourcc == codec_info[i].fcc)
|
|
return codec_info[i].codec;
|
|
i++;
|
|
}
|
|
|
|
// try to look through aliases
|
|
i = 0;
|
|
while (fourcc_aliases[i].alias != 0) {
|
|
if (fourcc == fourcc_aliases[i].alias) {
|
|
int j = 0;
|
|
while (codec_info[j].name != NULL) {
|
|
if (fourcc_aliases[i].primary_fcc == codec_info[j].fcc)
|
|
return codec_info[j].codec;
|
|
j++;
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
return (codec_t) -1;
|
|
}
|
|
|
|
const char *get_codec_file_extension(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].file_extension;
|
|
i++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int is_codec_opaque(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].opaque;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int is_codec_interframe(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].interframe;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int get_halign(codec_t codec)
|
|
{
|
|
int i = 0;
|
|
|
|
while (codec_info[i].name != NULL) {
|
|
if (codec == codec_info[i].codec)
|
|
return codec_info[i].h_align;
|
|
i++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int get_haligned(int width_pixels, codec_t codec)
|
|
{
|
|
int h_align = get_halign(codec);
|
|
return ((width_pixels + h_align - 1) / h_align) * h_align;
|
|
}
|
|
|
|
int vc_get_linesize(unsigned int width, codec_t codec)
|
|
{
|
|
if (codec_info[codec].h_align) {
|
|
width =
|
|
((width + codec_info[codec].h_align -
|
|
1) / codec_info[codec].h_align) *
|
|
codec_info[codec].h_align;
|
|
}
|
|
return width * codec_info[codec].bpp;
|
|
}
|
|
|
|
/* linear blend deinterlace */
|
|
void vc_deinterlace(unsigned char *src, long src_linesize, int lines)
|
|
{
|
|
if(((long int) src & 0x0F) == 0 && src_linesize % 16 == 0) {
|
|
vc_deinterlace_aligned(src, src_linesize, lines);
|
|
} else {
|
|
vc_deinterlace_unaligned(src, src_linesize, lines);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Aligned version of deinterlace filter
|
|
*
|
|
* @param src 16-byte aligned buffer
|
|
*/
|
|
static void vc_deinterlace_aligned(unsigned char *src, long src_linesize, int lines)
|
|
{
|
|
int i, j;
|
|
long pitch = src_linesize;
|
|
register long pitch2 = pitch * 2;
|
|
unsigned char *bline1, *bline2, *bline3;
|
|
register unsigned char *line1, *line2, *line3;
|
|
|
|
bline1 = src;
|
|
bline2 = src + pitch;
|
|
bline3 = src + 3 * pitch;
|
|
for (i = 0; i < src_linesize; i += 16) {
|
|
/* preload first two lines */
|
|
asm volatile ("movdqa (%0), %%xmm0\n"
|
|
"movdqa (%1), %%xmm1\n"::"r" ((unsigned long *)(void *)
|
|
bline1),
|
|
"r"((unsigned long *)(void *) bline2));
|
|
line1 = bline2;
|
|
line2 = bline2 + pitch;
|
|
line3 = bline3;
|
|
for (j = 0; j < lines - 4; j += 2) {
|
|
asm volatile ("movdqa (%1), %%xmm2\n"
|
|
"pavgb %%xmm2, %%xmm0\n"
|
|
"pavgb %%xmm1, %%xmm0\n"
|
|
"movdqa (%2), %%xmm1\n"
|
|
"movdqa %%xmm0, (%0)\n"
|
|
"pavgb %%xmm1, %%xmm0\n"
|
|
"pavgb %%xmm2, %%xmm0\n"
|
|
"movdqa %%xmm0, (%1)\n"::"r" ((unsigned
|
|
long *) (void *) line1),
|
|
"r"((unsigned long *) (void *) line2),
|
|
"r"((unsigned long *) (void *) line3)
|
|
);
|
|
line1 += pitch2;
|
|
line2 += pitch2;
|
|
line3 += pitch2;
|
|
}
|
|
bline1 += 16;
|
|
bline2 += 16;
|
|
bline3 += 16;
|
|
}
|
|
}
|
|
/**
|
|
* Unaligned version of deinterlace filter
|
|
*
|
|
* @param src 16-byte aligned buffer
|
|
*/
|
|
static void vc_deinterlace_unaligned(unsigned char *src, long src_linesize, int lines)
|
|
{
|
|
int i, j;
|
|
long pitch = src_linesize;
|
|
register long pitch2 = pitch * 2;
|
|
unsigned char *bline1, *bline2, *bline3;
|
|
register unsigned char *line1, *line2, *line3;
|
|
|
|
bline1 = src;
|
|
bline2 = src + pitch;
|
|
bline3 = src + 3 * pitch;
|
|
for (i = 0; i < src_linesize; i += 16) {
|
|
/* preload first two lines */
|
|
asm volatile ("movdqu (%0), %%xmm0\n"
|
|
"movdqu (%1), %%xmm1\n"::"r" (bline1),
|
|
"r" (bline2));
|
|
line1 = bline2;
|
|
line2 = bline2 + pitch;
|
|
line3 = bline3;
|
|
for (j = 0; j < lines - 4; j += 2) {
|
|
asm volatile ("movdqu (%1), %%xmm2\n"
|
|
"pavgb %%xmm2, %%xmm0\n"
|
|
"pavgb %%xmm1, %%xmm0\n"
|
|
"movdqu (%2), %%xmm1\n"
|
|
"movdqu %%xmm0, (%0)\n"
|
|
"pavgb %%xmm1, %%xmm0\n"
|
|
"pavgb %%xmm2, %%xmm0\n"
|
|
"movdqu %%xmm0, (%1)\n"::"r" (line1),
|
|
"r" (line2),
|
|
"r" (line3)
|
|
);
|
|
line1 += pitch2;
|
|
line2 += pitch2;
|
|
line3 += pitch2;
|
|
}
|
|
bline1 += 16;
|
|
bline2 += 16;
|
|
bline3 += 16;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts v210 to UYVY
|
|
*
|
|
* @param[out] dst 4-byte aligned output buffer where UYVY will be stored
|
|
* @param[in] src 4-byte aligned input buffer containing v210 (by definition of v210
|
|
* should be even aligned to 16B boundary)
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void vc_copylinev210(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
struct {
|
|
unsigned a:10;
|
|
unsigned b:10;
|
|
unsigned c:10;
|
|
unsigned p1:2;
|
|
} const *s;
|
|
register uint32_t *d;
|
|
register uint32_t tmp;
|
|
|
|
d = (uint32_t *)(void *) dst;
|
|
s = (const void *)src;
|
|
|
|
while (dst_len >= 12) {
|
|
tmp = (s->a >> 2) | (s->b >> 2) << 8 | (((s)->c >> 2) << 16);
|
|
s++;
|
|
*(d++) = tmp | ((s->a >> 2) << 24);
|
|
tmp = (s->b >> 2) | (((s)->c >> 2) << 8);
|
|
s++;
|
|
*(d++) = tmp | ((s->a >> 2) << 16) | ((s->b >> 2) << 24);
|
|
tmp = (s->c >> 2);
|
|
s++;
|
|
*(d++) =
|
|
tmp | ((s->a >> 2) << 8) | ((s->b >> 2) << 16) |
|
|
((s->c >> 2) << 24);
|
|
s++;
|
|
|
|
dst_len -= 12;
|
|
}
|
|
if (dst_len >= 4) {
|
|
tmp = (s->a >> 2) | (s->b >> 2) << 8 | (((s)->c >> 2) << 16);
|
|
s++;
|
|
*(d++) = tmp | ((s->a >> 2) << 24);
|
|
}
|
|
if (dst_len >= 8) {
|
|
tmp = (s->b >> 2) | (((s)->c >> 2) << 8);
|
|
s++;
|
|
*(d++) = tmp | ((s->a >> 2) << 16) | ((s->b >> 2) << 24);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from YUYV to UYVY.
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in YUYV
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void vc_copylineYUYV(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
register uint32_t *d;
|
|
register const uint32_t *s;
|
|
const uint32_t * const end = (uint32_t *)(void *) dst + dst_len / 4;
|
|
|
|
uint32_t mask[4] = {
|
|
0xff00ff00ul,
|
|
0xff00ff00ul,
|
|
0xff00ff00ul,
|
|
0xff00ff00ul};
|
|
|
|
d = (uint32_t *)(void *) dst;
|
|
s = (const uint32_t *)(const void *) src;
|
|
|
|
assert(dst_len % 4 == 0);
|
|
|
|
if((dst_len % 16 == 0)) {
|
|
asm("movdqa (%0), %%xmm4\n"
|
|
"movdqa %%xmm4, %%xmm5\n"
|
|
"psrldq $1, %%xmm5\n"
|
|
: :"r"(mask));
|
|
while(d < end) {
|
|
asm volatile ("movdqu (%0), %%xmm0\n"
|
|
"movdqu %%xmm0, %%xmm1\n"
|
|
"pand %%xmm4, %%xmm0\n"
|
|
"psrldq $1, %%xmm0\n"
|
|
"pand %%xmm5, %%xmm1\n"
|
|
"pslldq $1, %%xmm1\n"
|
|
"por %%xmm0, %%xmm1\n"
|
|
"movdqu %%xmm1, (%1)\n"::"r" (s), "r"(d));
|
|
s += 4;
|
|
d += 4;
|
|
}
|
|
} else {
|
|
while(d < end) {
|
|
register uint32_t tmp = *s;
|
|
*d = ((tmp & 0x00ff0000) << 8) | ((tmp & 0xff000000) >> 8) |
|
|
((tmp & 0x000000ff) << 8) | ((tmp & 0x0000ff00) >> 8);
|
|
s++;
|
|
d++;
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from R10k to RGBA
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in R10k
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void
|
|
vc_copyliner10k(unsigned char *dst, const unsigned char *src, int len, int rshift,
|
|
int gshift, int bshift)
|
|
{
|
|
struct {
|
|
unsigned r:8;
|
|
|
|
unsigned gh:6;
|
|
unsigned p1:2;
|
|
|
|
unsigned bh:4;
|
|
unsigned p2:2;
|
|
unsigned gl:2;
|
|
|
|
unsigned p3:2;
|
|
unsigned p4:2;
|
|
unsigned bl:4;
|
|
} const *s;
|
|
register uint32_t *d;
|
|
register uint32_t tmp;
|
|
|
|
d = (uint32_t *)(void *) dst;
|
|
s = (const void *)(const void *) src;
|
|
|
|
while (len > 0) {
|
|
tmp =
|
|
(s->
|
|
r << rshift) | (((s->gh << 2) | s->
|
|
gl) << gshift) | (((s->bh << 4) | s->
|
|
bl) << bshift);
|
|
s++;
|
|
*(d++) = tmp;
|
|
tmp =
|
|
(s->
|
|
r << rshift) | (((s->gh << 2) | s->
|
|
gl) << gshift) | (((s->bh << 4) | s->
|
|
bl) << bshift);
|
|
s++;
|
|
*(d++) = tmp;
|
|
tmp =
|
|
(s->
|
|
r << rshift) | (((s->gh << 2) | s->
|
|
gl) << gshift) | (((s->bh << 4) | s->
|
|
bl) << bshift);
|
|
s++;
|
|
*(d++) = tmp;
|
|
tmp =
|
|
(s->
|
|
r << rshift) | (((s->gh << 2) | s->
|
|
gl) << gshift) | (((s->bh << 4) | s->
|
|
bl) << bshift);
|
|
s++;
|
|
*(d++) = tmp;
|
|
len -= 16;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Changes color channels' order in RGBA
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in RGBA
|
|
* @param[in] dst length of data that should be writen to dst buffer (in bytes)
|
|
* @param[in] rshift destination rshift
|
|
* @param[in] rshift destination gshift
|
|
* @param[in] rshift destination bshift
|
|
*/
|
|
void
|
|
vc_copylineRGBA(unsigned char *dst, const unsigned char *src, int len, int rshift,
|
|
int gshift, int bshift)
|
|
{
|
|
register uint32_t *d = (uint32_t *)(void *) dst;
|
|
register const uint32_t *s = (const uint32_t *)(const void *) src;
|
|
register uint32_t tmp;
|
|
|
|
if (rshift == 0 && gshift == 8 && bshift == 16) {
|
|
memcpy(dst, src, len);
|
|
} else {
|
|
while (len > 0) {
|
|
register unsigned int r, g, b;
|
|
tmp = *(s++);
|
|
r = tmp & 0xff;
|
|
g = (tmp >> 8) & 0xff;
|
|
b = (tmp >> 16) & 0xff;
|
|
tmp = (r << rshift) | (g << gshift) | (b << bshift);
|
|
*(d++) = tmp;
|
|
tmp = *(s++);
|
|
r = tmp & 0xff;
|
|
g = (tmp >> 8) & 0xff;
|
|
b = (tmp >> 16) & 0xff;
|
|
tmp = (r << rshift) | (g << gshift) | (b << bshift);
|
|
*(d++) = tmp;
|
|
tmp = *(s++);
|
|
r = tmp & 0xff;
|
|
g = (tmp >> 8) & 0xff;
|
|
b = (tmp >> 16) & 0xff;
|
|
tmp = (r << rshift) | (g << gshift) | (b << bshift);
|
|
*(d++) = tmp;
|
|
tmp = *(s++);
|
|
r = tmp & 0xff;
|
|
g = (tmp >> 8) & 0xff;
|
|
b = (tmp >> 16) & 0xff;
|
|
tmp = (r << rshift) | (g << gshift) | (b << bshift);
|
|
*(d++) = tmp;
|
|
len -= 16;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from DVS10 to v210
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in DVS10
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void vc_copylineDVS10toV210(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
unsigned int *d;
|
|
const unsigned int *s1;
|
|
register unsigned int a,b;
|
|
d = (unsigned int *)(void *) dst;
|
|
s1 = (const unsigned int *)(const void *) src;
|
|
|
|
while(dst_len > 0) {
|
|
a = b = *s1++;
|
|
b = ((b >> 24) * 0x00010101) & 0x00300c03;
|
|
a <<= 2;
|
|
b |= a & (0xff<<2);
|
|
a <<= 2;
|
|
b |= a & (0xff00<<4);
|
|
a <<= 2;
|
|
b |= a & (0xff0000<<6);
|
|
*d++ = b;
|
|
dst_len -= 4;
|
|
}
|
|
}
|
|
|
|
/* convert 10bits Cb Y Cr A Y Cb Y A to 8bits Cb Y Cr Y Cb Y */
|
|
|
|
/* TODO: undo it - currently this decoder is broken */
|
|
#if 0 /* !(HAVE_MACOSX || HAVE_32B_LINUX) */
|
|
|
|
void vc_copylineDVS10(unsigned char *dst, unsigned char *src, int src_len)
|
|
{
|
|
register unsigned char *_d = dst, *_s = src;
|
|
|
|
while (src_len > 0) {
|
|
|
|
asm("movd %0, %%xmm4\n": :"r"(0xffffff));
|
|
|
|
asm volatile ("movdqa (%0), %%xmm0\n"
|
|
"movdqa 16(%0), %%xmm5\n"
|
|
"movdqa %%xmm0, %%xmm1\n"
|
|
"movdqa %%xmm0, %%xmm2\n"
|
|
"movdqa %%xmm0, %%xmm3\n"
|
|
"pand %%xmm4, %%xmm0\n"
|
|
"movdqa %%xmm5, %%xmm6\n"
|
|
"movdqa %%xmm5, %%xmm7\n"
|
|
"movdqa %%xmm5, %%xmm8\n"
|
|
"pand %%xmm4, %%xmm5\n"
|
|
"pslldq $4, %%xmm4\n"
|
|
"pand %%xmm4, %%xmm1\n"
|
|
"pand %%xmm4, %%xmm6\n"
|
|
"pslldq $4, %%xmm4\n"
|
|
"psrldq $1, %%xmm1\n"
|
|
"psrldq $1, %%xmm6\n"
|
|
"pand %%xmm4, %%xmm2\n"
|
|
"pand %%xmm4, %%xmm7\n"
|
|
"pslldq $4, %%xmm4\n"
|
|
"psrldq $2, %%xmm2\n"
|
|
"psrldq $2, %%xmm7\n"
|
|
"pand %%xmm4, %%xmm3\n"
|
|
"pand %%xmm4, %%xmm8\n"
|
|
"por %%xmm1, %%xmm0\n"
|
|
"psrldq $3, %%xmm3\n"
|
|
"psrldq $3, %%xmm8\n"
|
|
"por %%xmm2, %%xmm0\n"
|
|
"por %%xmm6, %%xmm5\n"
|
|
"por %%xmm3, %%xmm0\n"
|
|
"por %%xmm7, %%xmm5\n"
|
|
"movdq2q %%xmm0, %%mm0\n"
|
|
"por %%xmm8, %%xmm5\n"
|
|
"movdqa %%xmm5, %%xmm1\n"
|
|
"pslldq $12, %%xmm5\n"
|
|
"psrldq $4, %%xmm1\n"
|
|
"por %%xmm5, %%xmm0\n"
|
|
"psrldq $8, %%xmm0\n"
|
|
"movq %%mm0, (%1)\n"
|
|
"movdq2q %%xmm0, %%mm1\n"
|
|
"movdq2q %%xmm1, %%mm2\n"
|
|
"movq %%mm1, 8(%1)\n"
|
|
"movq %%mm2, 16(%1)\n"::"r" (_s), "r"(_d));
|
|
_s += 32;
|
|
_d += 24;
|
|
src_len -= 32;
|
|
}
|
|
}
|
|
|
|
#else
|
|
|
|
/**
|
|
* Converts from DVS10 to UYVY
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in UYVY
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void vc_copylineDVS10(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
int src_len = dst_len / 1.5; /* right units */
|
|
register const uint64_t *s;
|
|
register uint64_t *d;
|
|
|
|
register uint64_t a1, a2, a3, a4;
|
|
|
|
d = (uint64_t *)(void *) dst;
|
|
s = (const uint64_t *)(const void *) src;
|
|
|
|
while (src_len > 0) {
|
|
a1 = *(s++);
|
|
a2 = *(s++);
|
|
a3 = *(s++);
|
|
a4 = *(s++);
|
|
|
|
a1 = (a1 & 0xffffff) | ((a1 >> 8) & 0xffffff000000LL);
|
|
a2 = (a2 & 0xffffff) | ((a2 >> 8) & 0xffffff000000LL);
|
|
a3 = (a3 & 0xffffff) | ((a3 >> 8) & 0xffffff000000LL);
|
|
a4 = (a4 & 0xffffff) | ((a4 >> 8) & 0xffffff000000LL);
|
|
|
|
*(d++) = a1 | (a2 << 48); /* 0xa2|a2|a1|a1|a1|a1|a1|a1 */
|
|
*(d++) = (a2 >> 16) | (a3 << 32); /* 0xa3|a3|a3|a3|a2|a2|a2|a2 */
|
|
*(d++) = (a3 >> 32) | (a4 << 16); /* 0xa4|a4|a4|a4|a4|a4|a3|a3 */
|
|
|
|
src_len -= 16;
|
|
}
|
|
}
|
|
|
|
#endif /* !(HAVE_MACOSX || HAVE_32B_LINUX) */
|
|
|
|
void vc_copylineRGB(unsigned char *dst, const unsigned char *src, int dst_len, int rshift, int gshift, int bshift)
|
|
{
|
|
register unsigned int r, g, b;
|
|
union {
|
|
unsigned int out;
|
|
unsigned char c[4];
|
|
} u;
|
|
|
|
if (rshift == 0 && gshift == 8 && bshift == 16) {
|
|
memcpy(dst, src, dst_len);
|
|
} else {
|
|
while(dst_len > 0) {
|
|
r = *src++;
|
|
g = *src++;
|
|
b = *src++;
|
|
u.out = (r << rshift) | (g << gshift) | (b << bshift);
|
|
*dst++ = u.c[0];
|
|
*dst++ = u.c[1];
|
|
*dst++ = u.c[2];
|
|
dst_len -= 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from RGBA to RGB
|
|
*
|
|
* @param[out] dst 4B-aligned buffer that will contain result
|
|
* @param[in] src 4B-aligned buffer containing pixels in RGB
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
*/
|
|
void vc_copylineRGBAtoRGB(unsigned char *dst2, const unsigned char *src2, int dst_len)
|
|
{
|
|
register const uint32_t * src = (const uint32_t *)(const void *) src2;
|
|
register uint32_t * dst = (uint32_t *)(void *) dst2;
|
|
while(dst_len > 0) {
|
|
register uint32_t in1 = *src++;
|
|
register uint32_t in2 = *src++;
|
|
register uint32_t in3 = *src++;
|
|
register uint32_t in4 = *src++;
|
|
*dst++ = ((in2 & 0xff) << 24) | (in1 & 0xffffff);
|
|
*dst++ = ((in3 & 0xffff) << 16) | ((in2 & 0xffff00) >> 8);
|
|
*dst++ = ((in4 & 0xffffff) << 8) | ((in3 & 0xff0000) >> 16) ;
|
|
|
|
dst_len -= 12;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from RGBA to RGB. Channels in RGBA can be differently ordered.
|
|
*
|
|
* @param[out] dst2 4B-aligned buffer that will contain result
|
|
* @param[in] src2 4B-aligned buffer containing pixels in RGB
|
|
* @param[in] dst_len length of data that should be writen to dst buffer (in bytes)
|
|
* @param[in] rshift source rshift
|
|
* @param[in] gshift source gshift
|
|
* @param[in] bshift source bshift
|
|
*/
|
|
void vc_copylineRGBAtoRGBwithShift(unsigned char *dst2, const unsigned char *src2, int dst_len, int rshift, int gshift, int bshift)
|
|
{
|
|
register const uint32_t * src = (const uint32_t *)(const void *) src2;
|
|
register uint32_t * dst = (uint32_t *)(void *) dst2;
|
|
while(dst_len > 0) {
|
|
register uint32_t in1 = *src++;
|
|
register uint32_t in2 = *src++;
|
|
register uint32_t in3 = *src++;
|
|
register uint32_t in4 = *src++;
|
|
|
|
*dst++ = ((in2 >> rshift)) << 24 |
|
|
((in1 >> bshift) & 0xff) << 16 |
|
|
((in1 >> gshift) & 0xff) << 8 |
|
|
((in1 >> rshift) & 0xff);
|
|
*dst++ = ((in3 >> gshift)) << 24 |
|
|
((in3 >> rshift) & 0xff) << 16 |
|
|
((in2 >> bshift) & 0xff) << 8 |
|
|
((in2 >> gshift) & 0xff);
|
|
*dst++ = ((in4 >> bshift)) << 24 |
|
|
((in4 >> gshift) & 0xff) << 16 |
|
|
((in4 >> rshift) & 0xff) << 8 |
|
|
((in3 >> bshift) & 0xff);
|
|
|
|
dst_len -= 12;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts from AGBR to RGB
|
|
*
|
|
* @see vc_copylineRGBAtoRGBwithShift
|
|
* @see vc_copylineRGBAtoRGB
|
|
*/
|
|
void vc_copylineABGRtoRGB(unsigned char *dst2, const unsigned char *src2, int dst_len)
|
|
{
|
|
register const uint32_t * src = (const uint32_t *)(const void *) src2;
|
|
register uint32_t * dst = (uint32_t *)(void *) dst2;
|
|
while(dst_len > 0) {
|
|
register uint32_t in1 = *src++;
|
|
register uint32_t in2 = *src++;
|
|
register uint32_t in3 = *src++;
|
|
register uint32_t in4 = *src++;
|
|
|
|
*dst++ = (in2 & 0xff0000) << 8 |
|
|
(in1 & 0xff) << 16 |
|
|
(in1 & 0xff00) |
|
|
(in1 & 0xff0000) >> 16;
|
|
*dst++ = (in3 & 0xff00) << 16 |
|
|
(in3 & 0xff0000) |
|
|
(in2 & 0xff) << 8 |
|
|
(in2 & 0xff00) >> 8;
|
|
*dst++ = (in4 & 0xff) << 24 |
|
|
(in4 & 0xff00) << 8 |
|
|
(in4 & 0xff0000) >> 8 |
|
|
(in3 & 0xff);
|
|
|
|
dst_len -= 12;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts RGB to RGBA
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src 4B-aligned input buffer
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
* @param[in] rshift RGBA rshift
|
|
* @param[in] gshift RGBA gshift
|
|
* @param[in] bshift RGBA gshift
|
|
*/
|
|
void vc_copylineRGBtoRGBA(unsigned char *dst, const unsigned char *src, int dst_len, int rshift, int gshift, int bshift)
|
|
{
|
|
register unsigned int r, g, b;
|
|
register uint32_t *d = (uint32_t *)(void *) dst;
|
|
|
|
while(dst_len > 0) {
|
|
r = *src++;
|
|
g = *src++;
|
|
b = *src++;
|
|
|
|
*d++ = (r << rshift) | (g << gshift) | (b << bshift);
|
|
dst_len -= 4;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param rshift rshift in bytes
|
|
* @param gshift gshift in bytes
|
|
* @param bshift bshift in bytes
|
|
*/
|
|
static void vc_copylineToUYVY(unsigned char *dst, const unsigned char *src, int dst_len,
|
|
int rshift, int gshift, int bshift, int pix_size) {
|
|
register int r, g, b;
|
|
register int y1, y2, u ,v;
|
|
register uint32_t *d = (uint32_t *)(void *) dst;
|
|
|
|
while(dst_len > 0) {
|
|
r = *(src + rshift);
|
|
g = *(src + gshift);
|
|
b = *(src + bshift);
|
|
src += pix_size;
|
|
y1 = 19595 * r + 38469 * g + 7471 * b;
|
|
u = -9642 * r -18931 * g + 28573 * b;
|
|
v = 40304 * r - 33750 * g - 6554 * b;
|
|
r = *(src + rshift);
|
|
g = *(src + gshift);
|
|
b = *(src + bshift);
|
|
src += pix_size;
|
|
y2 = 19595 * r + 38469 * g + 7471 * b;
|
|
u += -9642 * r -18931 * g + 28573 * b;
|
|
v += 40304 * r - 33750 * g - 6554 * b;
|
|
u = u / 2 + (1<<23);
|
|
v = v / 2 + (1<<23);
|
|
|
|
*d++ = (min(max(y2, 0), (1<<24)-1) >> 16) << 24 |
|
|
(min(max(v, 0), (1<<24)-1) >> 16) << 16 |
|
|
(min(max(y1, 0), (1<<24)-1) >> 16) << 8 |
|
|
(min(max(u, 0), (1<<24)-1) >> 16);
|
|
dst_len -= 4;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts UYVY to RGB
|
|
* Uses Rec. 709 with standard SDI ceiling and floor
|
|
* @todo make it faster if needed
|
|
*/
|
|
void vc_copylineUYVYtoRGB(unsigned char *dst, const unsigned char *src, int dst_len) {
|
|
while(dst_len > 0) {
|
|
register int y1, y2, u ,v;
|
|
u = *src++;
|
|
y1 = *src++;
|
|
v = *src++;
|
|
y2 = *src++;
|
|
*dst++ = min(max(1.164*(y1 - 16) + 1.793*(v - 128), 0), 255);
|
|
*dst++ = min(max(1.164*(y1 - 16) - 0.534*(v - 128) - 0.213*(u - 128), 0), 255);
|
|
*dst++ = min(max(1.164*(y1 - 16) + 2.115*(u - 128), 0), 255);
|
|
*dst++ = min(max(1.164*(y2 - 16) + 1.793*(v - 128), 0), 255);
|
|
*dst++ = min(max(1.164*(y2 - 16) - 0.534*(v - 128) - 0.213*(u - 128), 0), 255);
|
|
*dst++ = min(max(1.164*(y2 - 16) + 2.115*(u - 128), 0), 255);
|
|
|
|
dst_len -= 6;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts RGB to UYVY.
|
|
* Uses full scale Rec. 601 YUV (aka JPEG)
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src buffer
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
*/
|
|
void vc_copylineRGBtoUYVY(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
vc_copylineToUYVY(dst, src, dst_len, 0, 1, 2, 3);
|
|
}
|
|
|
|
/**
|
|
* Converts BGR to UYVY.
|
|
* Uses full scale Rec. 601 YUV (aka JPEG)
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src buffer in BGR
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
*/
|
|
void vc_copylineBGRtoUYVY(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
vc_copylineToUYVY(dst, src, dst_len, 2, 1, 0, 3);
|
|
}
|
|
|
|
/**
|
|
* Converts RGBA to UYVY.
|
|
* Uses full scale Rec. 601 YUV (aka JPEG)
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src buffer in RGBA
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
*/
|
|
void vc_copylineRGBAtoUYVY(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
vc_copylineToUYVY(dst, src, dst_len, 0, 1, 2, 4);
|
|
}
|
|
|
|
/**
|
|
* Converts BGR to RGB.
|
|
*
|
|
* @param[out] dst output buffer in RGB
|
|
* @param[in] src buffer in BGR
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
*/
|
|
void vc_copylineBGRtoRGB(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
register int r, g, b;
|
|
|
|
while(dst_len > 0) {
|
|
b = *src++;
|
|
g = *src++;
|
|
r = *src++;
|
|
*dst++ = r;
|
|
*dst++ = g;
|
|
*dst++ = b;
|
|
dst_len -= 3;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts DPX10 to RGBA
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src 4B-aligned input buffer
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
* @param[in] rshift RGBA rshift
|
|
* @param[in] gshift RGBA gshift
|
|
* @param[in] bshift RGBA gshift
|
|
*/
|
|
void
|
|
vc_copylineDPX10toRGBA(unsigned char *dst, const unsigned char *src, int dst_len, int rshift, int gshift, int bshift)
|
|
{
|
|
|
|
register const unsigned int *in = (const unsigned int *)(const void *) src;
|
|
register unsigned int *out = (unsigned int *)(void *) dst;
|
|
register int r,g,b;
|
|
|
|
while(dst_len > 0) {
|
|
register unsigned int val = *in;
|
|
r = val >> 24;
|
|
g = 0xff & (val >> 14);
|
|
b = 0xff & (val >> 4);
|
|
|
|
*out++ = (r << rshift) | (g << gshift) | (b << bshift);
|
|
++in;
|
|
dst_len -= 4;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Converts DPX10 to RGB
|
|
*
|
|
* @param[out] dst 4B-aligned output buffer
|
|
* @param[in] src 4B-aligned input buffer
|
|
* @param[in] dst_len number of bytes that should be written to outpu buffer
|
|
*/
|
|
void
|
|
vc_copylineDPX10toRGB(unsigned char *dst, const unsigned char *src, int dst_len)
|
|
{
|
|
|
|
register const unsigned int *in = (const unsigned int *)(const void *) src;
|
|
register unsigned int *out = (unsigned int *)(void *) dst;
|
|
register int r1,g1,b1,r2,g2,b2;
|
|
|
|
while(dst_len > 0) {
|
|
register unsigned int val;
|
|
|
|
val = *in++;
|
|
r1 = val >> 24;
|
|
g1 = 0xff & (val >> 14);
|
|
b1 = 0xff & (val >> 4);
|
|
|
|
val = *in++;
|
|
r2 = val >> 24;
|
|
g2 = 0xff & (val >> 14);
|
|
b2 = 0xff & (val >> 4);
|
|
|
|
*out++ = r1 | g1 << 8 | b1 << 16 | r2 << 24;
|
|
|
|
val = *in++;
|
|
r1 = val >> 24;
|
|
g1 = 0xff & (val >> 14);
|
|
b1 = 0xff & (val >> 4);
|
|
|
|
*out++ = g2 | b2 << 8 | r1 << 16 | g1 << 24;
|
|
|
|
val = *in++;
|
|
r2 = val >> 24;
|
|
g2 = 0xff & (val >> 14);
|
|
b2 = 0xff & (val >> 4);
|
|
|
|
*out++ = b1 | r2 << 8 | g2 << 16 | b2 << 24;
|
|
|
|
dst_len -= 12;
|
|
}
|
|
}
|
|
|
|
|
|
int codec_is_a_rgb(codec_t codec)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; codec_info[i].name != NULL; i++) {
|
|
if (codec == codec_info[i].codec) {
|
|
return codec_info[i].rgb;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|