FFmpeg
sw_yuv2rgb.c
Go to the documentation of this file.
1 /*
2  * This file is part of FFmpeg.
3  *
4  * FFmpeg is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * FFmpeg is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License along
15  * with FFmpeg; if not, write to the Free Software Foundation, Inc.,
16  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17  */
18 
19 #include <string.h>
20 
21 #include "libavutil/common.h"
22 #include "libavutil/intreadwrite.h"
23 #include "libavutil/mem_internal.h"
24 #include "libavutil/pixdesc.h"
25 
26 #include "libswscale/swscale.h"
28 
29 #include "checkasm.h"
30 
31 #define randomize_buffers(buf, size) \
32  do { \
33  for (int j = 0; j < size; j += 4) \
34  AV_WN32(buf + j, rnd()); \
35  } while (0)
36 
37 static const int dst_fmts[] = {
38 // AV_PIX_FMT_BGR48BE,
39 // AV_PIX_FMT_BGR48LE,
40 // AV_PIX_FMT_RGB48BE,
41 // AV_PIX_FMT_RGB48LE,
52 // AV_PIX_FMT_RGB444,
53 // AV_PIX_FMT_BGR444,
54 // AV_PIX_FMT_RGB8,
55 // AV_PIX_FMT_BGR8,
56 // AV_PIX_FMT_RGB4,
57 // AV_PIX_FMT_BGR4,
58 // AV_PIX_FMT_RGB4_BYTE,
59 // AV_PIX_FMT_BGR4_BYTE,
60 // AV_PIX_FMT_MONOBLACK,
61 };
62 
63 static int cmp_off_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
64 {
65  for (size_t i = 0; i < n; i++) {
66  if (abs(ref[i] - test[i]) > accuracy)
67  return 1;
68  }
69  return 0;
70 }
71 
72 static int cmp_555_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
73 {
74  const uint16_t *ref16 = (const uint16_t *) ref;
75  const uint16_t *test16 = (const uint16_t *) test;
76  for (size_t i = 0; i < n; i++) {
77  if (abs(( ref16[i] & 0x1f) - ( test16[i] & 0x1f)) > accuracy)
78  return 1;
79  if (abs(((ref16[i] >> 5) & 0x1f) - ((test16[i] >> 5) & 0x1f)) > accuracy)
80  return 1;
81  if (abs(((ref16[i] >> 10) & 0x1f) - ((test16[i] >> 10) & 0x1f)) > accuracy)
82  return 1;
83  }
84  return 0;
85 }
86 
87 static int cmp_565_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
88 {
89  const uint16_t *ref16 = (const uint16_t *) ref;
90  const uint16_t *test16 = (const uint16_t *) test;
91  for (size_t i = 0; i < n; i++) {
92  if (abs(( ref16[i] & 0x1f) - ( test16[i] & 0x1f)) > accuracy)
93  return 1;
94  if (abs(((ref16[i] >> 5) & 0x3f) - ((test16[i] >> 5) & 0x3f)) > accuracy)
95  return 1;
96  if (abs(((ref16[i] >> 11) & 0x1f) - ((test16[i] >> 11) & 0x1f)) > accuracy)
97  return 1;
98  }
99  return 0;
100 }
101 
102 static void check_yuv2rgb(int src_pix_fmt)
103 {
104  const AVPixFmtDescriptor *src_desc = av_pix_fmt_desc_get(src_pix_fmt);
105 #define MAX_LINE_SIZE 1920
106  static const int input_sizes[] = {8, 128, 1080, MAX_LINE_SIZE};
107 
109  int, SwsContext *c, const uint8_t *src[],
110  int srcStride[], int srcSliceY, int srcSliceH,
111  uint8_t *dst[], int dstStride[]);
112 
113  LOCAL_ALIGNED_8(uint8_t, src_y, [MAX_LINE_SIZE * 2]);
114  LOCAL_ALIGNED_8(uint8_t, src_u, [MAX_LINE_SIZE]);
115  LOCAL_ALIGNED_8(uint8_t, src_v, [MAX_LINE_SIZE]);
116  LOCAL_ALIGNED_8(uint8_t, src_a, [MAX_LINE_SIZE * 2]);
117  const uint8_t *src[4] = { src_y, src_u, src_v, src_a };
118 
119  LOCAL_ALIGNED_8(uint8_t, dst0_, [2 * MAX_LINE_SIZE * 6]);
120  uint8_t *dst0[4] = { dst0_ };
121  uint8_t *lines0[2] = { dst0_, dst0_ + MAX_LINE_SIZE * 6 };
122 
123  LOCAL_ALIGNED_8(uint8_t, dst1_, [2 * MAX_LINE_SIZE * 6]);
124  uint8_t *dst1[4] = { dst1_ };
125  uint8_t *lines1[2] = { dst1_, dst1_ + MAX_LINE_SIZE * 6 };
126 
127  randomize_buffers(src_y, MAX_LINE_SIZE * 2);
130  randomize_buffers(src_a, MAX_LINE_SIZE * 2);
131 
132  for (int dfi = 0; dfi < FF_ARRAY_ELEMS(dst_fmts); dfi++) {
133  int dst_pix_fmt = dst_fmts[dfi];
134  const AVPixFmtDescriptor *dst_desc = av_pix_fmt_desc_get(dst_pix_fmt);
135  int sample_size = av_get_padded_bits_per_pixel(dst_desc) >> 3;
136  for (int isi = 0; isi < FF_ARRAY_ELEMS(input_sizes); isi++) {
137  struct SwsContext *ctx;
138  int log_level;
139  int width = input_sizes[isi];
140  int srcSliceY = 0;
141  int srcSliceH = 2;
142  int srcStride[4] = {
143  width,
144  width >> src_desc->log2_chroma_w,
145  width >> src_desc->log2_chroma_w,
146  width,
147  };
148  int dstStride[4] = { MAX_LINE_SIZE * 6 };
149 
150  // override log level to prevent spamming of the message
151  // "No accelerated colorspace conversion found from %s to %s"
152  log_level = av_log_get_level();
154  ctx = sws_getContext(width, srcSliceH, src_pix_fmt,
155  width, srcSliceH, dst_pix_fmt,
156  0, NULL, NULL, NULL);
157  av_log_set_level(log_level);
158  if (!ctx)
159  fail();
160 
161  if (check_func(ctx->convert_unscaled, "%s_%s_%d", src_desc->name, dst_desc->name, width)) {
162  memset(dst0_, 0xFF, 2 * MAX_LINE_SIZE * 6);
163  memset(dst1_, 0xFF, 2 * MAX_LINE_SIZE * 6);
164 
165  call_ref(ctx, src, srcStride, srcSliceY,
166  srcSliceH, dst0, dstStride);
167  call_new(ctx, src, srcStride, srcSliceY,
168  srcSliceH, dst1, dstStride);
169 
170  if (dst_pix_fmt == AV_PIX_FMT_ARGB ||
171  dst_pix_fmt == AV_PIX_FMT_ABGR ||
172  dst_pix_fmt == AV_PIX_FMT_RGBA ||
173  dst_pix_fmt == AV_PIX_FMT_BGRA ||
174  dst_pix_fmt == AV_PIX_FMT_RGB24 ||
175  dst_pix_fmt == AV_PIX_FMT_BGR24) {
176  if (cmp_off_by_n(lines0[0], lines1[0], width * sample_size, 3) ||
177  cmp_off_by_n(lines0[1], lines1[1], width * sample_size, 3))
178  fail();
179  } else if (dst_pix_fmt == AV_PIX_FMT_RGB565 ||
180  dst_pix_fmt == AV_PIX_FMT_BGR565) {
181  if (cmp_565_by_n(lines0[0], lines1[0], width, 2) ||
182  cmp_565_by_n(lines0[1], lines1[1], width, 2))
183  fail();
184  } else if (dst_pix_fmt == AV_PIX_FMT_RGB555 ||
185  dst_pix_fmt == AV_PIX_FMT_BGR555) {
186  if (cmp_555_by_n(lines0[0], lines1[0], width, 2) ||
187  cmp_555_by_n(lines0[1], lines1[1], width, 2))
188  fail();
189  } else {
190  fail();
191  }
192 
193  bench_new(ctx, src, srcStride, srcSliceY,
194  srcSliceH, dst0, dstStride);
195  }
197  }
198  }
199 }
200 
201 #undef MAX_LINE_SIZE
202 
204 {
206  report("yuv420p");
208  report("yuv422p");
210  report("yuva420p");
211 }
declare_func_emms
#define declare_func_emms(cpu_flags, ret,...)
Definition: checkasm.h:183
mem_internal.h
av_pix_fmt_desc_get
const AVPixFmtDescriptor * av_pix_fmt_desc_get(enum AVPixelFormat pix_fmt)
Definition: pixdesc.c:2965
pixdesc.h
AVPixFmtDescriptor::name
const char * name
Definition: pixdesc.h:70
check_func
#define check_func(func,...)
Definition: checkasm.h:177
test
Definition: idctdsp.c:35
AV_PIX_FMT_BGR24
@ AV_PIX_FMT_BGR24
packed RGB 8:8:8, 24bpp, BGRBGR...
Definition: pixfmt.h:76
AV_PIX_FMT_BGRA
@ AV_PIX_FMT_BGRA
packed BGRA 8:8:8:8, 32bpp, BGRABGRA...
Definition: pixfmt.h:102
call_ref
#define call_ref(...)
Definition: checkasm.h:192
randomize_buffers
#define randomize_buffers(buf, size)
Definition: sw_yuv2rgb.c:31
cmp_565_by_n
static int cmp_565_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
Definition: sw_yuv2rgb.c:87
fail
#define fail()
Definition: checkasm.h:186
checkasm.h
MAX_LINE_SIZE
#define MAX_LINE_SIZE
AV_LOG_ERROR
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
Definition: log.h:180
FF_ARRAY_ELEMS
#define FF_ARRAY_ELEMS(a)
Definition: sinewin_tablegen.c:29
cmp_555_by_n
static int cmp_555_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
Definition: sw_yuv2rgb.c:72
width
#define width
intreadwrite.h
AV_PIX_FMT_YUVA420P
@ AV_PIX_FMT_YUVA420P
planar YUV 4:2:0, 20bpp, (1 Cr & Cb sample per 2x2 Y & A samples)
Definition: pixfmt.h:108
input_sizes
static const int input_sizes[]
Definition: sw_rgb.c:186
check_yuv2rgb
static void check_yuv2rgb(int src_pix_fmt)
Definition: sw_yuv2rgb.c:102
ctx
AVFormatContext * ctx
Definition: movenc.c:49
AVPixFmtDescriptor::log2_chroma_w
uint8_t log2_chroma_w
Amount to shift the luma width right to find the chroma width.
Definition: pixdesc.h:80
AV_PIX_FMT_YUV420P
@ AV_PIX_FMT_YUV420P
planar YUV 4:2:0, 12bpp, (1 Cr & Cb sample per 2x2 Y samples)
Definition: pixfmt.h:73
AV_PIX_FMT_RGBA
@ AV_PIX_FMT_RGBA
packed RGBA 8:8:8:8, 32bpp, RGBARGBA...
Definition: pixfmt.h:100
LOCAL_ALIGNED_8
#define LOCAL_ALIGNED_8(t, v,...)
Definition: mem_internal.h:144
av_log_get_level
int av_log_get_level(void)
Get the current log level.
Definition: log.c:442
call_new
#define call_new(...)
Definition: checkasm.h:295
NULL
#define NULL
Definition: coverity.c:32
abs
#define abs(x)
Definition: cuda_runtime.h:35
cmp_off_by_n
static int cmp_off_by_n(const uint8_t *ref, const uint8_t *test, size_t n, int accuracy)
Definition: sw_yuv2rgb.c:63
AV_PIX_FMT_ABGR
@ AV_PIX_FMT_ABGR
packed ABGR 8:8:8:8, 32bpp, ABGRABGR...
Definition: pixfmt.h:101
c
Undefined Behavior In the C some operations are like signed integer dereferencing freed accessing outside allocated Undefined Behavior must not occur in a C it is not safe even if the output of undefined operations is unused The unsafety may seem nit picking but Optimizing compilers have in fact optimized code on the assumption that no undefined Behavior occurs Optimizing code based on wrong assumptions can and has in some cases lead to effects beyond the output of computations The signed integer overflow problem in speed critical code Code which is highly optimized and works with signed integers sometimes has the problem that often the output of the computation does not c
Definition: undefined.txt:32
AV_PIX_FMT_RGB24
@ AV_PIX_FMT_RGB24
packed RGB 8:8:8, 24bpp, RGBRGB...
Definition: pixfmt.h:75
av_get_padded_bits_per_pixel
int av_get_padded_bits_per_pixel(const AVPixFmtDescriptor *pixdesc)
Return the number of bits per pixel for the pixel format described by pixdesc, including any padding ...
Definition: pixdesc.c:2930
sws_getContext
struct SwsContext * sws_getContext(int srcW, int srcH, enum AVPixelFormat srcFormat, int dstW, int dstH, enum AVPixelFormat dstFormat, int flags, SwsFilter *srcFilter, SwsFilter *dstFilter, const double *param)
Allocate and return an SwsContext.
Definition: utils.c:2106
checkasm_check_sw_yuv2rgb
void checkasm_check_sw_yuv2rgb(void)
Definition: sw_yuv2rgb.c:203
AV_PIX_FMT_BGR555
#define AV_PIX_FMT_BGR555
Definition: pixfmt.h:471
srcSliceH
return srcSliceH
Definition: yuv2rgb_template.c:87
AV_PIX_FMT_ARGB
@ AV_PIX_FMT_ARGB
packed ARGB 8:8:8:8, 32bpp, ARGBARGB...
Definition: pixfmt.h:99
report
#define report
Definition: checkasm.h:189
av_log_set_level
void av_log_set_level(int level)
Set the log level.
Definition: log.c:447
bench_new
#define bench_new(...)
Definition: checkasm.h:366
i
#define i(width, name, range_min, range_max)
Definition: cbs_h2645.c:256
common.h
AV_PIX_FMT_RGB555
#define AV_PIX_FMT_RGB555
Definition: pixfmt.h:466
swscale_internal.h
AV_PIX_FMT_BGR565
#define AV_PIX_FMT_BGR565
Definition: pixfmt.h:470
AV_PIX_FMT_RGB565
#define AV_PIX_FMT_RGB565
Definition: pixfmt.h:465
AV_CPU_FLAG_MMX
#define AV_CPU_FLAG_MMX
standard MMX
Definition: cpu.h:29
sws_freeContext
void sws_freeContext(struct SwsContext *swsContext)
Free the swscaler context swsContext.
Definition: utils.c:2437
ref
static int ref[MAX_W *MAX_W]
Definition: jpeg2000dwt.c:112
AV_CPU_FLAG_MMXEXT
#define AV_CPU_FLAG_MMXEXT
SSE integer functions or AMD MMX ext.
Definition: cpu.h:30
AV_PIX_FMT_YUV422P
@ AV_PIX_FMT_YUV422P
planar YUV 4:2:2, 16bpp, (1 Cr & Cb sample per 2x1 Y samples)
Definition: pixfmt.h:77
AVPixFmtDescriptor
Descriptor that unambiguously describes how the bits of a pixel are stored in the up to 4 data planes...
Definition: pixdesc.h:69
src
INIT_CLIP pixel * src
Definition: h264pred_template.c:418
dst_fmts
static const int dst_fmts[]
Definition: sw_yuv2rgb.c:37
SwsContext
Definition: swscale_internal.h:301
swscale.h