1 // SPDX-License-Identifier: GPL-2.0-only
3 * v4l2-tpg-core.c - Test Pattern Generator
5 * Note: gen_twopix and tpg_gen_text are based on code from vivi.c. See the
6 * vivi.c source for the copyright information of those functions.
8 * Copyright 2014 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
11 #include <linux/module.h>
12 #include <media/tpg/v4l2-tpg.h>
14 /* Must remain in sync with enum tpg_pattern */
15 const char * const tpg_pattern_strings
[] = {
19 "Horizontal 100% Colorbar",
29 "2x2 Red/Green Checkers",
30 "1x1 Red/Green Checkers",
31 "Alternating Hor Lines",
32 "Alternating Vert Lines",
33 "One Pixel Wide Cross",
34 "Two Pixels Wide Cross",
35 "Ten Pixels Wide Cross",
40 EXPORT_SYMBOL_GPL(tpg_pattern_strings
);
42 /* Must remain in sync with enum tpg_aspect */
43 const char * const tpg_aspect_strings
[] = {
44 "Source Width x Height",
51 EXPORT_SYMBOL_GPL(tpg_aspect_strings
);
54 * Sine table: sin[0] = 127 * sin(-180 degrees)
55 * sin[128] = 127 * sin(0 degrees)
56 * sin[256] = 127 * sin(180 degrees)
58 static const s8 sin
[257] = {
59 0, -4, -7, -11, -13, -18, -20, -22, -26, -29, -33, -35, -37, -41, -43, -48,
60 -50, -52, -56, -58, -62, -63, -65, -69, -71, -75, -76, -78, -82, -83, -87, -88,
61 -90, -93, -94, -97, -99, -101, -103, -104, -107, -108, -110, -111, -112, -114, -115, -117,
62 -118, -119, -120, -121, -122, -123, -123, -124, -125, -125, -126, -126, -127, -127, -127, -127,
63 -127, -127, -127, -127, -126, -126, -125, -125, -124, -124, -123, -122, -121, -120, -119, -118,
64 -117, -116, -114, -113, -111, -110, -109, -107, -105, -103, -101, -100, -97, -96, -93, -91,
65 -90, -87, -85, -82, -80, -76, -75, -73, -69, -67, -63, -62, -60, -56, -54, -50,
66 -48, -46, -41, -39, -35, -33, -31, -26, -24, -20, -18, -15, -11, -9, -4, -2,
67 0, 2, 4, 9, 11, 15, 18, 20, 24, 26, 31, 33, 35, 39, 41, 46,
68 48, 50, 54, 56, 60, 62, 64, 67, 69, 73, 75, 76, 80, 82, 85, 87,
69 90, 91, 93, 96, 97, 100, 101, 103, 105, 107, 109, 110, 111, 113, 114, 116,
70 117, 118, 119, 120, 121, 122, 123, 124, 124, 125, 125, 126, 126, 127, 127, 127,
71 127, 127, 127, 127, 127, 126, 126, 125, 125, 124, 123, 123, 122, 121, 120, 119,
72 118, 117, 115, 114, 112, 111, 110, 108, 107, 104, 103, 101, 99, 97, 94, 93,
73 90, 88, 87, 83, 82, 78, 76, 75, 71, 69, 65, 64, 62, 58, 56, 52,
74 50, 48, 43, 41, 37, 35, 33, 29, 26, 22, 20, 18, 13, 11, 7, 4,
78 #define cos(idx) sin[((idx) + 64) % sizeof(sin)]
80 /* Global font descriptor */
81 static const u8
*font8x16
;
83 void tpg_set_font(const u8
*f
)
87 EXPORT_SYMBOL_GPL(tpg_set_font
);
89 void tpg_init(struct tpg_data
*tpg
, unsigned w
, unsigned h
)
91 memset(tpg
, 0, sizeof(*tpg
));
92 tpg
->scaled_width
= tpg
->src_width
= w
;
93 tpg
->src_height
= tpg
->buf_height
= h
;
94 tpg
->crop
.width
= tpg
->compose
.width
= w
;
95 tpg
->crop
.height
= tpg
->compose
.height
= h
;
96 tpg
->recalc_colors
= true;
97 tpg
->recalc_square_border
= true;
98 tpg
->brightness
= 128;
100 tpg
->saturation
= 128;
102 tpg
->mv_hor_mode
= TPG_MOVE_NONE
;
103 tpg
->mv_vert_mode
= TPG_MOVE_NONE
;
104 tpg
->field
= V4L2_FIELD_NONE
;
105 tpg_s_fourcc(tpg
, V4L2_PIX_FMT_RGB24
);
106 tpg
->colorspace
= V4L2_COLORSPACE_SRGB
;
107 tpg
->perc_fill
= 100;
108 tpg
->hsv_enc
= V4L2_HSV_ENC_180
;
110 EXPORT_SYMBOL_GPL(tpg_init
);
112 int tpg_alloc(struct tpg_data
*tpg
, unsigned max_w
)
117 tpg
->max_line_width
= max_w
;
118 for (pat
= 0; pat
< TPG_MAX_PAT_LINES
; pat
++) {
119 for (plane
= 0; plane
< TPG_MAX_PLANES
; plane
++) {
120 unsigned pixelsz
= plane
? 2 : 4;
122 tpg
->lines
[pat
][plane
] =
123 vzalloc(array3_size(max_w
, 2, pixelsz
));
124 if (!tpg
->lines
[pat
][plane
])
128 tpg
->downsampled_lines
[pat
][plane
] =
129 vzalloc(array3_size(max_w
, 2, pixelsz
));
130 if (!tpg
->downsampled_lines
[pat
][plane
])
134 for (plane
= 0; plane
< TPG_MAX_PLANES
; plane
++) {
135 unsigned pixelsz
= plane
? 2 : 4;
137 tpg
->contrast_line
[plane
] =
138 vzalloc(array_size(pixelsz
, max_w
));
139 if (!tpg
->contrast_line
[plane
])
141 tpg
->black_line
[plane
] =
142 vzalloc(array_size(pixelsz
, max_w
));
143 if (!tpg
->black_line
[plane
])
145 tpg
->random_line
[plane
] =
146 vzalloc(array3_size(max_w
, 2, pixelsz
));
147 if (!tpg
->random_line
[plane
])
152 EXPORT_SYMBOL_GPL(tpg_alloc
);
154 void tpg_free(struct tpg_data
*tpg
)
159 for (pat
= 0; pat
< TPG_MAX_PAT_LINES
; pat
++)
160 for (plane
= 0; plane
< TPG_MAX_PLANES
; plane
++) {
161 vfree(tpg
->lines
[pat
][plane
]);
162 tpg
->lines
[pat
][plane
] = NULL
;
165 vfree(tpg
->downsampled_lines
[pat
][plane
]);
166 tpg
->downsampled_lines
[pat
][plane
] = NULL
;
168 for (plane
= 0; plane
< TPG_MAX_PLANES
; plane
++) {
169 vfree(tpg
->contrast_line
[plane
]);
170 vfree(tpg
->black_line
[plane
]);
171 vfree(tpg
->random_line
[plane
]);
172 tpg
->contrast_line
[plane
] = NULL
;
173 tpg
->black_line
[plane
] = NULL
;
174 tpg
->random_line
[plane
] = NULL
;
177 EXPORT_SYMBOL_GPL(tpg_free
);
179 bool tpg_s_fourcc(struct tpg_data
*tpg
, u32 fourcc
)
181 tpg
->fourcc
= fourcc
;
184 tpg
->recalc_colors
= true;
185 tpg
->interleaved
= false;
186 tpg
->vdownsampling
[0] = 1;
187 tpg
->hdownsampling
[0] = 1;
193 case V4L2_PIX_FMT_SBGGR8
:
194 case V4L2_PIX_FMT_SGBRG8
:
195 case V4L2_PIX_FMT_SGRBG8
:
196 case V4L2_PIX_FMT_SRGGB8
:
197 case V4L2_PIX_FMT_SBGGR10
:
198 case V4L2_PIX_FMT_SGBRG10
:
199 case V4L2_PIX_FMT_SGRBG10
:
200 case V4L2_PIX_FMT_SRGGB10
:
201 case V4L2_PIX_FMT_SBGGR12
:
202 case V4L2_PIX_FMT_SGBRG12
:
203 case V4L2_PIX_FMT_SGRBG12
:
204 case V4L2_PIX_FMT_SRGGB12
:
205 case V4L2_PIX_FMT_SBGGR16
:
206 case V4L2_PIX_FMT_SGBRG16
:
207 case V4L2_PIX_FMT_SGRBG16
:
208 case V4L2_PIX_FMT_SRGGB16
:
209 tpg
->interleaved
= true;
210 tpg
->vdownsampling
[1] = 1;
211 tpg
->hdownsampling
[1] = 1;
214 case V4L2_PIX_FMT_RGB332
:
215 case V4L2_PIX_FMT_RGB565
:
216 case V4L2_PIX_FMT_RGB565X
:
217 case V4L2_PIX_FMT_RGB444
:
218 case V4L2_PIX_FMT_XRGB444
:
219 case V4L2_PIX_FMT_ARGB444
:
220 case V4L2_PIX_FMT_RGBX444
:
221 case V4L2_PIX_FMT_RGBA444
:
222 case V4L2_PIX_FMT_XBGR444
:
223 case V4L2_PIX_FMT_ABGR444
:
224 case V4L2_PIX_FMT_BGRX444
:
225 case V4L2_PIX_FMT_BGRA444
:
226 case V4L2_PIX_FMT_RGB555
:
227 case V4L2_PIX_FMT_XRGB555
:
228 case V4L2_PIX_FMT_ARGB555
:
229 case V4L2_PIX_FMT_RGBX555
:
230 case V4L2_PIX_FMT_RGBA555
:
231 case V4L2_PIX_FMT_XBGR555
:
232 case V4L2_PIX_FMT_ABGR555
:
233 case V4L2_PIX_FMT_BGRX555
:
234 case V4L2_PIX_FMT_BGRA555
:
235 case V4L2_PIX_FMT_RGB555X
:
236 case V4L2_PIX_FMT_XRGB555X
:
237 case V4L2_PIX_FMT_ARGB555X
:
238 case V4L2_PIX_FMT_BGR666
:
239 case V4L2_PIX_FMT_RGB24
:
240 case V4L2_PIX_FMT_BGR24
:
241 case V4L2_PIX_FMT_RGB32
:
242 case V4L2_PIX_FMT_BGR32
:
243 case V4L2_PIX_FMT_XRGB32
:
244 case V4L2_PIX_FMT_XBGR32
:
245 case V4L2_PIX_FMT_ARGB32
:
246 case V4L2_PIX_FMT_ABGR32
:
247 case V4L2_PIX_FMT_RGBX32
:
248 case V4L2_PIX_FMT_BGRX32
:
249 case V4L2_PIX_FMT_RGBA32
:
250 case V4L2_PIX_FMT_BGRA32
:
251 tpg
->color_enc
= TGP_COLOR_ENC_RGB
;
253 case V4L2_PIX_FMT_GREY
:
254 case V4L2_PIX_FMT_Y10
:
255 case V4L2_PIX_FMT_Y12
:
256 case V4L2_PIX_FMT_Y16
:
257 case V4L2_PIX_FMT_Y16_BE
:
258 case V4L2_PIX_FMT_Z16
:
259 tpg
->color_enc
= TGP_COLOR_ENC_LUMA
;
261 case V4L2_PIX_FMT_YUV444
:
262 case V4L2_PIX_FMT_YUV555
:
263 case V4L2_PIX_FMT_YUV565
:
264 case V4L2_PIX_FMT_YUV32
:
265 case V4L2_PIX_FMT_AYUV32
:
266 case V4L2_PIX_FMT_XYUV32
:
267 case V4L2_PIX_FMT_VUYA32
:
268 case V4L2_PIX_FMT_VUYX32
:
269 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
271 case V4L2_PIX_FMT_YUV420M
:
272 case V4L2_PIX_FMT_YVU420M
:
275 case V4L2_PIX_FMT_YUV420
:
276 case V4L2_PIX_FMT_YVU420
:
277 tpg
->vdownsampling
[1] = 2;
278 tpg
->vdownsampling
[2] = 2;
279 tpg
->hdownsampling
[1] = 2;
280 tpg
->hdownsampling
[2] = 2;
282 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
284 case V4L2_PIX_FMT_YUV422M
:
285 case V4L2_PIX_FMT_YVU422M
:
288 case V4L2_PIX_FMT_YUV422P
:
289 tpg
->vdownsampling
[1] = 1;
290 tpg
->vdownsampling
[2] = 1;
291 tpg
->hdownsampling
[1] = 2;
292 tpg
->hdownsampling
[2] = 2;
294 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
296 case V4L2_PIX_FMT_NV16M
:
297 case V4L2_PIX_FMT_NV61M
:
300 case V4L2_PIX_FMT_NV16
:
301 case V4L2_PIX_FMT_NV61
:
302 tpg
->vdownsampling
[1] = 1;
303 tpg
->hdownsampling
[1] = 1;
306 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
308 case V4L2_PIX_FMT_NV12M
:
309 case V4L2_PIX_FMT_NV21M
:
312 case V4L2_PIX_FMT_NV12
:
313 case V4L2_PIX_FMT_NV21
:
314 tpg
->vdownsampling
[1] = 2;
315 tpg
->hdownsampling
[1] = 1;
318 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
320 case V4L2_PIX_FMT_YUV444M
:
321 case V4L2_PIX_FMT_YVU444M
:
324 tpg
->vdownsampling
[1] = 1;
325 tpg
->vdownsampling
[2] = 1;
326 tpg
->hdownsampling
[1] = 1;
327 tpg
->hdownsampling
[2] = 1;
328 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
330 case V4L2_PIX_FMT_NV24
:
331 case V4L2_PIX_FMT_NV42
:
332 tpg
->vdownsampling
[1] = 1;
333 tpg
->hdownsampling
[1] = 1;
335 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
337 case V4L2_PIX_FMT_YUYV
:
338 case V4L2_PIX_FMT_UYVY
:
339 case V4L2_PIX_FMT_YVYU
:
340 case V4L2_PIX_FMT_VYUY
:
342 tpg
->color_enc
= TGP_COLOR_ENC_YCBCR
;
344 case V4L2_PIX_FMT_HSV24
:
345 case V4L2_PIX_FMT_HSV32
:
346 tpg
->color_enc
= TGP_COLOR_ENC_HSV
;
353 case V4L2_PIX_FMT_GREY
:
354 case V4L2_PIX_FMT_RGB332
:
355 tpg
->twopixelsize
[0] = 2;
357 case V4L2_PIX_FMT_RGB565
:
358 case V4L2_PIX_FMT_RGB565X
:
359 case V4L2_PIX_FMT_RGB444
:
360 case V4L2_PIX_FMT_XRGB444
:
361 case V4L2_PIX_FMT_ARGB444
:
362 case V4L2_PIX_FMT_RGBX444
:
363 case V4L2_PIX_FMT_RGBA444
:
364 case V4L2_PIX_FMT_XBGR444
:
365 case V4L2_PIX_FMT_ABGR444
:
366 case V4L2_PIX_FMT_BGRX444
:
367 case V4L2_PIX_FMT_BGRA444
:
368 case V4L2_PIX_FMT_RGB555
:
369 case V4L2_PIX_FMT_XRGB555
:
370 case V4L2_PIX_FMT_ARGB555
:
371 case V4L2_PIX_FMT_RGBX555
:
372 case V4L2_PIX_FMT_RGBA555
:
373 case V4L2_PIX_FMT_XBGR555
:
374 case V4L2_PIX_FMT_ABGR555
:
375 case V4L2_PIX_FMT_BGRX555
:
376 case V4L2_PIX_FMT_BGRA555
:
377 case V4L2_PIX_FMT_RGB555X
:
378 case V4L2_PIX_FMT_XRGB555X
:
379 case V4L2_PIX_FMT_ARGB555X
:
380 case V4L2_PIX_FMT_YUYV
:
381 case V4L2_PIX_FMT_UYVY
:
382 case V4L2_PIX_FMT_YVYU
:
383 case V4L2_PIX_FMT_VYUY
:
384 case V4L2_PIX_FMT_YUV444
:
385 case V4L2_PIX_FMT_YUV555
:
386 case V4L2_PIX_FMT_YUV565
:
387 case V4L2_PIX_FMT_Y10
:
388 case V4L2_PIX_FMT_Y12
:
389 case V4L2_PIX_FMT_Y16
:
390 case V4L2_PIX_FMT_Y16_BE
:
391 case V4L2_PIX_FMT_Z16
:
392 tpg
->twopixelsize
[0] = 2 * 2;
394 case V4L2_PIX_FMT_RGB24
:
395 case V4L2_PIX_FMT_BGR24
:
396 case V4L2_PIX_FMT_HSV24
:
397 tpg
->twopixelsize
[0] = 2 * 3;
399 case V4L2_PIX_FMT_BGR666
:
400 case V4L2_PIX_FMT_RGB32
:
401 case V4L2_PIX_FMT_BGR32
:
402 case V4L2_PIX_FMT_XRGB32
:
403 case V4L2_PIX_FMT_XBGR32
:
404 case V4L2_PIX_FMT_ARGB32
:
405 case V4L2_PIX_FMT_ABGR32
:
406 case V4L2_PIX_FMT_RGBX32
:
407 case V4L2_PIX_FMT_BGRX32
:
408 case V4L2_PIX_FMT_RGBA32
:
409 case V4L2_PIX_FMT_BGRA32
:
410 case V4L2_PIX_FMT_YUV32
:
411 case V4L2_PIX_FMT_AYUV32
:
412 case V4L2_PIX_FMT_XYUV32
:
413 case V4L2_PIX_FMT_VUYA32
:
414 case V4L2_PIX_FMT_VUYX32
:
415 case V4L2_PIX_FMT_HSV32
:
416 tpg
->twopixelsize
[0] = 2 * 4;
418 case V4L2_PIX_FMT_NV12
:
419 case V4L2_PIX_FMT_NV21
:
420 case V4L2_PIX_FMT_NV12M
:
421 case V4L2_PIX_FMT_NV21M
:
422 case V4L2_PIX_FMT_NV16
:
423 case V4L2_PIX_FMT_NV61
:
424 case V4L2_PIX_FMT_NV16M
:
425 case V4L2_PIX_FMT_NV61M
:
426 case V4L2_PIX_FMT_SBGGR8
:
427 case V4L2_PIX_FMT_SGBRG8
:
428 case V4L2_PIX_FMT_SGRBG8
:
429 case V4L2_PIX_FMT_SRGGB8
:
430 tpg
->twopixelsize
[0] = 2;
431 tpg
->twopixelsize
[1] = 2;
433 case V4L2_PIX_FMT_SRGGB10
:
434 case V4L2_PIX_FMT_SGRBG10
:
435 case V4L2_PIX_FMT_SGBRG10
:
436 case V4L2_PIX_FMT_SBGGR10
:
437 case V4L2_PIX_FMT_SRGGB12
:
438 case V4L2_PIX_FMT_SGRBG12
:
439 case V4L2_PIX_FMT_SGBRG12
:
440 case V4L2_PIX_FMT_SBGGR12
:
441 case V4L2_PIX_FMT_SRGGB16
:
442 case V4L2_PIX_FMT_SGRBG16
:
443 case V4L2_PIX_FMT_SGBRG16
:
444 case V4L2_PIX_FMT_SBGGR16
:
445 tpg
->twopixelsize
[0] = 4;
446 tpg
->twopixelsize
[1] = 4;
448 case V4L2_PIX_FMT_YUV444M
:
449 case V4L2_PIX_FMT_YVU444M
:
450 case V4L2_PIX_FMT_YUV422M
:
451 case V4L2_PIX_FMT_YVU422M
:
452 case V4L2_PIX_FMT_YUV422P
:
453 case V4L2_PIX_FMT_YUV420
:
454 case V4L2_PIX_FMT_YVU420
:
455 case V4L2_PIX_FMT_YUV420M
:
456 case V4L2_PIX_FMT_YVU420M
:
457 tpg
->twopixelsize
[0] = 2;
458 tpg
->twopixelsize
[1] = 2;
459 tpg
->twopixelsize
[2] = 2;
461 case V4L2_PIX_FMT_NV24
:
462 case V4L2_PIX_FMT_NV42
:
463 tpg
->twopixelsize
[0] = 2;
464 tpg
->twopixelsize
[1] = 4;
469 EXPORT_SYMBOL_GPL(tpg_s_fourcc
);
471 void tpg_s_crop_compose(struct tpg_data
*tpg
, const struct v4l2_rect
*crop
,
472 const struct v4l2_rect
*compose
)
475 tpg
->compose
= *compose
;
476 tpg
->scaled_width
= (tpg
->src_width
* tpg
->compose
.width
+
477 tpg
->crop
.width
- 1) / tpg
->crop
.width
;
478 tpg
->scaled_width
&= ~1;
479 if (tpg
->scaled_width
> tpg
->max_line_width
)
480 tpg
->scaled_width
= tpg
->max_line_width
;
481 if (tpg
->scaled_width
< 2)
482 tpg
->scaled_width
= 2;
483 tpg
->recalc_lines
= true;
485 EXPORT_SYMBOL_GPL(tpg_s_crop_compose
);
487 void tpg_reset_source(struct tpg_data
*tpg
, unsigned width
, unsigned height
,
492 tpg
->src_width
= width
;
493 tpg
->src_height
= height
;
495 tpg
->buf_height
= height
;
496 if (V4L2_FIELD_HAS_T_OR_B(field
))
497 tpg
->buf_height
/= 2;
498 tpg
->scaled_width
= width
;
499 tpg
->crop
.top
= tpg
->crop
.left
= 0;
500 tpg
->crop
.width
= width
;
501 tpg
->crop
.height
= height
;
502 tpg
->compose
.top
= tpg
->compose
.left
= 0;
503 tpg
->compose
.width
= width
;
504 tpg
->compose
.height
= tpg
->buf_height
;
505 for (p
= 0; p
< tpg
->planes
; p
++)
506 tpg
->bytesperline
[p
] = (width
* tpg
->twopixelsize
[p
]) /
507 (2 * tpg
->hdownsampling
[p
]);
508 tpg
->recalc_square_border
= true;
510 EXPORT_SYMBOL_GPL(tpg_reset_source
);
512 static enum tpg_color
tpg_get_textbg_color(struct tpg_data
*tpg
)
514 switch (tpg
->pattern
) {
516 return TPG_COLOR_100_WHITE
;
517 case TPG_PAT_CSC_COLORBAR
:
518 return TPG_COLOR_CSC_BLACK
;
520 return TPG_COLOR_100_BLACK
;
524 static enum tpg_color
tpg_get_textfg_color(struct tpg_data
*tpg
)
526 switch (tpg
->pattern
) {
527 case TPG_PAT_75_COLORBAR
:
528 case TPG_PAT_CSC_COLORBAR
:
529 return TPG_COLOR_CSC_WHITE
;
531 return TPG_COLOR_100_BLACK
;
533 return TPG_COLOR_100_WHITE
;
537 static inline int rec709_to_linear(int v
)
539 v
= clamp(v
, 0, 0xff0);
540 return tpg_rec709_to_linear
[v
];
543 static inline int linear_to_rec709(int v
)
545 v
= clamp(v
, 0, 0xff0);
546 return tpg_linear_to_rec709
[v
];
549 static void color_to_hsv(struct tpg_data
*tpg
, int r
, int g
, int b
,
550 int *h
, int *s
, int *v
)
552 int max_rgb
, min_rgb
, diff_rgb
;
562 max_rgb
= max3(r
, g
, b
);
571 min_rgb
= min3(r
, g
, b
);
572 diff_rgb
= max_rgb
- min_rgb
;
573 aux
= 255 * diff_rgb
;
582 third_size
= (tpg
->real_hsv_enc
== V4L2_HSV_ENC_180
) ? 60 : 85;
588 } else if (max_rgb
== g
) {
593 third
= third_size
* 2;
596 aux
*= third_size
/ 2;
602 if (tpg
->real_hsv_enc
== V4L2_HSV_ENC_180
) {
614 static void rgb2ycbcr(const int m
[3][3], int r
, int g
, int b
,
615 int y_offset
, int *y
, int *cb
, int *cr
)
617 *y
= ((m
[0][0] * r
+ m
[0][1] * g
+ m
[0][2] * b
) >> 16) + (y_offset
<< 4);
618 *cb
= ((m
[1][0] * r
+ m
[1][1] * g
+ m
[1][2] * b
) >> 16) + (128 << 4);
619 *cr
= ((m
[2][0] * r
+ m
[2][1] * g
+ m
[2][2] * b
) >> 16) + (128 << 4);
622 static void color_to_ycbcr(struct tpg_data
*tpg
, int r
, int g
, int b
,
623 int *y
, int *cb
, int *cr
)
625 #define COEFF(v, r) ((int)(0.5 + (v) * (r) * 256.0))
627 static const int bt601
[3][3] = {
628 { COEFF(0.299, 219), COEFF(0.587, 219), COEFF(0.114, 219) },
629 { COEFF(-0.1687, 224), COEFF(-0.3313, 224), COEFF(0.5, 224) },
630 { COEFF(0.5, 224), COEFF(-0.4187, 224), COEFF(-0.0813, 224) },
632 static const int bt601_full
[3][3] = {
633 { COEFF(0.299, 255), COEFF(0.587, 255), COEFF(0.114, 255) },
634 { COEFF(-0.1687, 255), COEFF(-0.3313, 255), COEFF(0.5, 255) },
635 { COEFF(0.5, 255), COEFF(-0.4187, 255), COEFF(-0.0813, 255) },
637 static const int rec709
[3][3] = {
638 { COEFF(0.2126, 219), COEFF(0.7152, 219), COEFF(0.0722, 219) },
639 { COEFF(-0.1146, 224), COEFF(-0.3854, 224), COEFF(0.5, 224) },
640 { COEFF(0.5, 224), COEFF(-0.4542, 224), COEFF(-0.0458, 224) },
642 static const int rec709_full
[3][3] = {
643 { COEFF(0.2126, 255), COEFF(0.7152, 255), COEFF(0.0722, 255) },
644 { COEFF(-0.1146, 255), COEFF(-0.3854, 255), COEFF(0.5, 255) },
645 { COEFF(0.5, 255), COEFF(-0.4542, 255), COEFF(-0.0458, 255) },
647 static const int smpte240m
[3][3] = {
648 { COEFF(0.212, 219), COEFF(0.701, 219), COEFF(0.087, 219) },
649 { COEFF(-0.116, 224), COEFF(-0.384, 224), COEFF(0.5, 224) },
650 { COEFF(0.5, 224), COEFF(-0.445, 224), COEFF(-0.055, 224) },
652 static const int smpte240m_full
[3][3] = {
653 { COEFF(0.212, 255), COEFF(0.701, 255), COEFF(0.087, 255) },
654 { COEFF(-0.116, 255), COEFF(-0.384, 255), COEFF(0.5, 255) },
655 { COEFF(0.5, 255), COEFF(-0.445, 255), COEFF(-0.055, 255) },
657 static const int bt2020
[3][3] = {
658 { COEFF(0.2627, 219), COEFF(0.6780, 219), COEFF(0.0593, 219) },
659 { COEFF(-0.1396, 224), COEFF(-0.3604, 224), COEFF(0.5, 224) },
660 { COEFF(0.5, 224), COEFF(-0.4598, 224), COEFF(-0.0402, 224) },
662 static const int bt2020_full
[3][3] = {
663 { COEFF(0.2627, 255), COEFF(0.6780, 255), COEFF(0.0593, 255) },
664 { COEFF(-0.1396, 255), COEFF(-0.3604, 255), COEFF(0.5, 255) },
665 { COEFF(0.5, 255), COEFF(-0.4598, 255), COEFF(-0.0402, 255) },
667 static const int bt2020c
[4] = {
668 COEFF(1.0 / 1.9404, 224), COEFF(1.0 / 1.5816, 224),
669 COEFF(1.0 / 1.7184, 224), COEFF(1.0 / 0.9936, 224),
671 static const int bt2020c_full
[4] = {
672 COEFF(1.0 / 1.9404, 255), COEFF(1.0 / 1.5816, 255),
673 COEFF(1.0 / 1.7184, 255), COEFF(1.0 / 0.9936, 255),
676 bool full
= tpg
->real_quantization
== V4L2_QUANTIZATION_FULL_RANGE
;
677 unsigned y_offset
= full
? 0 : 16;
680 switch (tpg
->real_ycbcr_enc
) {
681 case V4L2_YCBCR_ENC_601
:
682 rgb2ycbcr(full
? bt601_full
: bt601
, r
, g
, b
, y_offset
, y
, cb
, cr
);
684 case V4L2_YCBCR_ENC_XV601
:
685 /* Ignore quantization range, there is only one possible
686 * Y'CbCr encoding. */
687 rgb2ycbcr(bt601
, r
, g
, b
, 16, y
, cb
, cr
);
689 case V4L2_YCBCR_ENC_XV709
:
690 /* Ignore quantization range, there is only one possible
691 * Y'CbCr encoding. */
692 rgb2ycbcr(rec709
, r
, g
, b
, 16, y
, cb
, cr
);
694 case V4L2_YCBCR_ENC_BT2020
:
695 rgb2ycbcr(full
? bt2020_full
: bt2020
, r
, g
, b
, y_offset
, y
, cb
, cr
);
697 case V4L2_YCBCR_ENC_BT2020_CONST_LUM
:
698 lin_y
= (COEFF(0.2627, 255) * rec709_to_linear(r
) +
699 COEFF(0.6780, 255) * rec709_to_linear(g
) +
700 COEFF(0.0593, 255) * rec709_to_linear(b
)) >> 16;
701 yc
= linear_to_rec709(lin_y
);
702 *y
= full
? yc
: (yc
* 219) / 255 + (16 << 4);
704 *cb
= (((b
- yc
) * (full
? bt2020c_full
[0] : bt2020c
[0])) >> 16) + (128 << 4);
706 *cb
= (((b
- yc
) * (full
? bt2020c_full
[1] : bt2020c
[1])) >> 16) + (128 << 4);
708 *cr
= (((r
- yc
) * (full
? bt2020c_full
[2] : bt2020c
[2])) >> 16) + (128 << 4);
710 *cr
= (((r
- yc
) * (full
? bt2020c_full
[3] : bt2020c
[3])) >> 16) + (128 << 4);
712 case V4L2_YCBCR_ENC_SMPTE240M
:
713 rgb2ycbcr(full
? smpte240m_full
: smpte240m
, r
, g
, b
, y_offset
, y
, cb
, cr
);
715 case V4L2_YCBCR_ENC_709
:
717 rgb2ycbcr(full
? rec709_full
: rec709
, r
, g
, b
, y_offset
, y
, cb
, cr
);
722 static void ycbcr2rgb(const int m
[3][3], int y
, int cb
, int cr
,
723 int y_offset
, int *r
, int *g
, int *b
)
728 *r
= m
[0][0] * y
+ m
[0][1] * cb
+ m
[0][2] * cr
;
729 *g
= m
[1][0] * y
+ m
[1][1] * cb
+ m
[1][2] * cr
;
730 *b
= m
[2][0] * y
+ m
[2][1] * cb
+ m
[2][2] * cr
;
731 *r
= clamp(*r
>> 12, 0, 0xff0);
732 *g
= clamp(*g
>> 12, 0, 0xff0);
733 *b
= clamp(*b
>> 12, 0, 0xff0);
736 static void ycbcr_to_color(struct tpg_data
*tpg
, int y
, int cb
, int cr
,
737 int *r
, int *g
, int *b
)
740 #define COEFF(v, r) ((int)(0.5 + (v) * ((255.0 * 255.0 * 16.0) / (r))))
741 static const int bt601
[3][3] = {
742 { COEFF(1, 219), COEFF(0, 224), COEFF(1.4020, 224) },
743 { COEFF(1, 219), COEFF(-0.3441, 224), COEFF(-0.7141, 224) },
744 { COEFF(1, 219), COEFF(1.7720, 224), COEFF(0, 224) },
746 static const int bt601_full
[3][3] = {
747 { COEFF(1, 255), COEFF(0, 255), COEFF(1.4020, 255) },
748 { COEFF(1, 255), COEFF(-0.3441, 255), COEFF(-0.7141, 255) },
749 { COEFF(1, 255), COEFF(1.7720, 255), COEFF(0, 255) },
751 static const int rec709
[3][3] = {
752 { COEFF(1, 219), COEFF(0, 224), COEFF(1.5748, 224) },
753 { COEFF(1, 219), COEFF(-0.1873, 224), COEFF(-0.4681, 224) },
754 { COEFF(1, 219), COEFF(1.8556, 224), COEFF(0, 224) },
756 static const int rec709_full
[3][3] = {
757 { COEFF(1, 255), COEFF(0, 255), COEFF(1.5748, 255) },
758 { COEFF(1, 255), COEFF(-0.1873, 255), COEFF(-0.4681, 255) },
759 { COEFF(1, 255), COEFF(1.8556, 255), COEFF(0, 255) },
761 static const int smpte240m
[3][3] = {
762 { COEFF(1, 219), COEFF(0, 224), COEFF(1.5756, 224) },
763 { COEFF(1, 219), COEFF(-0.2253, 224), COEFF(-0.4767, 224) },
764 { COEFF(1, 219), COEFF(1.8270, 224), COEFF(0, 224) },
766 static const int smpte240m_full
[3][3] = {
767 { COEFF(1, 255), COEFF(0, 255), COEFF(1.5756, 255) },
768 { COEFF(1, 255), COEFF(-0.2253, 255), COEFF(-0.4767, 255) },
769 { COEFF(1, 255), COEFF(1.8270, 255), COEFF(0, 255) },
771 static const int bt2020
[3][3] = {
772 { COEFF(1, 219), COEFF(0, 224), COEFF(1.4746, 224) },
773 { COEFF(1, 219), COEFF(-0.1646, 224), COEFF(-0.5714, 224) },
774 { COEFF(1, 219), COEFF(1.8814, 224), COEFF(0, 224) },
776 static const int bt2020_full
[3][3] = {
777 { COEFF(1, 255), COEFF(0, 255), COEFF(1.4746, 255) },
778 { COEFF(1, 255), COEFF(-0.1646, 255), COEFF(-0.5714, 255) },
779 { COEFF(1, 255), COEFF(1.8814, 255), COEFF(0, 255) },
781 static const int bt2020c
[4] = {
782 COEFF(1.9404, 224), COEFF(1.5816, 224),
783 COEFF(1.7184, 224), COEFF(0.9936, 224),
785 static const int bt2020c_full
[4] = {
786 COEFF(1.9404, 255), COEFF(1.5816, 255),
787 COEFF(1.7184, 255), COEFF(0.9936, 255),
790 bool full
= tpg
->real_quantization
== V4L2_QUANTIZATION_FULL_RANGE
;
791 unsigned y_offset
= full
? 0 : 16;
792 int y_fac
= full
? COEFF(1.0, 255) : COEFF(1.0, 219);
793 int lin_r
, lin_g
, lin_b
, lin_y
;
795 switch (tpg
->real_ycbcr_enc
) {
796 case V4L2_YCBCR_ENC_601
:
797 ycbcr2rgb(full
? bt601_full
: bt601
, y
, cb
, cr
, y_offset
, r
, g
, b
);
799 case V4L2_YCBCR_ENC_XV601
:
800 /* Ignore quantization range, there is only one possible
801 * Y'CbCr encoding. */
802 ycbcr2rgb(bt601
, y
, cb
, cr
, 16, r
, g
, b
);
804 case V4L2_YCBCR_ENC_XV709
:
805 /* Ignore quantization range, there is only one possible
806 * Y'CbCr encoding. */
807 ycbcr2rgb(rec709
, y
, cb
, cr
, 16, r
, g
, b
);
809 case V4L2_YCBCR_ENC_BT2020
:
810 ycbcr2rgb(full
? bt2020_full
: bt2020
, y
, cb
, cr
, y_offset
, r
, g
, b
);
812 case V4L2_YCBCR_ENC_BT2020_CONST_LUM
:
813 y
-= full
? 0 : 16 << 4;
818 *b
= y_fac
* y
+ (full
? bt2020c_full
[0] : bt2020c
[0]) * cb
;
820 *b
= y_fac
* y
+ (full
? bt2020c_full
[1] : bt2020c
[1]) * cb
;
823 *r
= y_fac
* y
+ (full
? bt2020c_full
[2] : bt2020c
[2]) * cr
;
825 *r
= y_fac
* y
+ (full
? bt2020c_full
[3] : bt2020c
[3]) * cr
;
827 lin_r
= rec709_to_linear(*r
);
828 lin_b
= rec709_to_linear(*b
);
829 lin_y
= rec709_to_linear((y
* 255) / (full
? 255 : 219));
831 lin_g
= COEFF(1.0 / 0.6780, 255) * lin_y
-
832 COEFF(0.2627 / 0.6780, 255) * lin_r
-
833 COEFF(0.0593 / 0.6780, 255) * lin_b
;
834 *g
= linear_to_rec709(lin_g
>> 12);
836 case V4L2_YCBCR_ENC_SMPTE240M
:
837 ycbcr2rgb(full
? smpte240m_full
: smpte240m
, y
, cb
, cr
, y_offset
, r
, g
, b
);
839 case V4L2_YCBCR_ENC_709
:
841 ycbcr2rgb(full
? rec709_full
: rec709
, y
, cb
, cr
, y_offset
, r
, g
, b
);
846 /* precalculate color bar values to speed up rendering */
847 static void precalculate_color(struct tpg_data
*tpg
, int k
)
850 int r
= tpg_colors
[col
].r
;
851 int g
= tpg_colors
[col
].g
;
852 int b
= tpg_colors
[col
].b
;
854 bool ycbcr_valid
= false;
856 if (k
== TPG_COLOR_TEXTBG
) {
857 col
= tpg_get_textbg_color(tpg
);
859 r
= tpg_colors
[col
].r
;
860 g
= tpg_colors
[col
].g
;
861 b
= tpg_colors
[col
].b
;
862 } else if (k
== TPG_COLOR_TEXTFG
) {
863 col
= tpg_get_textfg_color(tpg
);
865 r
= tpg_colors
[col
].r
;
866 g
= tpg_colors
[col
].g
;
867 b
= tpg_colors
[col
].b
;
868 } else if (tpg
->pattern
== TPG_PAT_NOISE
) {
869 r
= g
= b
= prandom_u32_max(256);
870 } else if (k
== TPG_COLOR_RANDOM
) {
871 r
= g
= b
= tpg
->qual_offset
+ prandom_u32_max(196);
872 } else if (k
>= TPG_COLOR_RAMP
) {
873 r
= g
= b
= k
- TPG_COLOR_RAMP
;
876 if (tpg
->pattern
== TPG_PAT_CSC_COLORBAR
&& col
<= TPG_COLOR_CSC_BLACK
) {
877 r
= tpg_csc_colors
[tpg
->colorspace
][tpg
->real_xfer_func
][col
].r
;
878 g
= tpg_csc_colors
[tpg
->colorspace
][tpg
->real_xfer_func
][col
].g
;
879 b
= tpg_csc_colors
[tpg
->colorspace
][tpg
->real_xfer_func
][col
].b
;
886 if (tpg
->qual
== TPG_QUAL_GRAY
||
887 tpg
->color_enc
== TGP_COLOR_ENC_LUMA
) {
888 /* Rec. 709 Luma function */
889 /* (0.2126, 0.7152, 0.0722) * (255 * 256) */
890 r
= g
= b
= (13879 * r
+ 46688 * g
+ 4713 * b
) >> 16;
894 * The assumption is that the RGB output is always full range,
895 * so only if the rgb_range overrides the 'real' rgb range do
896 * we need to convert the RGB values.
898 * Remember that r, g and b are still in the 0 - 0xff0 range.
900 if (tpg
->real_rgb_range
== V4L2_DV_RGB_RANGE_LIMITED
&&
901 tpg
->rgb_range
== V4L2_DV_RGB_RANGE_FULL
&&
902 tpg
->color_enc
== TGP_COLOR_ENC_RGB
) {
904 * Convert from full range (which is what r, g and b are)
905 * to limited range (which is the 'real' RGB range), which
906 * is then interpreted as full range.
908 r
= (r
* 219) / 255 + (16 << 4);
909 g
= (g
* 219) / 255 + (16 << 4);
910 b
= (b
* 219) / 255 + (16 << 4);
911 } else if (tpg
->real_rgb_range
!= V4L2_DV_RGB_RANGE_LIMITED
&&
912 tpg
->rgb_range
== V4L2_DV_RGB_RANGE_LIMITED
&&
913 tpg
->color_enc
== TGP_COLOR_ENC_RGB
) {
916 * Clamp r, g and b to the limited range and convert to full
917 * range since that's what we deliver.
919 r
= clamp(r
, 16 << 4, 235 << 4);
920 g
= clamp(g
, 16 << 4, 235 << 4);
921 b
= clamp(b
, 16 << 4, 235 << 4);
922 r
= (r
- (16 << 4)) * 255 / 219;
923 g
= (g
- (16 << 4)) * 255 / 219;
924 b
= (b
- (16 << 4)) * 255 / 219;
927 if ((tpg
->brightness
!= 128 || tpg
->contrast
!= 128 ||
928 tpg
->saturation
!= 128 || tpg
->hue
) &&
929 tpg
->color_enc
!= TGP_COLOR_ENC_LUMA
) {
930 /* Implement these operations */
933 /* First convert to YCbCr */
935 color_to_ycbcr(tpg
, r
, g
, b
, &y
, &cb
, &cr
);
937 y
= (16 << 4) + ((y
- (16 << 4)) * tpg
->contrast
) / 128;
938 y
+= (tpg
->brightness
<< 4) - (128 << 4);
942 tmp_cb
= (cb
* cos(128 + tpg
->hue
)) / 127 + (cr
* sin
[128 + tpg
->hue
]) / 127;
943 tmp_cr
= (cr
* cos(128 + tpg
->hue
)) / 127 - (cb
* sin
[128 + tpg
->hue
]) / 127;
945 cb
= (128 << 4) + (tmp_cb
* tpg
->contrast
* tpg
->saturation
) / (128 * 128);
946 cr
= (128 << 4) + (tmp_cr
* tpg
->contrast
* tpg
->saturation
) / (128 * 128);
947 if (tpg
->color_enc
== TGP_COLOR_ENC_YCBCR
)
950 ycbcr_to_color(tpg
, y
, cb
, cr
, &r
, &g
, &b
);
951 } else if ((tpg
->brightness
!= 128 || tpg
->contrast
!= 128) &&
952 tpg
->color_enc
== TGP_COLOR_ENC_LUMA
) {
953 r
= (16 << 4) + ((r
- (16 << 4)) * tpg
->contrast
) / 128;
954 r
+= (tpg
->brightness
<< 4) - (128 << 4);
957 switch (tpg
->color_enc
) {
958 case TGP_COLOR_ENC_HSV
:
962 color_to_hsv(tpg
, r
, g
, b
, &h
, &s
, &v
);
963 tpg
->colors
[k
][0] = h
;
964 tpg
->colors
[k
][1] = s
;
965 tpg
->colors
[k
][2] = v
;
968 case TGP_COLOR_ENC_YCBCR
:
970 /* Convert to YCbCr */
972 color_to_ycbcr(tpg
, r
, g
, b
, &y
, &cb
, &cr
);
978 * XV601/709 use the header/footer margins to encode R', G'
979 * and B' values outside the range [0-1]. So do not clamp
982 if (tpg
->real_quantization
== V4L2_QUANTIZATION_LIM_RANGE
&&
983 tpg
->real_ycbcr_enc
!= V4L2_YCBCR_ENC_XV601
&&
984 tpg
->real_ycbcr_enc
!= V4L2_YCBCR_ENC_XV709
) {
985 y
= clamp(y
, 16, 235);
986 cb
= clamp(cb
, 16, 240);
987 cr
= clamp(cr
, 16, 240);
989 y
= clamp(y
, 1, 254);
990 cb
= clamp(cb
, 1, 254);
991 cr
= clamp(cr
, 1, 254);
993 switch (tpg
->fourcc
) {
994 case V4L2_PIX_FMT_YUV444
:
999 case V4L2_PIX_FMT_YUV555
:
1004 case V4L2_PIX_FMT_YUV565
:
1010 tpg
->colors
[k
][0] = y
;
1011 tpg
->colors
[k
][1] = cb
;
1012 tpg
->colors
[k
][2] = cr
;
1015 case TGP_COLOR_ENC_LUMA
:
1017 tpg
->colors
[k
][0] = r
>> 4;
1020 case TGP_COLOR_ENC_RGB
:
1022 if (tpg
->real_quantization
== V4L2_QUANTIZATION_LIM_RANGE
) {
1023 r
= (r
* 219) / 255 + (16 << 4);
1024 g
= (g
* 219) / 255 + (16 << 4);
1025 b
= (b
* 219) / 255 + (16 << 4);
1027 switch (tpg
->fourcc
) {
1028 case V4L2_PIX_FMT_RGB332
:
1033 case V4L2_PIX_FMT_RGB565
:
1034 case V4L2_PIX_FMT_RGB565X
:
1039 case V4L2_PIX_FMT_RGB444
:
1040 case V4L2_PIX_FMT_XRGB444
:
1041 case V4L2_PIX_FMT_ARGB444
:
1042 case V4L2_PIX_FMT_RGBX444
:
1043 case V4L2_PIX_FMT_RGBA444
:
1044 case V4L2_PIX_FMT_XBGR444
:
1045 case V4L2_PIX_FMT_ABGR444
:
1046 case V4L2_PIX_FMT_BGRX444
:
1047 case V4L2_PIX_FMT_BGRA444
:
1052 case V4L2_PIX_FMT_RGB555
:
1053 case V4L2_PIX_FMT_XRGB555
:
1054 case V4L2_PIX_FMT_ARGB555
:
1055 case V4L2_PIX_FMT_RGBX555
:
1056 case V4L2_PIX_FMT_RGBA555
:
1057 case V4L2_PIX_FMT_XBGR555
:
1058 case V4L2_PIX_FMT_ABGR555
:
1059 case V4L2_PIX_FMT_BGRX555
:
1060 case V4L2_PIX_FMT_BGRA555
:
1061 case V4L2_PIX_FMT_RGB555X
:
1062 case V4L2_PIX_FMT_XRGB555X
:
1063 case V4L2_PIX_FMT_ARGB555X
:
1068 case V4L2_PIX_FMT_BGR666
:
1080 tpg
->colors
[k
][0] = r
;
1081 tpg
->colors
[k
][1] = g
;
1082 tpg
->colors
[k
][2] = b
;
1088 static void tpg_precalculate_colors(struct tpg_data
*tpg
)
1092 for (k
= 0; k
< TPG_COLOR_MAX
; k
++)
1093 precalculate_color(tpg
, k
);
1096 /* 'odd' is true for pixels 1, 3, 5, etc. and false for pixels 0, 2, 4, etc. */
1097 static void gen_twopix(struct tpg_data
*tpg
,
1098 u8 buf
[TPG_MAX_PLANES
][8], int color
, bool odd
)
1100 unsigned offset
= odd
* tpg
->twopixelsize
[0] / 2;
1101 u8 alpha
= tpg
->alpha_component
;
1102 u8 r_y_h
, g_u_s
, b_v
;
1104 if (tpg
->alpha_red_only
&& color
!= TPG_COLOR_CSC_RED
&&
1105 color
!= TPG_COLOR_100_RED
&&
1106 color
!= TPG_COLOR_75_RED
)
1108 if (color
== TPG_COLOR_RANDOM
)
1109 precalculate_color(tpg
, color
);
1110 r_y_h
= tpg
->colors
[color
][0]; /* R or precalculated Y, H */
1111 g_u_s
= tpg
->colors
[color
][1]; /* G or precalculated U, V */
1112 b_v
= tpg
->colors
[color
][2]; /* B or precalculated V */
1114 switch (tpg
->fourcc
) {
1115 case V4L2_PIX_FMT_GREY
:
1116 buf
[0][offset
] = r_y_h
;
1118 case V4L2_PIX_FMT_Y10
:
1119 buf
[0][offset
] = (r_y_h
<< 2) & 0xff;
1120 buf
[0][offset
+1] = r_y_h
>> 6;
1122 case V4L2_PIX_FMT_Y12
:
1123 buf
[0][offset
] = (r_y_h
<< 4) & 0xff;
1124 buf
[0][offset
+1] = r_y_h
>> 4;
1126 case V4L2_PIX_FMT_Y16
:
1127 case V4L2_PIX_FMT_Z16
:
1129 * Ideally both bytes should be set to r_y_h, but then you won't
1130 * be able to detect endian problems. So keep it 0 except for
1131 * the corner case where r_y_h is 0xff so white really will be
1134 buf
[0][offset
] = r_y_h
== 0xff ? r_y_h
: 0;
1135 buf
[0][offset
+1] = r_y_h
;
1137 case V4L2_PIX_FMT_Y16_BE
:
1138 /* See comment for V4L2_PIX_FMT_Y16 above */
1139 buf
[0][offset
] = r_y_h
;
1140 buf
[0][offset
+1] = r_y_h
== 0xff ? r_y_h
: 0;
1142 case V4L2_PIX_FMT_YUV422M
:
1143 case V4L2_PIX_FMT_YUV422P
:
1144 case V4L2_PIX_FMT_YUV420
:
1145 case V4L2_PIX_FMT_YUV420M
:
1146 buf
[0][offset
] = r_y_h
;
1148 buf
[1][0] = (buf
[1][0] + g_u_s
) / 2;
1149 buf
[2][0] = (buf
[2][0] + b_v
) / 2;
1150 buf
[1][1] = buf
[1][0];
1151 buf
[2][1] = buf
[2][0];
1157 case V4L2_PIX_FMT_YVU422M
:
1158 case V4L2_PIX_FMT_YVU420
:
1159 case V4L2_PIX_FMT_YVU420M
:
1160 buf
[0][offset
] = r_y_h
;
1162 buf
[1][0] = (buf
[1][0] + b_v
) / 2;
1163 buf
[2][0] = (buf
[2][0] + g_u_s
) / 2;
1164 buf
[1][1] = buf
[1][0];
1165 buf
[2][1] = buf
[2][0];
1172 case V4L2_PIX_FMT_NV12
:
1173 case V4L2_PIX_FMT_NV12M
:
1174 case V4L2_PIX_FMT_NV16
:
1175 case V4L2_PIX_FMT_NV16M
:
1176 buf
[0][offset
] = r_y_h
;
1178 buf
[1][0] = (buf
[1][0] + g_u_s
) / 2;
1179 buf
[1][1] = (buf
[1][1] + b_v
) / 2;
1185 case V4L2_PIX_FMT_NV21
:
1186 case V4L2_PIX_FMT_NV21M
:
1187 case V4L2_PIX_FMT_NV61
:
1188 case V4L2_PIX_FMT_NV61M
:
1189 buf
[0][offset
] = r_y_h
;
1191 buf
[1][0] = (buf
[1][0] + b_v
) / 2;
1192 buf
[1][1] = (buf
[1][1] + g_u_s
) / 2;
1199 case V4L2_PIX_FMT_YUV444M
:
1200 buf
[0][offset
] = r_y_h
;
1201 buf
[1][offset
] = g_u_s
;
1202 buf
[2][offset
] = b_v
;
1205 case V4L2_PIX_FMT_YVU444M
:
1206 buf
[0][offset
] = r_y_h
;
1207 buf
[1][offset
] = b_v
;
1208 buf
[2][offset
] = g_u_s
;
1211 case V4L2_PIX_FMT_NV24
:
1212 buf
[0][offset
] = r_y_h
;
1213 buf
[1][2 * offset
] = g_u_s
;
1214 buf
[1][(2 * offset
+ 1) % 8] = b_v
;
1217 case V4L2_PIX_FMT_NV42
:
1218 buf
[0][offset
] = r_y_h
;
1219 buf
[1][2 * offset
] = b_v
;
1220 buf
[1][(2 * offset
+ 1) % 8] = g_u_s
;
1223 case V4L2_PIX_FMT_YUYV
:
1224 buf
[0][offset
] = r_y_h
;
1226 buf
[0][1] = (buf
[0][1] + g_u_s
) / 2;
1227 buf
[0][3] = (buf
[0][3] + b_v
) / 2;
1233 case V4L2_PIX_FMT_UYVY
:
1234 buf
[0][offset
+ 1] = r_y_h
;
1236 buf
[0][0] = (buf
[0][0] + g_u_s
) / 2;
1237 buf
[0][2] = (buf
[0][2] + b_v
) / 2;
1243 case V4L2_PIX_FMT_YVYU
:
1244 buf
[0][offset
] = r_y_h
;
1246 buf
[0][1] = (buf
[0][1] + b_v
) / 2;
1247 buf
[0][3] = (buf
[0][3] + g_u_s
) / 2;
1253 case V4L2_PIX_FMT_VYUY
:
1254 buf
[0][offset
+ 1] = r_y_h
;
1256 buf
[0][0] = (buf
[0][0] + b_v
) / 2;
1257 buf
[0][2] = (buf
[0][2] + g_u_s
) / 2;
1263 case V4L2_PIX_FMT_RGB332
:
1264 buf
[0][offset
] = (r_y_h
<< 5) | (g_u_s
<< 2) | b_v
;
1266 case V4L2_PIX_FMT_YUV565
:
1267 case V4L2_PIX_FMT_RGB565
:
1268 buf
[0][offset
] = (g_u_s
<< 5) | b_v
;
1269 buf
[0][offset
+ 1] = (r_y_h
<< 3) | (g_u_s
>> 3);
1271 case V4L2_PIX_FMT_RGB565X
:
1272 buf
[0][offset
] = (r_y_h
<< 3) | (g_u_s
>> 3);
1273 buf
[0][offset
+ 1] = (g_u_s
<< 5) | b_v
;
1275 case V4L2_PIX_FMT_RGB444
:
1276 case V4L2_PIX_FMT_XRGB444
:
1279 case V4L2_PIX_FMT_YUV444
:
1280 case V4L2_PIX_FMT_ARGB444
:
1281 buf
[0][offset
] = (g_u_s
<< 4) | b_v
;
1282 buf
[0][offset
+ 1] = (alpha
& 0xf0) | r_y_h
;
1284 case V4L2_PIX_FMT_RGBX444
:
1287 case V4L2_PIX_FMT_RGBA444
:
1288 buf
[0][offset
] = (b_v
<< 4) | (alpha
>> 4);
1289 buf
[0][offset
+ 1] = (r_y_h
<< 4) | g_u_s
;
1291 case V4L2_PIX_FMT_XBGR444
:
1294 case V4L2_PIX_FMT_ABGR444
:
1295 buf
[0][offset
] = (g_u_s
<< 4) | r_y_h
;
1296 buf
[0][offset
+ 1] = (alpha
& 0xf0) | b_v
;
1298 case V4L2_PIX_FMT_BGRX444
:
1301 case V4L2_PIX_FMT_BGRA444
:
1302 buf
[0][offset
] = (r_y_h
<< 4) | (alpha
>> 4);
1303 buf
[0][offset
+ 1] = (b_v
<< 4) | g_u_s
;
1305 case V4L2_PIX_FMT_RGB555
:
1306 case V4L2_PIX_FMT_XRGB555
:
1309 case V4L2_PIX_FMT_YUV555
:
1310 case V4L2_PIX_FMT_ARGB555
:
1311 buf
[0][offset
] = (g_u_s
<< 5) | b_v
;
1312 buf
[0][offset
+ 1] = (alpha
& 0x80) | (r_y_h
<< 2)
1315 case V4L2_PIX_FMT_RGBX555
:
1318 case V4L2_PIX_FMT_RGBA555
:
1319 buf
[0][offset
] = (g_u_s
<< 6) | (b_v
<< 1) |
1320 ((alpha
& 0x80) >> 7);
1321 buf
[0][offset
+ 1] = (r_y_h
<< 3) | (g_u_s
>> 2);
1323 case V4L2_PIX_FMT_XBGR555
:
1326 case V4L2_PIX_FMT_ABGR555
:
1327 buf
[0][offset
] = (g_u_s
<< 5) | r_y_h
;
1328 buf
[0][offset
+ 1] = (alpha
& 0x80) | (b_v
<< 2)
1331 case V4L2_PIX_FMT_BGRX555
:
1334 case V4L2_PIX_FMT_BGRA555
:
1335 buf
[0][offset
] = (g_u_s
<< 6) | (r_y_h
<< 1) |
1336 ((alpha
& 0x80) >> 7);
1337 buf
[0][offset
+ 1] = (b_v
<< 3) | (g_u_s
>> 2);
1339 case V4L2_PIX_FMT_RGB555X
:
1340 case V4L2_PIX_FMT_XRGB555X
:
1343 case V4L2_PIX_FMT_ARGB555X
:
1344 buf
[0][offset
] = (alpha
& 0x80) | (r_y_h
<< 2) | (g_u_s
>> 3);
1345 buf
[0][offset
+ 1] = (g_u_s
<< 5) | b_v
;
1347 case V4L2_PIX_FMT_RGB24
:
1348 case V4L2_PIX_FMT_HSV24
:
1349 buf
[0][offset
] = r_y_h
;
1350 buf
[0][offset
+ 1] = g_u_s
;
1351 buf
[0][offset
+ 2] = b_v
;
1353 case V4L2_PIX_FMT_BGR24
:
1354 buf
[0][offset
] = b_v
;
1355 buf
[0][offset
+ 1] = g_u_s
;
1356 buf
[0][offset
+ 2] = r_y_h
;
1358 case V4L2_PIX_FMT_BGR666
:
1359 buf
[0][offset
] = (b_v
<< 2) | (g_u_s
>> 4);
1360 buf
[0][offset
+ 1] = (g_u_s
<< 4) | (r_y_h
>> 2);
1361 buf
[0][offset
+ 2] = r_y_h
<< 6;
1362 buf
[0][offset
+ 3] = 0;
1364 case V4L2_PIX_FMT_RGB32
:
1365 case V4L2_PIX_FMT_XRGB32
:
1366 case V4L2_PIX_FMT_HSV32
:
1367 case V4L2_PIX_FMT_XYUV32
:
1370 case V4L2_PIX_FMT_YUV32
:
1371 case V4L2_PIX_FMT_ARGB32
:
1372 case V4L2_PIX_FMT_AYUV32
:
1373 buf
[0][offset
] = alpha
;
1374 buf
[0][offset
+ 1] = r_y_h
;
1375 buf
[0][offset
+ 2] = g_u_s
;
1376 buf
[0][offset
+ 3] = b_v
;
1378 case V4L2_PIX_FMT_RGBX32
:
1381 case V4L2_PIX_FMT_RGBA32
:
1382 buf
[0][offset
] = r_y_h
;
1383 buf
[0][offset
+ 1] = g_u_s
;
1384 buf
[0][offset
+ 2] = b_v
;
1385 buf
[0][offset
+ 3] = alpha
;
1387 case V4L2_PIX_FMT_BGR32
:
1388 case V4L2_PIX_FMT_XBGR32
:
1389 case V4L2_PIX_FMT_VUYX32
:
1392 case V4L2_PIX_FMT_ABGR32
:
1393 case V4L2_PIX_FMT_VUYA32
:
1394 buf
[0][offset
] = b_v
;
1395 buf
[0][offset
+ 1] = g_u_s
;
1396 buf
[0][offset
+ 2] = r_y_h
;
1397 buf
[0][offset
+ 3] = alpha
;
1399 case V4L2_PIX_FMT_BGRX32
:
1402 case V4L2_PIX_FMT_BGRA32
:
1403 buf
[0][offset
] = alpha
;
1404 buf
[0][offset
+ 1] = b_v
;
1405 buf
[0][offset
+ 2] = g_u_s
;
1406 buf
[0][offset
+ 3] = r_y_h
;
1408 case V4L2_PIX_FMT_SBGGR8
:
1409 buf
[0][offset
] = odd
? g_u_s
: b_v
;
1410 buf
[1][offset
] = odd
? r_y_h
: g_u_s
;
1412 case V4L2_PIX_FMT_SGBRG8
:
1413 buf
[0][offset
] = odd
? b_v
: g_u_s
;
1414 buf
[1][offset
] = odd
? g_u_s
: r_y_h
;
1416 case V4L2_PIX_FMT_SGRBG8
:
1417 buf
[0][offset
] = odd
? r_y_h
: g_u_s
;
1418 buf
[1][offset
] = odd
? g_u_s
: b_v
;
1420 case V4L2_PIX_FMT_SRGGB8
:
1421 buf
[0][offset
] = odd
? g_u_s
: r_y_h
;
1422 buf
[1][offset
] = odd
? b_v
: g_u_s
;
1424 case V4L2_PIX_FMT_SBGGR10
:
1425 buf
[0][offset
] = odd
? g_u_s
<< 2 : b_v
<< 2;
1426 buf
[0][offset
+ 1] = odd
? g_u_s
>> 6 : b_v
>> 6;
1427 buf
[1][offset
] = odd
? r_y_h
<< 2 : g_u_s
<< 2;
1428 buf
[1][offset
+ 1] = odd
? r_y_h
>> 6 : g_u_s
>> 6;
1429 buf
[0][offset
] |= (buf
[0][offset
] >> 2) & 3;
1430 buf
[1][offset
] |= (buf
[1][offset
] >> 2) & 3;
1432 case V4L2_PIX_FMT_SGBRG10
:
1433 buf
[0][offset
] = odd
? b_v
<< 2 : g_u_s
<< 2;
1434 buf
[0][offset
+ 1] = odd
? b_v
>> 6 : g_u_s
>> 6;
1435 buf
[1][offset
] = odd
? g_u_s
<< 2 : r_y_h
<< 2;
1436 buf
[1][offset
+ 1] = odd
? g_u_s
>> 6 : r_y_h
>> 6;
1437 buf
[0][offset
] |= (buf
[0][offset
] >> 2) & 3;
1438 buf
[1][offset
] |= (buf
[1][offset
] >> 2) & 3;
1440 case V4L2_PIX_FMT_SGRBG10
:
1441 buf
[0][offset
] = odd
? r_y_h
<< 2 : g_u_s
<< 2;
1442 buf
[0][offset
+ 1] = odd
? r_y_h
>> 6 : g_u_s
>> 6;
1443 buf
[1][offset
] = odd
? g_u_s
<< 2 : b_v
<< 2;
1444 buf
[1][offset
+ 1] = odd
? g_u_s
>> 6 : b_v
>> 6;
1445 buf
[0][offset
] |= (buf
[0][offset
] >> 2) & 3;
1446 buf
[1][offset
] |= (buf
[1][offset
] >> 2) & 3;
1448 case V4L2_PIX_FMT_SRGGB10
:
1449 buf
[0][offset
] = odd
? g_u_s
<< 2 : r_y_h
<< 2;
1450 buf
[0][offset
+ 1] = odd
? g_u_s
>> 6 : r_y_h
>> 6;
1451 buf
[1][offset
] = odd
? b_v
<< 2 : g_u_s
<< 2;
1452 buf
[1][offset
+ 1] = odd
? b_v
>> 6 : g_u_s
>> 6;
1453 buf
[0][offset
] |= (buf
[0][offset
] >> 2) & 3;
1454 buf
[1][offset
] |= (buf
[1][offset
] >> 2) & 3;
1456 case V4L2_PIX_FMT_SBGGR12
:
1457 buf
[0][offset
] = odd
? g_u_s
<< 4 : b_v
<< 4;
1458 buf
[0][offset
+ 1] = odd
? g_u_s
>> 4 : b_v
>> 4;
1459 buf
[1][offset
] = odd
? r_y_h
<< 4 : g_u_s
<< 4;
1460 buf
[1][offset
+ 1] = odd
? r_y_h
>> 4 : g_u_s
>> 4;
1461 buf
[0][offset
] |= (buf
[0][offset
] >> 4) & 0xf;
1462 buf
[1][offset
] |= (buf
[1][offset
] >> 4) & 0xf;
1464 case V4L2_PIX_FMT_SGBRG12
:
1465 buf
[0][offset
] = odd
? b_v
<< 4 : g_u_s
<< 4;
1466 buf
[0][offset
+ 1] = odd
? b_v
>> 4 : g_u_s
>> 4;
1467 buf
[1][offset
] = odd
? g_u_s
<< 4 : r_y_h
<< 4;
1468 buf
[1][offset
+ 1] = odd
? g_u_s
>> 4 : r_y_h
>> 4;
1469 buf
[0][offset
] |= (buf
[0][offset
] >> 4) & 0xf;
1470 buf
[1][offset
] |= (buf
[1][offset
] >> 4) & 0xf;
1472 case V4L2_PIX_FMT_SGRBG12
:
1473 buf
[0][offset
] = odd
? r_y_h
<< 4 : g_u_s
<< 4;
1474 buf
[0][offset
+ 1] = odd
? r_y_h
>> 4 : g_u_s
>> 4;
1475 buf
[1][offset
] = odd
? g_u_s
<< 4 : b_v
<< 4;
1476 buf
[1][offset
+ 1] = odd
? g_u_s
>> 4 : b_v
>> 4;
1477 buf
[0][offset
] |= (buf
[0][offset
] >> 4) & 0xf;
1478 buf
[1][offset
] |= (buf
[1][offset
] >> 4) & 0xf;
1480 case V4L2_PIX_FMT_SRGGB12
:
1481 buf
[0][offset
] = odd
? g_u_s
<< 4 : r_y_h
<< 4;
1482 buf
[0][offset
+ 1] = odd
? g_u_s
>> 4 : r_y_h
>> 4;
1483 buf
[1][offset
] = odd
? b_v
<< 4 : g_u_s
<< 4;
1484 buf
[1][offset
+ 1] = odd
? b_v
>> 4 : g_u_s
>> 4;
1485 buf
[0][offset
] |= (buf
[0][offset
] >> 4) & 0xf;
1486 buf
[1][offset
] |= (buf
[1][offset
] >> 4) & 0xf;
1488 case V4L2_PIX_FMT_SBGGR16
:
1489 buf
[0][offset
] = buf
[0][offset
+ 1] = odd
? g_u_s
: b_v
;
1490 buf
[1][offset
] = buf
[1][offset
+ 1] = odd
? r_y_h
: g_u_s
;
1492 case V4L2_PIX_FMT_SGBRG16
:
1493 buf
[0][offset
] = buf
[0][offset
+ 1] = odd
? b_v
: g_u_s
;
1494 buf
[1][offset
] = buf
[1][offset
+ 1] = odd
? g_u_s
: r_y_h
;
1496 case V4L2_PIX_FMT_SGRBG16
:
1497 buf
[0][offset
] = buf
[0][offset
+ 1] = odd
? r_y_h
: g_u_s
;
1498 buf
[1][offset
] = buf
[1][offset
+ 1] = odd
? g_u_s
: b_v
;
1500 case V4L2_PIX_FMT_SRGGB16
:
1501 buf
[0][offset
] = buf
[0][offset
+ 1] = odd
? g_u_s
: r_y_h
;
1502 buf
[1][offset
] = buf
[1][offset
+ 1] = odd
? b_v
: g_u_s
;
1507 unsigned tpg_g_interleaved_plane(const struct tpg_data
*tpg
, unsigned buf_line
)
1509 switch (tpg
->fourcc
) {
1510 case V4L2_PIX_FMT_SBGGR8
:
1511 case V4L2_PIX_FMT_SGBRG8
:
1512 case V4L2_PIX_FMT_SGRBG8
:
1513 case V4L2_PIX_FMT_SRGGB8
:
1514 case V4L2_PIX_FMT_SBGGR10
:
1515 case V4L2_PIX_FMT_SGBRG10
:
1516 case V4L2_PIX_FMT_SGRBG10
:
1517 case V4L2_PIX_FMT_SRGGB10
:
1518 case V4L2_PIX_FMT_SBGGR12
:
1519 case V4L2_PIX_FMT_SGBRG12
:
1520 case V4L2_PIX_FMT_SGRBG12
:
1521 case V4L2_PIX_FMT_SRGGB12
:
1522 case V4L2_PIX_FMT_SBGGR16
:
1523 case V4L2_PIX_FMT_SGBRG16
:
1524 case V4L2_PIX_FMT_SGRBG16
:
1525 case V4L2_PIX_FMT_SRGGB16
:
1526 return buf_line
& 1;
1531 EXPORT_SYMBOL_GPL(tpg_g_interleaved_plane
);
1533 /* Return how many pattern lines are used by the current pattern. */
1534 static unsigned tpg_get_pat_lines(const struct tpg_data
*tpg
)
1536 switch (tpg
->pattern
) {
1537 case TPG_PAT_CHECKERS_16X16
:
1538 case TPG_PAT_CHECKERS_2X2
:
1539 case TPG_PAT_CHECKERS_1X1
:
1540 case TPG_PAT_COLOR_CHECKERS_2X2
:
1541 case TPG_PAT_COLOR_CHECKERS_1X1
:
1542 case TPG_PAT_ALTERNATING_HLINES
:
1543 case TPG_PAT_CROSS_1_PIXEL
:
1544 case TPG_PAT_CROSS_2_PIXELS
:
1545 case TPG_PAT_CROSS_10_PIXELS
:
1547 case TPG_PAT_100_COLORSQUARES
:
1548 case TPG_PAT_100_HCOLORBAR
:
1555 /* Which pattern line should be used for the given frame line. */
1556 static unsigned tpg_get_pat_line(const struct tpg_data
*tpg
, unsigned line
)
1558 switch (tpg
->pattern
) {
1559 case TPG_PAT_CHECKERS_16X16
:
1560 return (line
>> 4) & 1;
1561 case TPG_PAT_CHECKERS_1X1
:
1562 case TPG_PAT_COLOR_CHECKERS_1X1
:
1563 case TPG_PAT_ALTERNATING_HLINES
:
1565 case TPG_PAT_CHECKERS_2X2
:
1566 case TPG_PAT_COLOR_CHECKERS_2X2
:
1567 return (line
& 2) >> 1;
1568 case TPG_PAT_100_COLORSQUARES
:
1569 case TPG_PAT_100_HCOLORBAR
:
1570 return (line
* 8) / tpg
->src_height
;
1571 case TPG_PAT_CROSS_1_PIXEL
:
1572 return line
== tpg
->src_height
/ 2;
1573 case TPG_PAT_CROSS_2_PIXELS
:
1574 return (line
+ 1) / 2 == tpg
->src_height
/ 4;
1575 case TPG_PAT_CROSS_10_PIXELS
:
1576 return (line
+ 10) / 20 == tpg
->src_height
/ 40;
1583 * Which color should be used for the given pattern line and X coordinate.
1584 * Note: x is in the range 0 to 2 * tpg->src_width.
1586 static enum tpg_color
tpg_get_color(const struct tpg_data
*tpg
,
1587 unsigned pat_line
, unsigned x
)
1589 /* Maximum number of bars are TPG_COLOR_MAX - otherwise, the input print code
1590 should be modified */
1591 static const enum tpg_color bars
[3][8] = {
1592 /* Standard ITU-R 75% color bar sequence */
1593 { TPG_COLOR_CSC_WHITE
, TPG_COLOR_75_YELLOW
,
1594 TPG_COLOR_75_CYAN
, TPG_COLOR_75_GREEN
,
1595 TPG_COLOR_75_MAGENTA
, TPG_COLOR_75_RED
,
1596 TPG_COLOR_75_BLUE
, TPG_COLOR_100_BLACK
, },
1597 /* Standard ITU-R 100% color bar sequence */
1598 { TPG_COLOR_100_WHITE
, TPG_COLOR_100_YELLOW
,
1599 TPG_COLOR_100_CYAN
, TPG_COLOR_100_GREEN
,
1600 TPG_COLOR_100_MAGENTA
, TPG_COLOR_100_RED
,
1601 TPG_COLOR_100_BLUE
, TPG_COLOR_100_BLACK
, },
1602 /* Color bar sequence suitable to test CSC */
1603 { TPG_COLOR_CSC_WHITE
, TPG_COLOR_CSC_YELLOW
,
1604 TPG_COLOR_CSC_CYAN
, TPG_COLOR_CSC_GREEN
,
1605 TPG_COLOR_CSC_MAGENTA
, TPG_COLOR_CSC_RED
,
1606 TPG_COLOR_CSC_BLUE
, TPG_COLOR_CSC_BLACK
, },
1609 switch (tpg
->pattern
) {
1610 case TPG_PAT_75_COLORBAR
:
1611 case TPG_PAT_100_COLORBAR
:
1612 case TPG_PAT_CSC_COLORBAR
:
1613 return bars
[tpg
->pattern
][((x
* 8) / tpg
->src_width
) % 8];
1614 case TPG_PAT_100_COLORSQUARES
:
1615 return bars
[1][(pat_line
+ (x
* 8) / tpg
->src_width
) % 8];
1616 case TPG_PAT_100_HCOLORBAR
:
1617 return bars
[1][pat_line
];
1619 return TPG_COLOR_100_BLACK
;
1621 return TPG_COLOR_100_WHITE
;
1623 return TPG_COLOR_100_RED
;
1625 return TPG_COLOR_100_GREEN
;
1627 return TPG_COLOR_100_BLUE
;
1628 case TPG_PAT_CHECKERS_16X16
:
1629 return (((x
>> 4) & 1) ^ (pat_line
& 1)) ?
1630 TPG_COLOR_100_BLACK
: TPG_COLOR_100_WHITE
;
1631 case TPG_PAT_CHECKERS_1X1
:
1632 return ((x
& 1) ^ (pat_line
& 1)) ?
1633 TPG_COLOR_100_WHITE
: TPG_COLOR_100_BLACK
;
1634 case TPG_PAT_COLOR_CHECKERS_1X1
:
1635 return ((x
& 1) ^ (pat_line
& 1)) ?
1636 TPG_COLOR_100_RED
: TPG_COLOR_100_BLUE
;
1637 case TPG_PAT_CHECKERS_2X2
:
1638 return (((x
>> 1) & 1) ^ (pat_line
& 1)) ?
1639 TPG_COLOR_100_WHITE
: TPG_COLOR_100_BLACK
;
1640 case TPG_PAT_COLOR_CHECKERS_2X2
:
1641 return (((x
>> 1) & 1) ^ (pat_line
& 1)) ?
1642 TPG_COLOR_100_RED
: TPG_COLOR_100_BLUE
;
1643 case TPG_PAT_ALTERNATING_HLINES
:
1644 return pat_line
? TPG_COLOR_100_WHITE
: TPG_COLOR_100_BLACK
;
1645 case TPG_PAT_ALTERNATING_VLINES
:
1646 return (x
& 1) ? TPG_COLOR_100_WHITE
: TPG_COLOR_100_BLACK
;
1647 case TPG_PAT_CROSS_1_PIXEL
:
1648 if (pat_line
|| (x
% tpg
->src_width
) == tpg
->src_width
/ 2)
1649 return TPG_COLOR_100_BLACK
;
1650 return TPG_COLOR_100_WHITE
;
1651 case TPG_PAT_CROSS_2_PIXELS
:
1652 if (pat_line
|| ((x
% tpg
->src_width
) + 1) / 2 == tpg
->src_width
/ 4)
1653 return TPG_COLOR_100_BLACK
;
1654 return TPG_COLOR_100_WHITE
;
1655 case TPG_PAT_CROSS_10_PIXELS
:
1656 if (pat_line
|| ((x
% tpg
->src_width
) + 10) / 20 == tpg
->src_width
/ 40)
1657 return TPG_COLOR_100_BLACK
;
1658 return TPG_COLOR_100_WHITE
;
1659 case TPG_PAT_GRAY_RAMP
:
1660 return TPG_COLOR_RAMP
+ ((x
% tpg
->src_width
) * 256) / tpg
->src_width
;
1662 return TPG_COLOR_100_RED
;
1667 * Given the pixel aspect ratio and video aspect ratio calculate the
1668 * coordinates of a centered square and the coordinates of the border of
1669 * the active video area. The coordinates are relative to the source
1672 static void tpg_calculate_square_border(struct tpg_data
*tpg
)
1674 unsigned w
= tpg
->src_width
;
1675 unsigned h
= tpg
->src_height
;
1676 unsigned sq_w
, sq_h
;
1678 sq_w
= (w
* 2 / 5) & ~1;
1679 if (((w
- sq_w
) / 2) & 1)
1682 tpg
->square
.width
= sq_w
;
1683 if (tpg
->vid_aspect
== TPG_VIDEO_ASPECT_16X9_ANAMORPHIC
) {
1684 unsigned ana_sq_w
= (sq_w
/ 4) * 3;
1686 if (((w
- ana_sq_w
) / 2) & 1)
1688 tpg
->square
.width
= ana_sq_w
;
1690 tpg
->square
.left
= (w
- tpg
->square
.width
) / 2;
1691 if (tpg
->pix_aspect
== TPG_PIXEL_ASPECT_NTSC
)
1692 sq_h
= sq_w
* 10 / 11;
1693 else if (tpg
->pix_aspect
== TPG_PIXEL_ASPECT_PAL
)
1694 sq_h
= sq_w
* 59 / 54;
1695 tpg
->square
.height
= sq_h
;
1696 tpg
->square
.top
= (h
- sq_h
) / 2;
1697 tpg
->border
.left
= 0;
1698 tpg
->border
.width
= w
;
1699 tpg
->border
.top
= 0;
1700 tpg
->border
.height
= h
;
1701 switch (tpg
->vid_aspect
) {
1702 case TPG_VIDEO_ASPECT_4X3
:
1703 if (tpg
->pix_aspect
)
1705 if (3 * w
>= 4 * h
) {
1706 tpg
->border
.width
= ((4 * h
) / 3) & ~1;
1707 if (((w
- tpg
->border
.width
) / 2) & ~1)
1708 tpg
->border
.width
-= 2;
1709 tpg
->border
.left
= (w
- tpg
->border
.width
) / 2;
1712 tpg
->border
.height
= ((3 * w
) / 4) & ~1;
1713 tpg
->border
.top
= (h
- tpg
->border
.height
) / 2;
1715 case TPG_VIDEO_ASPECT_14X9_CENTRE
:
1716 if (tpg
->pix_aspect
) {
1717 tpg
->border
.height
= tpg
->pix_aspect
== TPG_PIXEL_ASPECT_NTSC
? 420 : 506;
1718 tpg
->border
.top
= (h
- tpg
->border
.height
) / 2;
1721 if (9 * w
>= 14 * h
) {
1722 tpg
->border
.width
= ((14 * h
) / 9) & ~1;
1723 if (((w
- tpg
->border
.width
) / 2) & ~1)
1724 tpg
->border
.width
-= 2;
1725 tpg
->border
.left
= (w
- tpg
->border
.width
) / 2;
1728 tpg
->border
.height
= ((9 * w
) / 14) & ~1;
1729 tpg
->border
.top
= (h
- tpg
->border
.height
) / 2;
1731 case TPG_VIDEO_ASPECT_16X9_CENTRE
:
1732 if (tpg
->pix_aspect
) {
1733 tpg
->border
.height
= tpg
->pix_aspect
== TPG_PIXEL_ASPECT_NTSC
? 368 : 442;
1734 tpg
->border
.top
= (h
- tpg
->border
.height
) / 2;
1737 if (9 * w
>= 16 * h
) {
1738 tpg
->border
.width
= ((16 * h
) / 9) & ~1;
1739 if (((w
- tpg
->border
.width
) / 2) & ~1)
1740 tpg
->border
.width
-= 2;
1741 tpg
->border
.left
= (w
- tpg
->border
.width
) / 2;
1744 tpg
->border
.height
= ((9 * w
) / 16) & ~1;
1745 tpg
->border
.top
= (h
- tpg
->border
.height
) / 2;
1752 static void tpg_precalculate_line(struct tpg_data
*tpg
)
1754 enum tpg_color contrast
;
1755 u8 pix
[TPG_MAX_PLANES
][8];
1760 switch (tpg
->pattern
) {
1762 contrast
= TPG_COLOR_100_RED
;
1764 case TPG_PAT_CSC_COLORBAR
:
1765 contrast
= TPG_COLOR_CSC_GREEN
;
1768 contrast
= TPG_COLOR_100_GREEN
;
1772 for (pat
= 0; pat
< tpg_get_pat_lines(tpg
); pat
++) {
1773 /* Coarse scaling with Bresenham */
1774 unsigned int_part
= tpg
->src_width
/ tpg
->scaled_width
;
1775 unsigned fract_part
= tpg
->src_width
% tpg
->scaled_width
;
1779 for (x
= 0; x
< tpg
->scaled_width
* 2; x
+= 2) {
1780 unsigned real_x
= src_x
;
1781 enum tpg_color color1
, color2
;
1783 real_x
= tpg
->hflip
? tpg
->src_width
* 2 - real_x
- 2 : real_x
;
1784 color1
= tpg_get_color(tpg
, pat
, real_x
);
1787 error
+= fract_part
;
1788 if (error
>= tpg
->scaled_width
) {
1789 error
-= tpg
->scaled_width
;
1794 real_x
= tpg
->hflip
? tpg
->src_width
* 2 - real_x
- 2 : real_x
;
1795 color2
= tpg_get_color(tpg
, pat
, real_x
);
1798 error
+= fract_part
;
1799 if (error
>= tpg
->scaled_width
) {
1800 error
-= tpg
->scaled_width
;
1804 gen_twopix(tpg
, pix
, tpg
->hflip
? color2
: color1
, 0);
1805 gen_twopix(tpg
, pix
, tpg
->hflip
? color1
: color2
, 1);
1806 for (p
= 0; p
< tpg
->planes
; p
++) {
1807 unsigned twopixsize
= tpg
->twopixelsize
[p
];
1808 unsigned hdiv
= tpg
->hdownsampling
[p
];
1809 u8
*pos
= tpg
->lines
[pat
][p
] + tpg_hdiv(tpg
, p
, x
);
1811 memcpy(pos
, pix
[p
], twopixsize
/ hdiv
);
1816 if (tpg
->vdownsampling
[tpg
->planes
- 1] > 1) {
1817 unsigned pat_lines
= tpg_get_pat_lines(tpg
);
1819 for (pat
= 0; pat
< pat_lines
; pat
++) {
1820 unsigned next_pat
= (pat
+ 1) % pat_lines
;
1822 for (p
= 1; p
< tpg
->planes
; p
++) {
1823 unsigned w
= tpg_hdiv(tpg
, p
, tpg
->scaled_width
* 2);
1824 u8
*pos1
= tpg
->lines
[pat
][p
];
1825 u8
*pos2
= tpg
->lines
[next_pat
][p
];
1826 u8
*dest
= tpg
->downsampled_lines
[pat
][p
];
1828 for (x
= 0; x
< w
; x
++, pos1
++, pos2
++, dest
++)
1829 *dest
= ((u16
)*pos1
+ (u16
)*pos2
) / 2;
1834 gen_twopix(tpg
, pix
, contrast
, 0);
1835 gen_twopix(tpg
, pix
, contrast
, 1);
1836 for (p
= 0; p
< tpg
->planes
; p
++) {
1837 unsigned twopixsize
= tpg
->twopixelsize
[p
];
1838 u8
*pos
= tpg
->contrast_line
[p
];
1840 for (x
= 0; x
< tpg
->scaled_width
; x
+= 2, pos
+= twopixsize
)
1841 memcpy(pos
, pix
[p
], twopixsize
);
1844 gen_twopix(tpg
, pix
, TPG_COLOR_100_BLACK
, 0);
1845 gen_twopix(tpg
, pix
, TPG_COLOR_100_BLACK
, 1);
1846 for (p
= 0; p
< tpg
->planes
; p
++) {
1847 unsigned twopixsize
= tpg
->twopixelsize
[p
];
1848 u8
*pos
= tpg
->black_line
[p
];
1850 for (x
= 0; x
< tpg
->scaled_width
; x
+= 2, pos
+= twopixsize
)
1851 memcpy(pos
, pix
[p
], twopixsize
);
1854 for (x
= 0; x
< tpg
->scaled_width
* 2; x
+= 2) {
1855 gen_twopix(tpg
, pix
, TPG_COLOR_RANDOM
, 0);
1856 gen_twopix(tpg
, pix
, TPG_COLOR_RANDOM
, 1);
1857 for (p
= 0; p
< tpg
->planes
; p
++) {
1858 unsigned twopixsize
= tpg
->twopixelsize
[p
];
1859 u8
*pos
= tpg
->random_line
[p
] + x
* twopixsize
/ 2;
1861 memcpy(pos
, pix
[p
], twopixsize
);
1865 gen_twopix(tpg
, tpg
->textbg
, TPG_COLOR_TEXTBG
, 0);
1866 gen_twopix(tpg
, tpg
->textbg
, TPG_COLOR_TEXTBG
, 1);
1867 gen_twopix(tpg
, tpg
->textfg
, TPG_COLOR_TEXTFG
, 0);
1868 gen_twopix(tpg
, tpg
->textfg
, TPG_COLOR_TEXTFG
, 1);
1871 /* need this to do rgb24 rendering */
1872 typedef struct { u16 __
; u8 _
; } __packed x24
;
1874 #define PRINTSTR(PIXTYPE) do { \
1875 unsigned vdiv = tpg->vdownsampling[p]; \
1876 unsigned hdiv = tpg->hdownsampling[p]; \
1880 memcpy(&fg, tpg->textfg[p], sizeof(PIXTYPE)); \
1881 memcpy(&bg, tpg->textbg[p], sizeof(PIXTYPE)); \
1883 for (line = first; line < 16; line += vdiv * step) { \
1884 int l = tpg->vflip ? 15 - line : line; \
1885 PIXTYPE *pos = (PIXTYPE *)(basep[p][(line / vdiv) & 1] + \
1886 ((y * step + l) / (vdiv * div)) * tpg->bytesperline[p] + \
1887 (x / hdiv) * sizeof(PIXTYPE)); \
1890 for (s = 0; s < len; s++) { \
1891 u8 chr = font8x16[(u8)text[s] * 16 + line]; \
1893 if (hdiv == 2 && tpg->hflip) { \
1894 pos[3] = (chr & (0x01 << 6) ? fg : bg); \
1895 pos[2] = (chr & (0x01 << 4) ? fg : bg); \
1896 pos[1] = (chr & (0x01 << 2) ? fg : bg); \
1897 pos[0] = (chr & (0x01 << 0) ? fg : bg); \
1898 } else if (hdiv == 2) { \
1899 pos[0] = (chr & (0x01 << 7) ? fg : bg); \
1900 pos[1] = (chr & (0x01 << 5) ? fg : bg); \
1901 pos[2] = (chr & (0x01 << 3) ? fg : bg); \
1902 pos[3] = (chr & (0x01 << 1) ? fg : bg); \
1903 } else if (tpg->hflip) { \
1904 pos[7] = (chr & (0x01 << 7) ? fg : bg); \
1905 pos[6] = (chr & (0x01 << 6) ? fg : bg); \
1906 pos[5] = (chr & (0x01 << 5) ? fg : bg); \
1907 pos[4] = (chr & (0x01 << 4) ? fg : bg); \
1908 pos[3] = (chr & (0x01 << 3) ? fg : bg); \
1909 pos[2] = (chr & (0x01 << 2) ? fg : bg); \
1910 pos[1] = (chr & (0x01 << 1) ? fg : bg); \
1911 pos[0] = (chr & (0x01 << 0) ? fg : bg); \
1913 pos[0] = (chr & (0x01 << 7) ? fg : bg); \
1914 pos[1] = (chr & (0x01 << 6) ? fg : bg); \
1915 pos[2] = (chr & (0x01 << 5) ? fg : bg); \
1916 pos[3] = (chr & (0x01 << 4) ? fg : bg); \
1917 pos[4] = (chr & (0x01 << 3) ? fg : bg); \
1918 pos[5] = (chr & (0x01 << 2) ? fg : bg); \
1919 pos[6] = (chr & (0x01 << 1) ? fg : bg); \
1920 pos[7] = (chr & (0x01 << 0) ? fg : bg); \
1923 pos += (tpg->hflip ? -8 : 8) / (int)hdiv; \
1928 static noinline
void tpg_print_str_2(const struct tpg_data
*tpg
, u8
*basep
[TPG_MAX_PLANES
][2],
1929 unsigned p
, unsigned first
, unsigned div
, unsigned step
,
1930 int y
, int x
, const char *text
, unsigned len
)
1935 static noinline
void tpg_print_str_4(const struct tpg_data
*tpg
, u8
*basep
[TPG_MAX_PLANES
][2],
1936 unsigned p
, unsigned first
, unsigned div
, unsigned step
,
1937 int y
, int x
, const char *text
, unsigned len
)
1942 static noinline
void tpg_print_str_6(const struct tpg_data
*tpg
, u8
*basep
[TPG_MAX_PLANES
][2],
1943 unsigned p
, unsigned first
, unsigned div
, unsigned step
,
1944 int y
, int x
, const char *text
, unsigned len
)
1949 static noinline
void tpg_print_str_8(const struct tpg_data
*tpg
, u8
*basep
[TPG_MAX_PLANES
][2],
1950 unsigned p
, unsigned first
, unsigned div
, unsigned step
,
1951 int y
, int x
, const char *text
, unsigned len
)
1956 void tpg_gen_text(const struct tpg_data
*tpg
, u8
*basep
[TPG_MAX_PLANES
][2],
1957 int y
, int x
, const char *text
)
1959 unsigned step
= V4L2_FIELD_HAS_T_OR_B(tpg
->field
) ? 2 : 1;
1960 unsigned div
= step
;
1965 if (font8x16
== NULL
|| basep
== NULL
|| text
== NULL
)
1970 /* Checks if it is possible to show string */
1971 if (y
+ 16 >= tpg
->compose
.height
|| x
+ 8 >= tpg
->compose
.width
)
1974 if (len
> (tpg
->compose
.width
- x
) / 8)
1975 len
= (tpg
->compose
.width
- x
) / 8;
1977 y
= tpg
->compose
.height
- y
- 16;
1979 x
= tpg
->compose
.width
- x
- 8;
1980 y
+= tpg
->compose
.top
;
1981 x
+= tpg
->compose
.left
;
1982 if (tpg
->field
== V4L2_FIELD_BOTTOM
)
1984 else if (tpg
->field
== V4L2_FIELD_SEQ_TB
|| tpg
->field
== V4L2_FIELD_SEQ_BT
)
1987 for (p
= 0; p
< tpg
->planes
; p
++) {
1989 switch (tpg
->twopixelsize
[p
]) {
1991 tpg_print_str_2(tpg
, basep
, p
, first
, div
, step
, y
, x
,
1995 tpg_print_str_4(tpg
, basep
, p
, first
, div
, step
, y
, x
,
1999 tpg_print_str_6(tpg
, basep
, p
, first
, div
, step
, y
, x
,
2003 tpg_print_str_8(tpg
, basep
, p
, first
, div
, step
, y
, x
,
2009 EXPORT_SYMBOL_GPL(tpg_gen_text
);
2011 const char *tpg_g_color_order(const struct tpg_data
*tpg
)
2013 switch (tpg
->pattern
) {
2014 case TPG_PAT_75_COLORBAR
:
2015 case TPG_PAT_100_COLORBAR
:
2016 case TPG_PAT_CSC_COLORBAR
:
2017 case TPG_PAT_100_HCOLORBAR
:
2018 return "White, yellow, cyan, green, magenta, red, blue, black";
2033 EXPORT_SYMBOL_GPL(tpg_g_color_order
);
2035 void tpg_update_mv_step(struct tpg_data
*tpg
)
2037 int factor
= tpg
->mv_hor_mode
> TPG_MOVE_NONE
? -1 : 1;
2041 switch (tpg
->mv_hor_mode
) {
2042 case TPG_MOVE_NEG_FAST
:
2043 case TPG_MOVE_POS_FAST
:
2044 tpg
->mv_hor_step
= ((tpg
->src_width
+ 319) / 320) * 4;
2048 tpg
->mv_hor_step
= ((tpg
->src_width
+ 639) / 640) * 4;
2050 case TPG_MOVE_NEG_SLOW
:
2051 case TPG_MOVE_POS_SLOW
:
2052 tpg
->mv_hor_step
= 2;
2055 tpg
->mv_hor_step
= 0;
2059 tpg
->mv_hor_step
= tpg
->src_width
- tpg
->mv_hor_step
;
2061 factor
= tpg
->mv_vert_mode
> TPG_MOVE_NONE
? -1 : 1;
2062 switch (tpg
->mv_vert_mode
) {
2063 case TPG_MOVE_NEG_FAST
:
2064 case TPG_MOVE_POS_FAST
:
2065 tpg
->mv_vert_step
= ((tpg
->src_width
+ 319) / 320) * 4;
2069 tpg
->mv_vert_step
= ((tpg
->src_width
+ 639) / 640) * 4;
2071 case TPG_MOVE_NEG_SLOW
:
2072 case TPG_MOVE_POS_SLOW
:
2073 tpg
->mv_vert_step
= 1;
2076 tpg
->mv_vert_step
= 0;
2080 tpg
->mv_vert_step
= tpg
->src_height
- tpg
->mv_vert_step
;
2082 EXPORT_SYMBOL_GPL(tpg_update_mv_step
);
2084 /* Map the line number relative to the crop rectangle to a frame line number */
2085 static unsigned tpg_calc_frameline(const struct tpg_data
*tpg
, unsigned src_y
,
2089 case V4L2_FIELD_TOP
:
2090 return tpg
->crop
.top
+ src_y
* 2;
2091 case V4L2_FIELD_BOTTOM
:
2092 return tpg
->crop
.top
+ src_y
* 2 + 1;
2094 return src_y
+ tpg
->crop
.top
;
2099 * Map the line number relative to the compose rectangle to a destination
2100 * buffer line number.
2102 static unsigned tpg_calc_buffer_line(const struct tpg_data
*tpg
, unsigned y
,
2105 y
+= tpg
->compose
.top
;
2107 case V4L2_FIELD_SEQ_TB
:
2109 return tpg
->buf_height
/ 2 + y
/ 2;
2111 case V4L2_FIELD_SEQ_BT
:
2114 return tpg
->buf_height
/ 2 + y
/ 2;
2120 static void tpg_recalc(struct tpg_data
*tpg
)
2122 if (tpg
->recalc_colors
) {
2123 tpg
->recalc_colors
= false;
2124 tpg
->recalc_lines
= true;
2125 tpg
->real_xfer_func
= tpg
->xfer_func
;
2126 tpg
->real_ycbcr_enc
= tpg
->ycbcr_enc
;
2127 tpg
->real_hsv_enc
= tpg
->hsv_enc
;
2128 tpg
->real_quantization
= tpg
->quantization
;
2130 if (tpg
->xfer_func
== V4L2_XFER_FUNC_DEFAULT
)
2131 tpg
->real_xfer_func
=
2132 V4L2_MAP_XFER_FUNC_DEFAULT(tpg
->colorspace
);
2134 if (tpg
->ycbcr_enc
== V4L2_YCBCR_ENC_DEFAULT
)
2135 tpg
->real_ycbcr_enc
=
2136 V4L2_MAP_YCBCR_ENC_DEFAULT(tpg
->colorspace
);
2138 if (tpg
->quantization
== V4L2_QUANTIZATION_DEFAULT
)
2139 tpg
->real_quantization
=
2140 V4L2_MAP_QUANTIZATION_DEFAULT(
2141 tpg
->color_enc
!= TGP_COLOR_ENC_YCBCR
,
2142 tpg
->colorspace
, tpg
->real_ycbcr_enc
);
2144 tpg_precalculate_colors(tpg
);
2146 if (tpg
->recalc_square_border
) {
2147 tpg
->recalc_square_border
= false;
2148 tpg_calculate_square_border(tpg
);
2150 if (tpg
->recalc_lines
) {
2151 tpg
->recalc_lines
= false;
2152 tpg_precalculate_line(tpg
);
2156 void tpg_calc_text_basep(struct tpg_data
*tpg
,
2157 u8
*basep
[TPG_MAX_PLANES
][2], unsigned p
, u8
*vbuf
)
2159 unsigned stride
= tpg
->bytesperline
[p
];
2160 unsigned h
= tpg
->buf_height
;
2166 h
/= tpg
->vdownsampling
[p
];
2167 if (tpg
->field
== V4L2_FIELD_SEQ_TB
)
2168 basep
[p
][1] += h
* stride
/ 2;
2169 else if (tpg
->field
== V4L2_FIELD_SEQ_BT
)
2170 basep
[p
][0] += h
* stride
/ 2;
2171 if (p
== 0 && tpg
->interleaved
)
2172 tpg_calc_text_basep(tpg
, basep
, 1, vbuf
);
2174 EXPORT_SYMBOL_GPL(tpg_calc_text_basep
);
2176 static int tpg_pattern_avg(const struct tpg_data
*tpg
,
2177 unsigned pat1
, unsigned pat2
)
2179 unsigned pat_lines
= tpg_get_pat_lines(tpg
);
2181 if (pat1
== (pat2
+ 1) % pat_lines
)
2183 if (pat2
== (pat1
+ 1) % pat_lines
)
2188 static const char *tpg_color_enc_str(enum tgp_color_enc
2191 switch (color_enc
) {
2192 case TGP_COLOR_ENC_HSV
:
2194 case TGP_COLOR_ENC_YCBCR
:
2196 case TGP_COLOR_ENC_LUMA
:
2198 case TGP_COLOR_ENC_RGB
:
2205 void tpg_log_status(struct tpg_data
*tpg
)
2207 pr_info("tpg source WxH: %ux%u (%s)\n",
2208 tpg
->src_width
, tpg
->src_height
,
2209 tpg_color_enc_str(tpg
->color_enc
));
2210 pr_info("tpg field: %u\n", tpg
->field
);
2211 pr_info("tpg crop: %ux%u@%dx%d\n", tpg
->crop
.width
, tpg
->crop
.height
,
2212 tpg
->crop
.left
, tpg
->crop
.top
);
2213 pr_info("tpg compose: %ux%u@%dx%d\n", tpg
->compose
.width
, tpg
->compose
.height
,
2214 tpg
->compose
.left
, tpg
->compose
.top
);
2215 pr_info("tpg colorspace: %d\n", tpg
->colorspace
);
2216 pr_info("tpg transfer function: %d/%d\n", tpg
->xfer_func
, tpg
->real_xfer_func
);
2217 if (tpg
->color_enc
== TGP_COLOR_ENC_HSV
)
2218 pr_info("tpg HSV encoding: %d/%d\n",
2219 tpg
->hsv_enc
, tpg
->real_hsv_enc
);
2220 else if (tpg
->color_enc
== TGP_COLOR_ENC_YCBCR
)
2221 pr_info("tpg Y'CbCr encoding: %d/%d\n",
2222 tpg
->ycbcr_enc
, tpg
->real_ycbcr_enc
);
2223 pr_info("tpg quantization: %d/%d\n", tpg
->quantization
, tpg
->real_quantization
);
2224 pr_info("tpg RGB range: %d/%d\n", tpg
->rgb_range
, tpg
->real_rgb_range
);
2226 EXPORT_SYMBOL_GPL(tpg_log_status
);
2229 * This struct contains common parameters used by both the drawing of the
2230 * test pattern and the drawing of the extras (borders, square, etc.)
2232 struct tpg_draw_params
{
2236 unsigned twopixsize
;
2240 unsigned frame_line
;
2241 unsigned frame_line_next
;
2244 unsigned mv_hor_old
;
2245 unsigned mv_hor_new
;
2246 unsigned mv_vert_old
;
2247 unsigned mv_vert_new
;
2251 unsigned wss_random_offset
;
2253 unsigned left_pillar_width
;
2254 unsigned right_pillar_start
;
2257 static void tpg_fill_params_pattern(const struct tpg_data
*tpg
, unsigned p
,
2258 struct tpg_draw_params
*params
)
2260 params
->mv_hor_old
=
2261 tpg_hscale_div(tpg
, p
, tpg
->mv_hor_count
% tpg
->src_width
);
2262 params
->mv_hor_new
=
2263 tpg_hscale_div(tpg
, p
, (tpg
->mv_hor_count
+ tpg
->mv_hor_step
) %
2265 params
->mv_vert_old
= tpg
->mv_vert_count
% tpg
->src_height
;
2266 params
->mv_vert_new
=
2267 (tpg
->mv_vert_count
+ tpg
->mv_vert_step
) % tpg
->src_height
;
2270 static void tpg_fill_params_extras(const struct tpg_data
*tpg
,
2272 struct tpg_draw_params
*params
)
2274 unsigned left_pillar_width
= 0;
2275 unsigned right_pillar_start
= params
->img_width
;
2277 params
->wss_width
= tpg
->crop
.left
< tpg
->src_width
/ 2 ?
2278 tpg
->src_width
/ 2 - tpg
->crop
.left
: 0;
2279 if (params
->wss_width
> tpg
->crop
.width
)
2280 params
->wss_width
= tpg
->crop
.width
;
2281 params
->wss_width
= tpg_hscale_div(tpg
, p
, params
->wss_width
);
2282 params
->wss_random_offset
=
2283 params
->twopixsize
* prandom_u32_max(tpg
->src_width
/ 2);
2285 if (tpg
->crop
.left
< tpg
->border
.left
) {
2286 left_pillar_width
= tpg
->border
.left
- tpg
->crop
.left
;
2287 if (left_pillar_width
> tpg
->crop
.width
)
2288 left_pillar_width
= tpg
->crop
.width
;
2289 left_pillar_width
= tpg_hscale_div(tpg
, p
, left_pillar_width
);
2291 params
->left_pillar_width
= left_pillar_width
;
2293 if (tpg
->crop
.left
+ tpg
->crop
.width
>
2294 tpg
->border
.left
+ tpg
->border
.width
) {
2295 right_pillar_start
=
2296 tpg
->border
.left
+ tpg
->border
.width
- tpg
->crop
.left
;
2297 right_pillar_start
=
2298 tpg_hscale_div(tpg
, p
, right_pillar_start
);
2299 if (right_pillar_start
> params
->img_width
)
2300 right_pillar_start
= params
->img_width
;
2302 params
->right_pillar_start
= right_pillar_start
;
2304 params
->sav_eav_f
= tpg
->field
==
2305 (params
->is_60hz
? V4L2_FIELD_TOP
: V4L2_FIELD_BOTTOM
);
2308 static void tpg_fill_plane_extras(const struct tpg_data
*tpg
,
2309 const struct tpg_draw_params
*params
,
2310 unsigned p
, unsigned h
, u8
*vbuf
)
2312 unsigned twopixsize
= params
->twopixsize
;
2313 unsigned img_width
= params
->img_width
;
2314 unsigned frame_line
= params
->frame_line
;
2315 const struct v4l2_rect
*sq
= &tpg
->square
;
2316 const struct v4l2_rect
*b
= &tpg
->border
;
2317 const struct v4l2_rect
*c
= &tpg
->crop
;
2319 if (params
->is_tv
&& !params
->is_60hz
&&
2320 frame_line
== 0 && params
->wss_width
) {
2322 * Replace the first half of the top line of a 50 Hz frame
2323 * with random data to simulate a WSS signal.
2325 u8
*wss
= tpg
->random_line
[p
] + params
->wss_random_offset
;
2327 memcpy(vbuf
, wss
, params
->wss_width
);
2330 if (tpg
->show_border
&& frame_line
>= b
->top
&&
2331 frame_line
< b
->top
+ b
->height
) {
2332 unsigned bottom
= b
->top
+ b
->height
- 1;
2333 unsigned left
= params
->left_pillar_width
;
2334 unsigned right
= params
->right_pillar_start
;
2336 if (frame_line
== b
->top
|| frame_line
== b
->top
+ 1 ||
2337 frame_line
== bottom
|| frame_line
== bottom
- 1) {
2338 memcpy(vbuf
+ left
, tpg
->contrast_line
[p
],
2341 if (b
->left
>= c
->left
&&
2342 b
->left
< c
->left
+ c
->width
)
2344 tpg
->contrast_line
[p
], twopixsize
);
2345 if (b
->left
+ b
->width
> c
->left
&&
2346 b
->left
+ b
->width
<= c
->left
+ c
->width
)
2347 memcpy(vbuf
+ right
- twopixsize
,
2348 tpg
->contrast_line
[p
], twopixsize
);
2351 if (tpg
->qual
!= TPG_QUAL_NOISE
&& frame_line
>= b
->top
&&
2352 frame_line
< b
->top
+ b
->height
) {
2353 memcpy(vbuf
, tpg
->black_line
[p
], params
->left_pillar_width
);
2354 memcpy(vbuf
+ params
->right_pillar_start
, tpg
->black_line
[p
],
2355 img_width
- params
->right_pillar_start
);
2357 if (tpg
->show_square
&& frame_line
>= sq
->top
&&
2358 frame_line
< sq
->top
+ sq
->height
&&
2359 sq
->left
< c
->left
+ c
->width
&&
2360 sq
->left
+ sq
->width
>= c
->left
) {
2361 unsigned left
= sq
->left
;
2362 unsigned width
= sq
->width
;
2364 if (c
->left
> left
) {
2365 width
-= c
->left
- left
;
2368 if (c
->left
+ c
->width
< left
+ width
)
2369 width
-= left
+ width
- c
->left
- c
->width
;
2371 left
= tpg_hscale_div(tpg
, p
, left
);
2372 width
= tpg_hscale_div(tpg
, p
, width
);
2373 memcpy(vbuf
+ left
, tpg
->contrast_line
[p
], width
);
2375 if (tpg
->insert_sav
) {
2376 unsigned offset
= tpg_hdiv(tpg
, p
, tpg
->compose
.width
/ 3);
2377 u8
*p
= vbuf
+ offset
;
2378 unsigned vact
= 0, hact
= 0;
2383 p
[3] = 0x80 | (params
->sav_eav_f
<< 6) |
2384 (vact
<< 5) | (hact
<< 4) |
2385 ((hact
^ vact
) << 3) |
2386 ((hact
^ params
->sav_eav_f
) << 2) |
2387 ((params
->sav_eav_f
^ vact
) << 1) |
2388 (hact
^ vact
^ params
->sav_eav_f
);
2390 if (tpg
->insert_eav
) {
2391 unsigned offset
= tpg_hdiv(tpg
, p
, tpg
->compose
.width
* 2 / 3);
2392 u8
*p
= vbuf
+ offset
;
2393 unsigned vact
= 0, hact
= 1;
2398 p
[3] = 0x80 | (params
->sav_eav_f
<< 6) |
2399 (vact
<< 5) | (hact
<< 4) |
2400 ((hact
^ vact
) << 3) |
2401 ((hact
^ params
->sav_eav_f
) << 2) |
2402 ((params
->sav_eav_f
^ vact
) << 1) |
2403 (hact
^ vact
^ params
->sav_eav_f
);
2407 static void tpg_fill_plane_pattern(const struct tpg_data
*tpg
,
2408 const struct tpg_draw_params
*params
,
2409 unsigned p
, unsigned h
, u8
*vbuf
)
2411 unsigned twopixsize
= params
->twopixsize
;
2412 unsigned img_width
= params
->img_width
;
2413 unsigned mv_hor_old
= params
->mv_hor_old
;
2414 unsigned mv_hor_new
= params
->mv_hor_new
;
2415 unsigned mv_vert_old
= params
->mv_vert_old
;
2416 unsigned mv_vert_new
= params
->mv_vert_new
;
2417 unsigned frame_line
= params
->frame_line
;
2418 unsigned frame_line_next
= params
->frame_line_next
;
2419 unsigned line_offset
= tpg_hscale_div(tpg
, p
, tpg
->crop
.left
);
2421 bool fill_blank
= false;
2422 unsigned pat_line_old
;
2423 unsigned pat_line_new
;
2424 u8
*linestart_older
;
2425 u8
*linestart_newer
;
2427 u8
*linestart_bottom
;
2429 even
= !(frame_line
& 1);
2431 if (h
>= params
->hmax
) {
2432 if (params
->hmax
== tpg
->compose
.height
)
2434 if (!tpg
->perc_fill_blank
)
2440 frame_line
= tpg
->src_height
- frame_line
- 1;
2441 frame_line_next
= tpg
->src_height
- frame_line_next
- 1;
2445 linestart_older
= tpg
->contrast_line
[p
];
2446 linestart_newer
= tpg
->contrast_line
[p
];
2447 } else if (tpg
->qual
!= TPG_QUAL_NOISE
&&
2448 (frame_line
< tpg
->border
.top
||
2449 frame_line
>= tpg
->border
.top
+ tpg
->border
.height
)) {
2450 linestart_older
= tpg
->black_line
[p
];
2451 linestart_newer
= tpg
->black_line
[p
];
2452 } else if (tpg
->pattern
== TPG_PAT_NOISE
|| tpg
->qual
== TPG_QUAL_NOISE
) {
2453 linestart_older
= tpg
->random_line
[p
] +
2454 twopixsize
* prandom_u32_max(tpg
->src_width
/ 2);
2455 linestart_newer
= tpg
->random_line
[p
] +
2456 twopixsize
* prandom_u32_max(tpg
->src_width
/ 2);
2458 unsigned frame_line_old
=
2459 (frame_line
+ mv_vert_old
) % tpg
->src_height
;
2460 unsigned frame_line_new
=
2461 (frame_line
+ mv_vert_new
) % tpg
->src_height
;
2462 unsigned pat_line_next_old
;
2463 unsigned pat_line_next_new
;
2465 pat_line_old
= tpg_get_pat_line(tpg
, frame_line_old
);
2466 pat_line_new
= tpg_get_pat_line(tpg
, frame_line_new
);
2467 linestart_older
= tpg
->lines
[pat_line_old
][p
] + mv_hor_old
;
2468 linestart_newer
= tpg
->lines
[pat_line_new
][p
] + mv_hor_new
;
2470 if (tpg
->vdownsampling
[p
] > 1 && frame_line
!= frame_line_next
) {
2474 * Now decide whether we need to use downsampled_lines[].
2475 * That's necessary if the two lines use different patterns.
2477 pat_line_next_old
= tpg_get_pat_line(tpg
,
2478 (frame_line_next
+ mv_vert_old
) % tpg
->src_height
);
2479 pat_line_next_new
= tpg_get_pat_line(tpg
,
2480 (frame_line_next
+ mv_vert_new
) % tpg
->src_height
);
2482 switch (tpg
->field
) {
2483 case V4L2_FIELD_INTERLACED
:
2484 case V4L2_FIELD_INTERLACED_BT
:
2485 case V4L2_FIELD_INTERLACED_TB
:
2486 avg_pat
= tpg_pattern_avg(tpg
, pat_line_old
, pat_line_new
);
2489 linestart_older
= tpg
->downsampled_lines
[avg_pat
][p
] + mv_hor_old
;
2490 linestart_newer
= linestart_older
;
2492 case V4L2_FIELD_NONE
:
2493 case V4L2_FIELD_TOP
:
2494 case V4L2_FIELD_BOTTOM
:
2495 case V4L2_FIELD_SEQ_BT
:
2496 case V4L2_FIELD_SEQ_TB
:
2497 avg_pat
= tpg_pattern_avg(tpg
, pat_line_old
, pat_line_next_old
);
2499 linestart_older
= tpg
->downsampled_lines
[avg_pat
][p
] +
2501 avg_pat
= tpg_pattern_avg(tpg
, pat_line_new
, pat_line_next_new
);
2503 linestart_newer
= tpg
->downsampled_lines
[avg_pat
][p
] +
2508 linestart_older
+= line_offset
;
2509 linestart_newer
+= line_offset
;
2511 if (tpg
->field_alternate
) {
2512 linestart_top
= linestart_bottom
= linestart_older
;
2513 } else if (params
->is_60hz
) {
2514 linestart_top
= linestart_newer
;
2515 linestart_bottom
= linestart_older
;
2517 linestart_top
= linestart_older
;
2518 linestart_bottom
= linestart_newer
;
2521 switch (tpg
->field
) {
2522 case V4L2_FIELD_INTERLACED
:
2523 case V4L2_FIELD_INTERLACED_TB
:
2524 case V4L2_FIELD_SEQ_TB
:
2525 case V4L2_FIELD_SEQ_BT
:
2527 memcpy(vbuf
, linestart_top
, img_width
);
2529 memcpy(vbuf
, linestart_bottom
, img_width
);
2531 case V4L2_FIELD_INTERLACED_BT
:
2533 memcpy(vbuf
, linestart_bottom
, img_width
);
2535 memcpy(vbuf
, linestart_top
, img_width
);
2537 case V4L2_FIELD_TOP
:
2538 memcpy(vbuf
, linestart_top
, img_width
);
2540 case V4L2_FIELD_BOTTOM
:
2541 memcpy(vbuf
, linestart_bottom
, img_width
);
2543 case V4L2_FIELD_NONE
:
2545 memcpy(vbuf
, linestart_older
, img_width
);
2550 void tpg_fill_plane_buffer(struct tpg_data
*tpg
, v4l2_std_id std
,
2551 unsigned p
, u8
*vbuf
)
2553 struct tpg_draw_params params
;
2554 unsigned factor
= V4L2_FIELD_HAS_T_OR_B(tpg
->field
) ? 2 : 1;
2556 /* Coarse scaling with Bresenham */
2557 unsigned int_part
= (tpg
->crop
.height
/ factor
) / tpg
->compose
.height
;
2558 unsigned fract_part
= (tpg
->crop
.height
/ factor
) % tpg
->compose
.height
;
2566 params
.is_60hz
= std
& V4L2_STD_525_60
;
2567 params
.twopixsize
= tpg
->twopixelsize
[p
];
2568 params
.img_width
= tpg_hdiv(tpg
, p
, tpg
->compose
.width
);
2569 params
.stride
= tpg
->bytesperline
[p
];
2570 params
.hmax
= (tpg
->compose
.height
* tpg
->perc_fill
) / 100;
2572 tpg_fill_params_pattern(tpg
, p
, ¶ms
);
2573 tpg_fill_params_extras(tpg
, p
, ¶ms
);
2575 vbuf
+= tpg_hdiv(tpg
, p
, tpg
->compose
.left
);
2577 for (h
= 0; h
< tpg
->compose
.height
; h
++) {
2580 params
.frame_line
= tpg_calc_frameline(tpg
, src_y
, tpg
->field
);
2581 params
.frame_line_next
= params
.frame_line
;
2582 buf_line
= tpg_calc_buffer_line(tpg
, h
, tpg
->field
);
2584 error
+= fract_part
;
2585 if (error
>= tpg
->compose
.height
) {
2586 error
-= tpg
->compose
.height
;
2591 * For line-interleaved formats determine the 'plane'
2592 * based on the buffer line.
2594 if (tpg_g_interleaved(tpg
))
2595 p
= tpg_g_interleaved_plane(tpg
, buf_line
);
2597 if (tpg
->vdownsampling
[p
] > 1) {
2599 * When doing vertical downsampling the field setting
2600 * matters: for SEQ_BT/TB we downsample each field
2601 * separately (i.e. lines 0+2 are combined, as are
2602 * lines 1+3), for the other field settings we combine
2603 * odd and even lines. Doing that for SEQ_BT/TB would
2606 if (tpg
->field
== V4L2_FIELD_SEQ_BT
||
2607 tpg
->field
== V4L2_FIELD_SEQ_TB
) {
2608 unsigned next_src_y
= src_y
;
2612 next_src_y
+= int_part
;
2613 if (error
+ fract_part
>= tpg
->compose
.height
)
2615 params
.frame_line_next
=
2616 tpg_calc_frameline(tpg
, next_src_y
, tpg
->field
);
2620 params
.frame_line_next
=
2621 tpg_calc_frameline(tpg
, src_y
, tpg
->field
);
2624 buf_line
/= tpg
->vdownsampling
[p
];
2626 tpg_fill_plane_pattern(tpg
, ¶ms
, p
, h
,
2627 vbuf
+ buf_line
* params
.stride
);
2628 tpg_fill_plane_extras(tpg
, ¶ms
, p
, h
,
2629 vbuf
+ buf_line
* params
.stride
);
2632 EXPORT_SYMBOL_GPL(tpg_fill_plane_buffer
);
2634 void tpg_fillbuffer(struct tpg_data
*tpg
, v4l2_std_id std
, unsigned p
, u8
*vbuf
)
2636 unsigned offset
= 0;
2639 if (tpg
->buffers
> 1) {
2640 tpg_fill_plane_buffer(tpg
, std
, p
, vbuf
);
2644 for (i
= 0; i
< tpg_g_planes(tpg
); i
++) {
2645 tpg_fill_plane_buffer(tpg
, std
, i
, vbuf
+ offset
);
2646 offset
+= tpg_calc_plane_size(tpg
, i
);
2649 EXPORT_SYMBOL_GPL(tpg_fillbuffer
);
2651 MODULE_DESCRIPTION("V4L2 Test Pattern Generator");
2652 MODULE_AUTHOR("Hans Verkuil");
2653 MODULE_LICENSE("GPL");