1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * v4l2-tpg-core.c - Test Pattern Generator 4 * 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. 7 * 8 * Copyright 2014 Cisco Systems, Inc. and/or its affiliates. All rights reserved. 9 */ 10 11 #include <linux/module.h> 12 #include <media/tpg/v4l2-tpg.h> 13 14 /* Must remain in sync with enum tpg_pattern */ 15 const char * const tpg_pattern_strings[] = { 16 "75% Colorbar", 17 "100% Colorbar", 18 "CSC Colorbar", 19 "Horizontal 100% Colorbar", 20 "100% Color Squares", 21 "100% Black", 22 "100% White", 23 "100% Red", 24 "100% Green", 25 "100% Blue", 26 "16x16 Checkers", 27 "2x2 Checkers", 28 "1x1 Checkers", 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", 36 "Gray Ramp", 37 "Noise", 38 NULL 39 }; 40 EXPORT_SYMBOL_GPL(tpg_pattern_strings); 41 42 /* Must remain in sync with enum tpg_aspect */ 43 const char * const tpg_aspect_strings[] = { 44 "Source Width x Height", 45 "4x3", 46 "14x9", 47 "16x9", 48 "16x9 Anamorphic", 49 NULL 50 }; 51 EXPORT_SYMBOL_GPL(tpg_aspect_strings); 52 53 /* 54 * Sine table: sin[0] = 127 * sin(-180 degrees) 55 * sin[128] = 127 * sin(0 degrees) 56 * sin[256] = 127 * sin(180 degrees) 57 */ 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, 75 0, 76 }; 77 78 #define cos(idx) sin[((idx) + 64) % sizeof(sin)] 79 80 /* Global font descriptor */ 81 static const u8 *font8x16; 82 83 void tpg_set_font(const u8 *f) 84 { 85 font8x16 = f; 86 } 87 EXPORT_SYMBOL_GPL(tpg_set_font); 88 89 void tpg_init(struct tpg_data *tpg, unsigned w, unsigned h) 90 { 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; 99 tpg->contrast = 128; 100 tpg->saturation = 128; 101 tpg->hue = 0; 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; 109 } 110 EXPORT_SYMBOL_GPL(tpg_init); 111 112 int tpg_alloc(struct tpg_data *tpg, unsigned max_w) 113 { 114 unsigned pat; 115 unsigned plane; 116 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; 121 122 tpg->lines[pat][plane] = 123 vzalloc(array3_size(max_w, 2, pixelsz)); 124 if (!tpg->lines[pat][plane]) 125 return -ENOMEM; 126 if (plane == 0) 127 continue; 128 tpg->downsampled_lines[pat][plane] = 129 vzalloc(array3_size(max_w, 2, pixelsz)); 130 if (!tpg->downsampled_lines[pat][plane]) 131 return -ENOMEM; 132 } 133 } 134 for (plane = 0; plane < TPG_MAX_PLANES; plane++) { 135 unsigned pixelsz = plane ? 2 : 4; 136 137 tpg->contrast_line[plane] = 138 vzalloc(array_size(pixelsz, max_w)); 139 if (!tpg->contrast_line[plane]) 140 return -ENOMEM; 141 tpg->black_line[plane] = 142 vzalloc(array_size(pixelsz, max_w)); 143 if (!tpg->black_line[plane]) 144 return -ENOMEM; 145 tpg->random_line[plane] = 146 vzalloc(array3_size(max_w, 2, pixelsz)); 147 if (!tpg->random_line[plane]) 148 return -ENOMEM; 149 } 150 return 0; 151 } 152 EXPORT_SYMBOL_GPL(tpg_alloc); 153 154 void tpg_free(struct tpg_data *tpg) 155 { 156 unsigned pat; 157 unsigned plane; 158 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; 163 if (plane == 0) 164 continue; 165 vfree(tpg->downsampled_lines[pat][plane]); 166 tpg->downsampled_lines[pat][plane] = NULL; 167 } 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; 175 } 176 } 177 EXPORT_SYMBOL_GPL(tpg_free); 178 179 bool tpg_s_fourcc(struct tpg_data *tpg, u32 fourcc) 180 { 181 tpg->fourcc = fourcc; 182 tpg->planes = 1; 183 tpg->buffers = 1; 184 tpg->recalc_colors = true; 185 tpg->interleaved = false; 186 tpg->vdownsampling[0] = 1; 187 tpg->hdownsampling[0] = 1; 188 tpg->hmask[0] = ~0; 189 tpg->hmask[1] = ~0; 190 tpg->hmask[2] = ~0; 191 192 switch (fourcc) { 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; 212 tpg->planes = 2; 213 /* fall through */ 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_RGB555: 221 case V4L2_PIX_FMT_XRGB555: 222 case V4L2_PIX_FMT_ARGB555: 223 case V4L2_PIX_FMT_RGB555X: 224 case V4L2_PIX_FMT_XRGB555X: 225 case V4L2_PIX_FMT_ARGB555X: 226 case V4L2_PIX_FMT_BGR666: 227 case V4L2_PIX_FMT_RGB24: 228 case V4L2_PIX_FMT_BGR24: 229 case V4L2_PIX_FMT_RGB32: 230 case V4L2_PIX_FMT_BGR32: 231 case V4L2_PIX_FMT_XRGB32: 232 case V4L2_PIX_FMT_XBGR32: 233 case V4L2_PIX_FMT_ARGB32: 234 case V4L2_PIX_FMT_ABGR32: 235 tpg->color_enc = TGP_COLOR_ENC_RGB; 236 break; 237 case V4L2_PIX_FMT_GREY: 238 case V4L2_PIX_FMT_Y10: 239 case V4L2_PIX_FMT_Y12: 240 case V4L2_PIX_FMT_Y16: 241 case V4L2_PIX_FMT_Y16_BE: 242 case V4L2_PIX_FMT_Z16: 243 tpg->color_enc = TGP_COLOR_ENC_LUMA; 244 break; 245 case V4L2_PIX_FMT_YUV444: 246 case V4L2_PIX_FMT_YUV555: 247 case V4L2_PIX_FMT_YUV565: 248 case V4L2_PIX_FMT_YUV32: 249 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 250 break; 251 case V4L2_PIX_FMT_YUV420M: 252 case V4L2_PIX_FMT_YVU420M: 253 tpg->buffers = 3; 254 /* fall through */ 255 case V4L2_PIX_FMT_YUV420: 256 case V4L2_PIX_FMT_YVU420: 257 tpg->vdownsampling[1] = 2; 258 tpg->vdownsampling[2] = 2; 259 tpg->hdownsampling[1] = 2; 260 tpg->hdownsampling[2] = 2; 261 tpg->planes = 3; 262 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 263 break; 264 case V4L2_PIX_FMT_YUV422M: 265 case V4L2_PIX_FMT_YVU422M: 266 tpg->buffers = 3; 267 /* fall through */ 268 case V4L2_PIX_FMT_YUV422P: 269 tpg->vdownsampling[1] = 1; 270 tpg->vdownsampling[2] = 1; 271 tpg->hdownsampling[1] = 2; 272 tpg->hdownsampling[2] = 2; 273 tpg->planes = 3; 274 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 275 break; 276 case V4L2_PIX_FMT_NV16M: 277 case V4L2_PIX_FMT_NV61M: 278 tpg->buffers = 2; 279 /* fall through */ 280 case V4L2_PIX_FMT_NV16: 281 case V4L2_PIX_FMT_NV61: 282 tpg->vdownsampling[1] = 1; 283 tpg->hdownsampling[1] = 1; 284 tpg->hmask[1] = ~1; 285 tpg->planes = 2; 286 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 287 break; 288 case V4L2_PIX_FMT_NV12M: 289 case V4L2_PIX_FMT_NV21M: 290 tpg->buffers = 2; 291 /* fall through */ 292 case V4L2_PIX_FMT_NV12: 293 case V4L2_PIX_FMT_NV21: 294 tpg->vdownsampling[1] = 2; 295 tpg->hdownsampling[1] = 1; 296 tpg->hmask[1] = ~1; 297 tpg->planes = 2; 298 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 299 break; 300 case V4L2_PIX_FMT_YUV444M: 301 case V4L2_PIX_FMT_YVU444M: 302 tpg->buffers = 3; 303 tpg->planes = 3; 304 tpg->vdownsampling[1] = 1; 305 tpg->vdownsampling[2] = 1; 306 tpg->hdownsampling[1] = 1; 307 tpg->hdownsampling[2] = 1; 308 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 309 break; 310 case V4L2_PIX_FMT_NV24: 311 case V4L2_PIX_FMT_NV42: 312 tpg->vdownsampling[1] = 1; 313 tpg->hdownsampling[1] = 1; 314 tpg->planes = 2; 315 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 316 break; 317 case V4L2_PIX_FMT_YUYV: 318 case V4L2_PIX_FMT_UYVY: 319 case V4L2_PIX_FMT_YVYU: 320 case V4L2_PIX_FMT_VYUY: 321 tpg->hmask[0] = ~1; 322 tpg->color_enc = TGP_COLOR_ENC_YCBCR; 323 break; 324 case V4L2_PIX_FMT_HSV24: 325 case V4L2_PIX_FMT_HSV32: 326 tpg->color_enc = TGP_COLOR_ENC_HSV; 327 break; 328 default: 329 return false; 330 } 331 332 switch (fourcc) { 333 case V4L2_PIX_FMT_GREY: 334 case V4L2_PIX_FMT_RGB332: 335 tpg->twopixelsize[0] = 2; 336 break; 337 case V4L2_PIX_FMT_RGB565: 338 case V4L2_PIX_FMT_RGB565X: 339 case V4L2_PIX_FMT_RGB444: 340 case V4L2_PIX_FMT_XRGB444: 341 case V4L2_PIX_FMT_ARGB444: 342 case V4L2_PIX_FMT_RGB555: 343 case V4L2_PIX_FMT_XRGB555: 344 case V4L2_PIX_FMT_ARGB555: 345 case V4L2_PIX_FMT_RGB555X: 346 case V4L2_PIX_FMT_XRGB555X: 347 case V4L2_PIX_FMT_ARGB555X: 348 case V4L2_PIX_FMT_YUYV: 349 case V4L2_PIX_FMT_UYVY: 350 case V4L2_PIX_FMT_YVYU: 351 case V4L2_PIX_FMT_VYUY: 352 case V4L2_PIX_FMT_YUV444: 353 case V4L2_PIX_FMT_YUV555: 354 case V4L2_PIX_FMT_YUV565: 355 case V4L2_PIX_FMT_Y10: 356 case V4L2_PIX_FMT_Y12: 357 case V4L2_PIX_FMT_Y16: 358 case V4L2_PIX_FMT_Y16_BE: 359 case V4L2_PIX_FMT_Z16: 360 tpg->twopixelsize[0] = 2 * 2; 361 break; 362 case V4L2_PIX_FMT_RGB24: 363 case V4L2_PIX_FMT_BGR24: 364 case V4L2_PIX_FMT_HSV24: 365 tpg->twopixelsize[0] = 2 * 3; 366 break; 367 case V4L2_PIX_FMT_BGR666: 368 case V4L2_PIX_FMT_RGB32: 369 case V4L2_PIX_FMT_BGR32: 370 case V4L2_PIX_FMT_XRGB32: 371 case V4L2_PIX_FMT_XBGR32: 372 case V4L2_PIX_FMT_ARGB32: 373 case V4L2_PIX_FMT_ABGR32: 374 case V4L2_PIX_FMT_YUV32: 375 case V4L2_PIX_FMT_HSV32: 376 tpg->twopixelsize[0] = 2 * 4; 377 break; 378 case V4L2_PIX_FMT_NV12: 379 case V4L2_PIX_FMT_NV21: 380 case V4L2_PIX_FMT_NV12M: 381 case V4L2_PIX_FMT_NV21M: 382 case V4L2_PIX_FMT_NV16: 383 case V4L2_PIX_FMT_NV61: 384 case V4L2_PIX_FMT_NV16M: 385 case V4L2_PIX_FMT_NV61M: 386 case V4L2_PIX_FMT_SBGGR8: 387 case V4L2_PIX_FMT_SGBRG8: 388 case V4L2_PIX_FMT_SGRBG8: 389 case V4L2_PIX_FMT_SRGGB8: 390 tpg->twopixelsize[0] = 2; 391 tpg->twopixelsize[1] = 2; 392 break; 393 case V4L2_PIX_FMT_SRGGB10: 394 case V4L2_PIX_FMT_SGRBG10: 395 case V4L2_PIX_FMT_SGBRG10: 396 case V4L2_PIX_FMT_SBGGR10: 397 case V4L2_PIX_FMT_SRGGB12: 398 case V4L2_PIX_FMT_SGRBG12: 399 case V4L2_PIX_FMT_SGBRG12: 400 case V4L2_PIX_FMT_SBGGR12: 401 case V4L2_PIX_FMT_SRGGB16: 402 case V4L2_PIX_FMT_SGRBG16: 403 case V4L2_PIX_FMT_SGBRG16: 404 case V4L2_PIX_FMT_SBGGR16: 405 tpg->twopixelsize[0] = 4; 406 tpg->twopixelsize[1] = 4; 407 break; 408 case V4L2_PIX_FMT_YUV444M: 409 case V4L2_PIX_FMT_YVU444M: 410 case V4L2_PIX_FMT_YUV422M: 411 case V4L2_PIX_FMT_YVU422M: 412 case V4L2_PIX_FMT_YUV422P: 413 case V4L2_PIX_FMT_YUV420: 414 case V4L2_PIX_FMT_YVU420: 415 case V4L2_PIX_FMT_YUV420M: 416 case V4L2_PIX_FMT_YVU420M: 417 tpg->twopixelsize[0] = 2; 418 tpg->twopixelsize[1] = 2; 419 tpg->twopixelsize[2] = 2; 420 break; 421 case V4L2_PIX_FMT_NV24: 422 case V4L2_PIX_FMT_NV42: 423 tpg->twopixelsize[0] = 2; 424 tpg->twopixelsize[1] = 4; 425 break; 426 } 427 return true; 428 } 429 EXPORT_SYMBOL_GPL(tpg_s_fourcc); 430 431 void tpg_s_crop_compose(struct tpg_data *tpg, const struct v4l2_rect *crop, 432 const struct v4l2_rect *compose) 433 { 434 tpg->crop = *crop; 435 tpg->compose = *compose; 436 tpg->scaled_width = (tpg->src_width * tpg->compose.width + 437 tpg->crop.width - 1) / tpg->crop.width; 438 tpg->scaled_width &= ~1; 439 if (tpg->scaled_width > tpg->max_line_width) 440 tpg->scaled_width = tpg->max_line_width; 441 if (tpg->scaled_width < 2) 442 tpg->scaled_width = 2; 443 tpg->recalc_lines = true; 444 } 445 EXPORT_SYMBOL_GPL(tpg_s_crop_compose); 446 447 void tpg_reset_source(struct tpg_data *tpg, unsigned width, unsigned height, 448 u32 field) 449 { 450 unsigned p; 451 452 tpg->src_width = width; 453 tpg->src_height = height; 454 tpg->field = field; 455 tpg->buf_height = height; 456 if (V4L2_FIELD_HAS_T_OR_B(field)) 457 tpg->buf_height /= 2; 458 tpg->scaled_width = width; 459 tpg->crop.top = tpg->crop.left = 0; 460 tpg->crop.width = width; 461 tpg->crop.height = height; 462 tpg->compose.top = tpg->compose.left = 0; 463 tpg->compose.width = width; 464 tpg->compose.height = tpg->buf_height; 465 for (p = 0; p < tpg->planes; p++) 466 tpg->bytesperline[p] = (width * tpg->twopixelsize[p]) / 467 (2 * tpg->hdownsampling[p]); 468 tpg->recalc_square_border = true; 469 } 470 EXPORT_SYMBOL_GPL(tpg_reset_source); 471 472 static enum tpg_color tpg_get_textbg_color(struct tpg_data *tpg) 473 { 474 switch (tpg->pattern) { 475 case TPG_PAT_BLACK: 476 return TPG_COLOR_100_WHITE; 477 case TPG_PAT_CSC_COLORBAR: 478 return TPG_COLOR_CSC_BLACK; 479 default: 480 return TPG_COLOR_100_BLACK; 481 } 482 } 483 484 static enum tpg_color tpg_get_textfg_color(struct tpg_data *tpg) 485 { 486 switch (tpg->pattern) { 487 case TPG_PAT_75_COLORBAR: 488 case TPG_PAT_CSC_COLORBAR: 489 return TPG_COLOR_CSC_WHITE; 490 case TPG_PAT_BLACK: 491 return TPG_COLOR_100_BLACK; 492 default: 493 return TPG_COLOR_100_WHITE; 494 } 495 } 496 497 static inline int rec709_to_linear(int v) 498 { 499 v = clamp(v, 0, 0xff0); 500 return tpg_rec709_to_linear[v]; 501 } 502 503 static inline int linear_to_rec709(int v) 504 { 505 v = clamp(v, 0, 0xff0); 506 return tpg_linear_to_rec709[v]; 507 } 508 509 static void color_to_hsv(struct tpg_data *tpg, int r, int g, int b, 510 int *h, int *s, int *v) 511 { 512 int max_rgb, min_rgb, diff_rgb; 513 int aux; 514 int third; 515 int third_size; 516 517 r >>= 4; 518 g >>= 4; 519 b >>= 4; 520 521 /* Value */ 522 max_rgb = max3(r, g, b); 523 *v = max_rgb; 524 if (!max_rgb) { 525 *h = 0; 526 *s = 0; 527 return; 528 } 529 530 /* Saturation */ 531 min_rgb = min3(r, g, b); 532 diff_rgb = max_rgb - min_rgb; 533 aux = 255 * diff_rgb; 534 aux += max_rgb / 2; 535 aux /= max_rgb; 536 *s = aux; 537 if (!aux) { 538 *h = 0; 539 return; 540 } 541 542 third_size = (tpg->real_hsv_enc == V4L2_HSV_ENC_180) ? 60 : 85; 543 544 /* Hue */ 545 if (max_rgb == r) { 546 aux = g - b; 547 third = 0; 548 } else if (max_rgb == g) { 549 aux = b - r; 550 third = third_size; 551 } else { 552 aux = r - g; 553 third = third_size * 2; 554 } 555 556 aux *= third_size / 2; 557 aux += diff_rgb / 2; 558 aux /= diff_rgb; 559 aux += third; 560 561 /* Clamp Hue */ 562 if (tpg->real_hsv_enc == V4L2_HSV_ENC_180) { 563 if (aux < 0) 564 aux += 180; 565 else if (aux > 180) 566 aux -= 180; 567 } else { 568 aux = aux & 0xff; 569 } 570 571 *h = aux; 572 } 573 574 static void rgb2ycbcr(const int m[3][3], int r, int g, int b, 575 int y_offset, int *y, int *cb, int *cr) 576 { 577 *y = ((m[0][0] * r + m[0][1] * g + m[0][2] * b) >> 16) + (y_offset << 4); 578 *cb = ((m[1][0] * r + m[1][1] * g + m[1][2] * b) >> 16) + (128 << 4); 579 *cr = ((m[2][0] * r + m[2][1] * g + m[2][2] * b) >> 16) + (128 << 4); 580 } 581 582 static void color_to_ycbcr(struct tpg_data *tpg, int r, int g, int b, 583 int *y, int *cb, int *cr) 584 { 585 #define COEFF(v, r) ((int)(0.5 + (v) * (r) * 256.0)) 586 587 static const int bt601[3][3] = { 588 { COEFF(0.299, 219), COEFF(0.587, 219), COEFF(0.114, 219) }, 589 { COEFF(-0.1687, 224), COEFF(-0.3313, 224), COEFF(0.5, 224) }, 590 { COEFF(0.5, 224), COEFF(-0.4187, 224), COEFF(-0.0813, 224) }, 591 }; 592 static const int bt601_full[3][3] = { 593 { COEFF(0.299, 255), COEFF(0.587, 255), COEFF(0.114, 255) }, 594 { COEFF(-0.1687, 255), COEFF(-0.3313, 255), COEFF(0.5, 255) }, 595 { COEFF(0.5, 255), COEFF(-0.4187, 255), COEFF(-0.0813, 255) }, 596 }; 597 static const int rec709[3][3] = { 598 { COEFF(0.2126, 219), COEFF(0.7152, 219), COEFF(0.0722, 219) }, 599 { COEFF(-0.1146, 224), COEFF(-0.3854, 224), COEFF(0.5, 224) }, 600 { COEFF(0.5, 224), COEFF(-0.4542, 224), COEFF(-0.0458, 224) }, 601 }; 602 static const int rec709_full[3][3] = { 603 { COEFF(0.2126, 255), COEFF(0.7152, 255), COEFF(0.0722, 255) }, 604 { COEFF(-0.1146, 255), COEFF(-0.3854, 255), COEFF(0.5, 255) }, 605 { COEFF(0.5, 255), COEFF(-0.4542, 255), COEFF(-0.0458, 255) }, 606 }; 607 static const int smpte240m[3][3] = { 608 { COEFF(0.212, 219), COEFF(0.701, 219), COEFF(0.087, 219) }, 609 { COEFF(-0.116, 224), COEFF(-0.384, 224), COEFF(0.5, 224) }, 610 { COEFF(0.5, 224), COEFF(-0.445, 224), COEFF(-0.055, 224) }, 611 }; 612 static const int smpte240m_full[3][3] = { 613 { COEFF(0.212, 255), COEFF(0.701, 255), COEFF(0.087, 255) }, 614 { COEFF(-0.116, 255), COEFF(-0.384, 255), COEFF(0.5, 255) }, 615 { COEFF(0.5, 255), COEFF(-0.445, 255), COEFF(-0.055, 255) }, 616 }; 617 static const int bt2020[3][3] = { 618 { COEFF(0.2627, 219), COEFF(0.6780, 219), COEFF(0.0593, 219) }, 619 { COEFF(-0.1396, 224), COEFF(-0.3604, 224), COEFF(0.5, 224) }, 620 { COEFF(0.5, 224), COEFF(-0.4598, 224), COEFF(-0.0402, 224) }, 621 }; 622 static const int bt2020_full[3][3] = { 623 { COEFF(0.2627, 255), COEFF(0.6780, 255), COEFF(0.0593, 255) }, 624 { COEFF(-0.1396, 255), COEFF(-0.3604, 255), COEFF(0.5, 255) }, 625 { COEFF(0.5, 255), COEFF(-0.4598, 255), COEFF(-0.0402, 255) }, 626 }; 627 static const int bt2020c[4] = { 628 COEFF(1.0 / 1.9404, 224), COEFF(1.0 / 1.5816, 224), 629 COEFF(1.0 / 1.7184, 224), COEFF(1.0 / 0.9936, 224), 630 }; 631 static const int bt2020c_full[4] = { 632 COEFF(1.0 / 1.9404, 255), COEFF(1.0 / 1.5816, 255), 633 COEFF(1.0 / 1.7184, 255), COEFF(1.0 / 0.9936, 255), 634 }; 635 636 bool full = tpg->real_quantization == V4L2_QUANTIZATION_FULL_RANGE; 637 unsigned y_offset = full ? 0 : 16; 638 int lin_y, yc; 639 640 switch (tpg->real_ycbcr_enc) { 641 case V4L2_YCBCR_ENC_601: 642 rgb2ycbcr(full ? bt601_full : bt601, r, g, b, y_offset, y, cb, cr); 643 break; 644 case V4L2_YCBCR_ENC_XV601: 645 /* Ignore quantization range, there is only one possible 646 * Y'CbCr encoding. */ 647 rgb2ycbcr(bt601, r, g, b, 16, y, cb, cr); 648 break; 649 case V4L2_YCBCR_ENC_XV709: 650 /* Ignore quantization range, there is only one possible 651 * Y'CbCr encoding. */ 652 rgb2ycbcr(rec709, r, g, b, 16, y, cb, cr); 653 break; 654 case V4L2_YCBCR_ENC_BT2020: 655 rgb2ycbcr(full ? bt2020_full : bt2020, r, g, b, y_offset, y, cb, cr); 656 break; 657 case V4L2_YCBCR_ENC_BT2020_CONST_LUM: 658 lin_y = (COEFF(0.2627, 255) * rec709_to_linear(r) + 659 COEFF(0.6780, 255) * rec709_to_linear(g) + 660 COEFF(0.0593, 255) * rec709_to_linear(b)) >> 16; 661 yc = linear_to_rec709(lin_y); 662 *y = full ? yc : (yc * 219) / 255 + (16 << 4); 663 if (b <= yc) 664 *cb = (((b - yc) * (full ? bt2020c_full[0] : bt2020c[0])) >> 16) + (128 << 4); 665 else 666 *cb = (((b - yc) * (full ? bt2020c_full[1] : bt2020c[1])) >> 16) + (128 << 4); 667 if (r <= yc) 668 *cr = (((r - yc) * (full ? bt2020c_full[2] : bt2020c[2])) >> 16) + (128 << 4); 669 else 670 *cr = (((r - yc) * (full ? bt2020c_full[3] : bt2020c[3])) >> 16) + (128 << 4); 671 break; 672 case V4L2_YCBCR_ENC_SMPTE240M: 673 rgb2ycbcr(full ? smpte240m_full : smpte240m, r, g, b, y_offset, y, cb, cr); 674 break; 675 case V4L2_YCBCR_ENC_709: 676 default: 677 rgb2ycbcr(full ? rec709_full : rec709, r, g, b, y_offset, y, cb, cr); 678 break; 679 } 680 } 681 682 static void ycbcr2rgb(const int m[3][3], int y, int cb, int cr, 683 int y_offset, int *r, int *g, int *b) 684 { 685 y -= y_offset << 4; 686 cb -= 128 << 4; 687 cr -= 128 << 4; 688 *r = m[0][0] * y + m[0][1] * cb + m[0][2] * cr; 689 *g = m[1][0] * y + m[1][1] * cb + m[1][2] * cr; 690 *b = m[2][0] * y + m[2][1] * cb + m[2][2] * cr; 691 *r = clamp(*r >> 12, 0, 0xff0); 692 *g = clamp(*g >> 12, 0, 0xff0); 693 *b = clamp(*b >> 12, 0, 0xff0); 694 } 695 696 static void ycbcr_to_color(struct tpg_data *tpg, int y, int cb, int cr, 697 int *r, int *g, int *b) 698 { 699 #undef COEFF 700 #define COEFF(v, r) ((int)(0.5 + (v) * ((255.0 * 255.0 * 16.0) / (r)))) 701 static const int bt601[3][3] = { 702 { COEFF(1, 219), COEFF(0, 224), COEFF(1.4020, 224) }, 703 { COEFF(1, 219), COEFF(-0.3441, 224), COEFF(-0.7141, 224) }, 704 { COEFF(1, 219), COEFF(1.7720, 224), COEFF(0, 224) }, 705 }; 706 static const int bt601_full[3][3] = { 707 { COEFF(1, 255), COEFF(0, 255), COEFF(1.4020, 255) }, 708 { COEFF(1, 255), COEFF(-0.3441, 255), COEFF(-0.7141, 255) }, 709 { COEFF(1, 255), COEFF(1.7720, 255), COEFF(0, 255) }, 710 }; 711 static const int rec709[3][3] = { 712 { COEFF(1, 219), COEFF(0, 224), COEFF(1.5748, 224) }, 713 { COEFF(1, 219), COEFF(-0.1873, 224), COEFF(-0.4681, 224) }, 714 { COEFF(1, 219), COEFF(1.8556, 224), COEFF(0, 224) }, 715 }; 716 static const int rec709_full[3][3] = { 717 { COEFF(1, 255), COEFF(0, 255), COEFF(1.5748, 255) }, 718 { COEFF(1, 255), COEFF(-0.1873, 255), COEFF(-0.4681, 255) }, 719 { COEFF(1, 255), COEFF(1.8556, 255), COEFF(0, 255) }, 720 }; 721 static const int smpte240m[3][3] = { 722 { COEFF(1, 219), COEFF(0, 224), COEFF(1.5756, 224) }, 723 { COEFF(1, 219), COEFF(-0.2253, 224), COEFF(-0.4767, 224) }, 724 { COEFF(1, 219), COEFF(1.8270, 224), COEFF(0, 224) }, 725 }; 726 static const int smpte240m_full[3][3] = { 727 { COEFF(1, 255), COEFF(0, 255), COEFF(1.5756, 255) }, 728 { COEFF(1, 255), COEFF(-0.2253, 255), COEFF(-0.4767, 255) }, 729 { COEFF(1, 255), COEFF(1.8270, 255), COEFF(0, 255) }, 730 }; 731 static const int bt2020[3][3] = { 732 { COEFF(1, 219), COEFF(0, 224), COEFF(1.4746, 224) }, 733 { COEFF(1, 219), COEFF(-0.1646, 224), COEFF(-0.5714, 224) }, 734 { COEFF(1, 219), COEFF(1.8814, 224), COEFF(0, 224) }, 735 }; 736 static const int bt2020_full[3][3] = { 737 { COEFF(1, 255), COEFF(0, 255), COEFF(1.4746, 255) }, 738 { COEFF(1, 255), COEFF(-0.1646, 255), COEFF(-0.5714, 255) }, 739 { COEFF(1, 255), COEFF(1.8814, 255), COEFF(0, 255) }, 740 }; 741 static const int bt2020c[4] = { 742 COEFF(1.9404, 224), COEFF(1.5816, 224), 743 COEFF(1.7184, 224), COEFF(0.9936, 224), 744 }; 745 static const int bt2020c_full[4] = { 746 COEFF(1.9404, 255), COEFF(1.5816, 255), 747 COEFF(1.7184, 255), COEFF(0.9936, 255), 748 }; 749 750 bool full = tpg->real_quantization == V4L2_QUANTIZATION_FULL_RANGE; 751 unsigned y_offset = full ? 0 : 16; 752 int y_fac = full ? COEFF(1.0, 255) : COEFF(1.0, 219); 753 int lin_r, lin_g, lin_b, lin_y; 754 755 switch (tpg->real_ycbcr_enc) { 756 case V4L2_YCBCR_ENC_601: 757 ycbcr2rgb(full ? bt601_full : bt601, y, cb, cr, y_offset, r, g, b); 758 break; 759 case V4L2_YCBCR_ENC_XV601: 760 /* Ignore quantization range, there is only one possible 761 * Y'CbCr encoding. */ 762 ycbcr2rgb(bt601, y, cb, cr, 16, r, g, b); 763 break; 764 case V4L2_YCBCR_ENC_XV709: 765 /* Ignore quantization range, there is only one possible 766 * Y'CbCr encoding. */ 767 ycbcr2rgb(rec709, y, cb, cr, 16, r, g, b); 768 break; 769 case V4L2_YCBCR_ENC_BT2020: 770 ycbcr2rgb(full ? bt2020_full : bt2020, y, cb, cr, y_offset, r, g, b); 771 break; 772 case V4L2_YCBCR_ENC_BT2020_CONST_LUM: 773 y -= full ? 0 : 16 << 4; 774 cb -= 128 << 4; 775 cr -= 128 << 4; 776 777 if (cb <= 0) 778 *b = y_fac * y + (full ? bt2020c_full[0] : bt2020c[0]) * cb; 779 else 780 *b = y_fac * y + (full ? bt2020c_full[1] : bt2020c[1]) * cb; 781 *b = *b >> 12; 782 if (cr <= 0) 783 *r = y_fac * y + (full ? bt2020c_full[2] : bt2020c[2]) * cr; 784 else 785 *r = y_fac * y + (full ? bt2020c_full[3] : bt2020c[3]) * cr; 786 *r = *r >> 12; 787 lin_r = rec709_to_linear(*r); 788 lin_b = rec709_to_linear(*b); 789 lin_y = rec709_to_linear((y * 255) / (full ? 255 : 219)); 790 791 lin_g = COEFF(1.0 / 0.6780, 255) * lin_y - 792 COEFF(0.2627 / 0.6780, 255) * lin_r - 793 COEFF(0.0593 / 0.6780, 255) * lin_b; 794 *g = linear_to_rec709(lin_g >> 12); 795 break; 796 case V4L2_YCBCR_ENC_SMPTE240M: 797 ycbcr2rgb(full ? smpte240m_full : smpte240m, y, cb, cr, y_offset, r, g, b); 798 break; 799 case V4L2_YCBCR_ENC_709: 800 default: 801 ycbcr2rgb(full ? rec709_full : rec709, y, cb, cr, y_offset, r, g, b); 802 break; 803 } 804 } 805 806 /* precalculate color bar values to speed up rendering */ 807 static void precalculate_color(struct tpg_data *tpg, int k) 808 { 809 int col = k; 810 int r = tpg_colors[col].r; 811 int g = tpg_colors[col].g; 812 int b = tpg_colors[col].b; 813 int y, cb, cr; 814 bool ycbcr_valid = false; 815 816 if (k == TPG_COLOR_TEXTBG) { 817 col = tpg_get_textbg_color(tpg); 818 819 r = tpg_colors[col].r; 820 g = tpg_colors[col].g; 821 b = tpg_colors[col].b; 822 } else if (k == TPG_COLOR_TEXTFG) { 823 col = tpg_get_textfg_color(tpg); 824 825 r = tpg_colors[col].r; 826 g = tpg_colors[col].g; 827 b = tpg_colors[col].b; 828 } else if (tpg->pattern == TPG_PAT_NOISE) { 829 r = g = b = prandom_u32_max(256); 830 } else if (k == TPG_COLOR_RANDOM) { 831 r = g = b = tpg->qual_offset + prandom_u32_max(196); 832 } else if (k >= TPG_COLOR_RAMP) { 833 r = g = b = k - TPG_COLOR_RAMP; 834 } 835 836 if (tpg->pattern == TPG_PAT_CSC_COLORBAR && col <= TPG_COLOR_CSC_BLACK) { 837 r = tpg_csc_colors[tpg->colorspace][tpg->real_xfer_func][col].r; 838 g = tpg_csc_colors[tpg->colorspace][tpg->real_xfer_func][col].g; 839 b = tpg_csc_colors[tpg->colorspace][tpg->real_xfer_func][col].b; 840 } else { 841 r <<= 4; 842 g <<= 4; 843 b <<= 4; 844 } 845 846 if (tpg->qual == TPG_QUAL_GRAY || 847 tpg->color_enc == TGP_COLOR_ENC_LUMA) { 848 /* Rec. 709 Luma function */ 849 /* (0.2126, 0.7152, 0.0722) * (255 * 256) */ 850 r = g = b = (13879 * r + 46688 * g + 4713 * b) >> 16; 851 } 852 853 /* 854 * The assumption is that the RGB output is always full range, 855 * so only if the rgb_range overrides the 'real' rgb range do 856 * we need to convert the RGB values. 857 * 858 * Remember that r, g and b are still in the 0 - 0xff0 range. 859 */ 860 if (tpg->real_rgb_range == V4L2_DV_RGB_RANGE_LIMITED && 861 tpg->rgb_range == V4L2_DV_RGB_RANGE_FULL && 862 tpg->color_enc == TGP_COLOR_ENC_RGB) { 863 /* 864 * Convert from full range (which is what r, g and b are) 865 * to limited range (which is the 'real' RGB range), which 866 * is then interpreted as full range. 867 */ 868 r = (r * 219) / 255 + (16 << 4); 869 g = (g * 219) / 255 + (16 << 4); 870 b = (b * 219) / 255 + (16 << 4); 871 } else if (tpg->real_rgb_range != V4L2_DV_RGB_RANGE_LIMITED && 872 tpg->rgb_range == V4L2_DV_RGB_RANGE_LIMITED && 873 tpg->color_enc == TGP_COLOR_ENC_RGB) { 874 875 /* 876 * Clamp r, g and b to the limited range and convert to full 877 * range since that's what we deliver. 878 */ 879 r = clamp(r, 16 << 4, 235 << 4); 880 g = clamp(g, 16 << 4, 235 << 4); 881 b = clamp(b, 16 << 4, 235 << 4); 882 r = (r - (16 << 4)) * 255 / 219; 883 g = (g - (16 << 4)) * 255 / 219; 884 b = (b - (16 << 4)) * 255 / 219; 885 } 886 887 if ((tpg->brightness != 128 || tpg->contrast != 128 || 888 tpg->saturation != 128 || tpg->hue) && 889 tpg->color_enc != TGP_COLOR_ENC_LUMA) { 890 /* Implement these operations */ 891 int tmp_cb, tmp_cr; 892 893 /* First convert to YCbCr */ 894 895 color_to_ycbcr(tpg, r, g, b, &y, &cb, &cr); 896 897 y = (16 << 4) + ((y - (16 << 4)) * tpg->contrast) / 128; 898 y += (tpg->brightness << 4) - (128 << 4); 899 900 cb -= 128 << 4; 901 cr -= 128 << 4; 902 tmp_cb = (cb * cos(128 + tpg->hue)) / 127 + (cr * sin[128 + tpg->hue]) / 127; 903 tmp_cr = (cr * cos(128 + tpg->hue)) / 127 - (cb * sin[128 + tpg->hue]) / 127; 904 905 cb = (128 << 4) + (tmp_cb * tpg->contrast * tpg->saturation) / (128 * 128); 906 cr = (128 << 4) + (tmp_cr * tpg->contrast * tpg->saturation) / (128 * 128); 907 if (tpg->color_enc == TGP_COLOR_ENC_YCBCR) 908 ycbcr_valid = true; 909 else 910 ycbcr_to_color(tpg, y, cb, cr, &r, &g, &b); 911 } else if ((tpg->brightness != 128 || tpg->contrast != 128) && 912 tpg->color_enc == TGP_COLOR_ENC_LUMA) { 913 r = (16 << 4) + ((r - (16 << 4)) * tpg->contrast) / 128; 914 r += (tpg->brightness << 4) - (128 << 4); 915 } 916 917 switch (tpg->color_enc) { 918 case TGP_COLOR_ENC_HSV: 919 { 920 int h, s, v; 921 922 color_to_hsv(tpg, r, g, b, &h, &s, &v); 923 tpg->colors[k][0] = h; 924 tpg->colors[k][1] = s; 925 tpg->colors[k][2] = v; 926 break; 927 } 928 case TGP_COLOR_ENC_YCBCR: 929 { 930 /* Convert to YCbCr */ 931 if (!ycbcr_valid) 932 color_to_ycbcr(tpg, r, g, b, &y, &cb, &cr); 933 934 y >>= 4; 935 cb >>= 4; 936 cr >>= 4; 937 /* 938 * XV601/709 use the header/footer margins to encode R', G' 939 * and B' values outside the range [0-1]. So do not clamp 940 * XV601/709 values. 941 */ 942 if (tpg->real_quantization == V4L2_QUANTIZATION_LIM_RANGE && 943 tpg->real_ycbcr_enc != V4L2_YCBCR_ENC_XV601 && 944 tpg->real_ycbcr_enc != V4L2_YCBCR_ENC_XV709) { 945 y = clamp(y, 16, 235); 946 cb = clamp(cb, 16, 240); 947 cr = clamp(cr, 16, 240); 948 } else { 949 y = clamp(y, 1, 254); 950 cb = clamp(cb, 1, 254); 951 cr = clamp(cr, 1, 254); 952 } 953 switch (tpg->fourcc) { 954 case V4L2_PIX_FMT_YUV444: 955 y >>= 4; 956 cb >>= 4; 957 cr >>= 4; 958 break; 959 case V4L2_PIX_FMT_YUV555: 960 y >>= 3; 961 cb >>= 3; 962 cr >>= 3; 963 break; 964 case V4L2_PIX_FMT_YUV565: 965 y >>= 3; 966 cb >>= 2; 967 cr >>= 3; 968 break; 969 } 970 tpg->colors[k][0] = y; 971 tpg->colors[k][1] = cb; 972 tpg->colors[k][2] = cr; 973 break; 974 } 975 case TGP_COLOR_ENC_LUMA: 976 { 977 tpg->colors[k][0] = r >> 4; 978 break; 979 } 980 case TGP_COLOR_ENC_RGB: 981 { 982 if (tpg->real_quantization == V4L2_QUANTIZATION_LIM_RANGE) { 983 r = (r * 219) / 255 + (16 << 4); 984 g = (g * 219) / 255 + (16 << 4); 985 b = (b * 219) / 255 + (16 << 4); 986 } 987 switch (tpg->fourcc) { 988 case V4L2_PIX_FMT_RGB332: 989 r >>= 9; 990 g >>= 9; 991 b >>= 10; 992 break; 993 case V4L2_PIX_FMT_RGB565: 994 case V4L2_PIX_FMT_RGB565X: 995 r >>= 7; 996 g >>= 6; 997 b >>= 7; 998 break; 999 case V4L2_PIX_FMT_RGB444: 1000 case V4L2_PIX_FMT_XRGB444: 1001 case V4L2_PIX_FMT_ARGB444: 1002 r >>= 8; 1003 g >>= 8; 1004 b >>= 8; 1005 break; 1006 case V4L2_PIX_FMT_RGB555: 1007 case V4L2_PIX_FMT_XRGB555: 1008 case V4L2_PIX_FMT_ARGB555: 1009 case V4L2_PIX_FMT_RGB555X: 1010 case V4L2_PIX_FMT_XRGB555X: 1011 case V4L2_PIX_FMT_ARGB555X: 1012 r >>= 7; 1013 g >>= 7; 1014 b >>= 7; 1015 break; 1016 case V4L2_PIX_FMT_BGR666: 1017 r >>= 6; 1018 g >>= 6; 1019 b >>= 6; 1020 break; 1021 default: 1022 r >>= 4; 1023 g >>= 4; 1024 b >>= 4; 1025 break; 1026 } 1027 1028 tpg->colors[k][0] = r; 1029 tpg->colors[k][1] = g; 1030 tpg->colors[k][2] = b; 1031 break; 1032 } 1033 } 1034 } 1035 1036 static void tpg_precalculate_colors(struct tpg_data *tpg) 1037 { 1038 int k; 1039 1040 for (k = 0; k < TPG_COLOR_MAX; k++) 1041 precalculate_color(tpg, k); 1042 } 1043 1044 /* 'odd' is true for pixels 1, 3, 5, etc. and false for pixels 0, 2, 4, etc. */ 1045 static void gen_twopix(struct tpg_data *tpg, 1046 u8 buf[TPG_MAX_PLANES][8], int color, bool odd) 1047 { 1048 unsigned offset = odd * tpg->twopixelsize[0] / 2; 1049 u8 alpha = tpg->alpha_component; 1050 u8 r_y_h, g_u_s, b_v; 1051 1052 if (tpg->alpha_red_only && color != TPG_COLOR_CSC_RED && 1053 color != TPG_COLOR_100_RED && 1054 color != TPG_COLOR_75_RED) 1055 alpha = 0; 1056 if (color == TPG_COLOR_RANDOM) 1057 precalculate_color(tpg, color); 1058 r_y_h = tpg->colors[color][0]; /* R or precalculated Y, H */ 1059 g_u_s = tpg->colors[color][1]; /* G or precalculated U, V */ 1060 b_v = tpg->colors[color][2]; /* B or precalculated V */ 1061 1062 switch (tpg->fourcc) { 1063 case V4L2_PIX_FMT_GREY: 1064 buf[0][offset] = r_y_h; 1065 break; 1066 case V4L2_PIX_FMT_Y10: 1067 buf[0][offset] = (r_y_h << 2) & 0xff; 1068 buf[0][offset+1] = r_y_h >> 6; 1069 break; 1070 case V4L2_PIX_FMT_Y12: 1071 buf[0][offset] = (r_y_h << 4) & 0xff; 1072 buf[0][offset+1] = r_y_h >> 4; 1073 break; 1074 case V4L2_PIX_FMT_Y16: 1075 case V4L2_PIX_FMT_Z16: 1076 /* 1077 * Ideally both bytes should be set to r_y_h, but then you won't 1078 * be able to detect endian problems. So keep it 0 except for 1079 * the corner case where r_y_h is 0xff so white really will be 1080 * white (0xffff). 1081 */ 1082 buf[0][offset] = r_y_h == 0xff ? r_y_h : 0; 1083 buf[0][offset+1] = r_y_h; 1084 break; 1085 case V4L2_PIX_FMT_Y16_BE: 1086 /* See comment for V4L2_PIX_FMT_Y16 above */ 1087 buf[0][offset] = r_y_h; 1088 buf[0][offset+1] = r_y_h == 0xff ? r_y_h : 0; 1089 break; 1090 case V4L2_PIX_FMT_YUV422M: 1091 case V4L2_PIX_FMT_YUV422P: 1092 case V4L2_PIX_FMT_YUV420: 1093 case V4L2_PIX_FMT_YUV420M: 1094 buf[0][offset] = r_y_h; 1095 if (odd) { 1096 buf[1][0] = (buf[1][0] + g_u_s) / 2; 1097 buf[2][0] = (buf[2][0] + b_v) / 2; 1098 buf[1][1] = buf[1][0]; 1099 buf[2][1] = buf[2][0]; 1100 break; 1101 } 1102 buf[1][0] = g_u_s; 1103 buf[2][0] = b_v; 1104 break; 1105 case V4L2_PIX_FMT_YVU422M: 1106 case V4L2_PIX_FMT_YVU420: 1107 case V4L2_PIX_FMT_YVU420M: 1108 buf[0][offset] = r_y_h; 1109 if (odd) { 1110 buf[1][0] = (buf[1][0] + b_v) / 2; 1111 buf[2][0] = (buf[2][0] + g_u_s) / 2; 1112 buf[1][1] = buf[1][0]; 1113 buf[2][1] = buf[2][0]; 1114 break; 1115 } 1116 buf[1][0] = b_v; 1117 buf[2][0] = g_u_s; 1118 break; 1119 1120 case V4L2_PIX_FMT_NV12: 1121 case V4L2_PIX_FMT_NV12M: 1122 case V4L2_PIX_FMT_NV16: 1123 case V4L2_PIX_FMT_NV16M: 1124 buf[0][offset] = r_y_h; 1125 if (odd) { 1126 buf[1][0] = (buf[1][0] + g_u_s) / 2; 1127 buf[1][1] = (buf[1][1] + b_v) / 2; 1128 break; 1129 } 1130 buf[1][0] = g_u_s; 1131 buf[1][1] = b_v; 1132 break; 1133 case V4L2_PIX_FMT_NV21: 1134 case V4L2_PIX_FMT_NV21M: 1135 case V4L2_PIX_FMT_NV61: 1136 case V4L2_PIX_FMT_NV61M: 1137 buf[0][offset] = r_y_h; 1138 if (odd) { 1139 buf[1][0] = (buf[1][0] + b_v) / 2; 1140 buf[1][1] = (buf[1][1] + g_u_s) / 2; 1141 break; 1142 } 1143 buf[1][0] = b_v; 1144 buf[1][1] = g_u_s; 1145 break; 1146 1147 case V4L2_PIX_FMT_YUV444M: 1148 buf[0][offset] = r_y_h; 1149 buf[1][offset] = g_u_s; 1150 buf[2][offset] = b_v; 1151 break; 1152 1153 case V4L2_PIX_FMT_YVU444M: 1154 buf[0][offset] = r_y_h; 1155 buf[1][offset] = b_v; 1156 buf[2][offset] = g_u_s; 1157 break; 1158 1159 case V4L2_PIX_FMT_NV24: 1160 buf[0][offset] = r_y_h; 1161 buf[1][2 * offset] = g_u_s; 1162 buf[1][(2 * offset + 1) % 8] = b_v; 1163 break; 1164 1165 case V4L2_PIX_FMT_NV42: 1166 buf[0][offset] = r_y_h; 1167 buf[1][2 * offset] = b_v; 1168 buf[1][(2 * offset + 1) % 8] = g_u_s; 1169 break; 1170 1171 case V4L2_PIX_FMT_YUYV: 1172 buf[0][offset] = r_y_h; 1173 if (odd) { 1174 buf[0][1] = (buf[0][1] + g_u_s) / 2; 1175 buf[0][3] = (buf[0][3] + b_v) / 2; 1176 break; 1177 } 1178 buf[0][1] = g_u_s; 1179 buf[0][3] = b_v; 1180 break; 1181 case V4L2_PIX_FMT_UYVY: 1182 buf[0][offset + 1] = r_y_h; 1183 if (odd) { 1184 buf[0][0] = (buf[0][0] + g_u_s) / 2; 1185 buf[0][2] = (buf[0][2] + b_v) / 2; 1186 break; 1187 } 1188 buf[0][0] = g_u_s; 1189 buf[0][2] = b_v; 1190 break; 1191 case V4L2_PIX_FMT_YVYU: 1192 buf[0][offset] = r_y_h; 1193 if (odd) { 1194 buf[0][1] = (buf[0][1] + b_v) / 2; 1195 buf[0][3] = (buf[0][3] + g_u_s) / 2; 1196 break; 1197 } 1198 buf[0][1] = b_v; 1199 buf[0][3] = g_u_s; 1200 break; 1201 case V4L2_PIX_FMT_VYUY: 1202 buf[0][offset + 1] = r_y_h; 1203 if (odd) { 1204 buf[0][0] = (buf[0][0] + b_v) / 2; 1205 buf[0][2] = (buf[0][2] + g_u_s) / 2; 1206 break; 1207 } 1208 buf[0][0] = b_v; 1209 buf[0][2] = g_u_s; 1210 break; 1211 case V4L2_PIX_FMT_RGB332: 1212 buf[0][offset] = (r_y_h << 5) | (g_u_s << 2) | b_v; 1213 break; 1214 case V4L2_PIX_FMT_YUV565: 1215 case V4L2_PIX_FMT_RGB565: 1216 buf[0][offset] = (g_u_s << 5) | b_v; 1217 buf[0][offset + 1] = (r_y_h << 3) | (g_u_s >> 3); 1218 break; 1219 case V4L2_PIX_FMT_RGB565X: 1220 buf[0][offset] = (r_y_h << 3) | (g_u_s >> 3); 1221 buf[0][offset + 1] = (g_u_s << 5) | b_v; 1222 break; 1223 case V4L2_PIX_FMT_RGB444: 1224 case V4L2_PIX_FMT_XRGB444: 1225 alpha = 0; 1226 /* fall through */ 1227 case V4L2_PIX_FMT_YUV444: 1228 case V4L2_PIX_FMT_ARGB444: 1229 buf[0][offset] = (g_u_s << 4) | b_v; 1230 buf[0][offset + 1] = (alpha & 0xf0) | r_y_h; 1231 break; 1232 case V4L2_PIX_FMT_RGB555: 1233 case V4L2_PIX_FMT_XRGB555: 1234 alpha = 0; 1235 /* fall through */ 1236 case V4L2_PIX_FMT_YUV555: 1237 case V4L2_PIX_FMT_ARGB555: 1238 buf[0][offset] = (g_u_s << 5) | b_v; 1239 buf[0][offset + 1] = (alpha & 0x80) | (r_y_h << 2) 1240 | (g_u_s >> 3); 1241 break; 1242 case V4L2_PIX_FMT_RGB555X: 1243 case V4L2_PIX_FMT_XRGB555X: 1244 alpha = 0; 1245 /* fall through */ 1246 case V4L2_PIX_FMT_ARGB555X: 1247 buf[0][offset] = (alpha & 0x80) | (r_y_h << 2) | (g_u_s >> 3); 1248 buf[0][offset + 1] = (g_u_s << 5) | b_v; 1249 break; 1250 case V4L2_PIX_FMT_RGB24: 1251 case V4L2_PIX_FMT_HSV24: 1252 buf[0][offset] = r_y_h; 1253 buf[0][offset + 1] = g_u_s; 1254 buf[0][offset + 2] = b_v; 1255 break; 1256 case V4L2_PIX_FMT_BGR24: 1257 buf[0][offset] = b_v; 1258 buf[0][offset + 1] = g_u_s; 1259 buf[0][offset + 2] = r_y_h; 1260 break; 1261 case V4L2_PIX_FMT_BGR666: 1262 buf[0][offset] = (b_v << 2) | (g_u_s >> 4); 1263 buf[0][offset + 1] = (g_u_s << 4) | (r_y_h >> 2); 1264 buf[0][offset + 2] = r_y_h << 6; 1265 buf[0][offset + 3] = 0; 1266 break; 1267 case V4L2_PIX_FMT_RGB32: 1268 case V4L2_PIX_FMT_XRGB32: 1269 case V4L2_PIX_FMT_HSV32: 1270 alpha = 0; 1271 /* fall through */ 1272 case V4L2_PIX_FMT_YUV32: 1273 case V4L2_PIX_FMT_ARGB32: 1274 buf[0][offset] = alpha; 1275 buf[0][offset + 1] = r_y_h; 1276 buf[0][offset + 2] = g_u_s; 1277 buf[0][offset + 3] = b_v; 1278 break; 1279 case V4L2_PIX_FMT_BGR32: 1280 case V4L2_PIX_FMT_XBGR32: 1281 alpha = 0; 1282 /* fall through */ 1283 case V4L2_PIX_FMT_ABGR32: 1284 buf[0][offset] = b_v; 1285 buf[0][offset + 1] = g_u_s; 1286 buf[0][offset + 2] = r_y_h; 1287 buf[0][offset + 3] = alpha; 1288 break; 1289 case V4L2_PIX_FMT_SBGGR8: 1290 buf[0][offset] = odd ? g_u_s : b_v; 1291 buf[1][offset] = odd ? r_y_h : g_u_s; 1292 break; 1293 case V4L2_PIX_FMT_SGBRG8: 1294 buf[0][offset] = odd ? b_v : g_u_s; 1295 buf[1][offset] = odd ? g_u_s : r_y_h; 1296 break; 1297 case V4L2_PIX_FMT_SGRBG8: 1298 buf[0][offset] = odd ? r_y_h : g_u_s; 1299 buf[1][offset] = odd ? g_u_s : b_v; 1300 break; 1301 case V4L2_PIX_FMT_SRGGB8: 1302 buf[0][offset] = odd ? g_u_s : r_y_h; 1303 buf[1][offset] = odd ? b_v : g_u_s; 1304 break; 1305 case V4L2_PIX_FMT_SBGGR10: 1306 buf[0][offset] = odd ? g_u_s << 2 : b_v << 2; 1307 buf[0][offset + 1] = odd ? g_u_s >> 6 : b_v >> 6; 1308 buf[1][offset] = odd ? r_y_h << 2 : g_u_s << 2; 1309 buf[1][offset + 1] = odd ? r_y_h >> 6 : g_u_s >> 6; 1310 buf[0][offset] |= (buf[0][offset] >> 2) & 3; 1311 buf[1][offset] |= (buf[1][offset] >> 2) & 3; 1312 break; 1313 case V4L2_PIX_FMT_SGBRG10: 1314 buf[0][offset] = odd ? b_v << 2 : g_u_s << 2; 1315 buf[0][offset + 1] = odd ? b_v >> 6 : g_u_s >> 6; 1316 buf[1][offset] = odd ? g_u_s << 2 : r_y_h << 2; 1317 buf[1][offset + 1] = odd ? g_u_s >> 6 : r_y_h >> 6; 1318 buf[0][offset] |= (buf[0][offset] >> 2) & 3; 1319 buf[1][offset] |= (buf[1][offset] >> 2) & 3; 1320 break; 1321 case V4L2_PIX_FMT_SGRBG10: 1322 buf[0][offset] = odd ? r_y_h << 2 : g_u_s << 2; 1323 buf[0][offset + 1] = odd ? r_y_h >> 6 : g_u_s >> 6; 1324 buf[1][offset] = odd ? g_u_s << 2 : b_v << 2; 1325 buf[1][offset + 1] = odd ? g_u_s >> 6 : b_v >> 6; 1326 buf[0][offset] |= (buf[0][offset] >> 2) & 3; 1327 buf[1][offset] |= (buf[1][offset] >> 2) & 3; 1328 break; 1329 case V4L2_PIX_FMT_SRGGB10: 1330 buf[0][offset] = odd ? g_u_s << 2 : r_y_h << 2; 1331 buf[0][offset + 1] = odd ? g_u_s >> 6 : r_y_h >> 6; 1332 buf[1][offset] = odd ? b_v << 2 : g_u_s << 2; 1333 buf[1][offset + 1] = odd ? b_v >> 6 : g_u_s >> 6; 1334 buf[0][offset] |= (buf[0][offset] >> 2) & 3; 1335 buf[1][offset] |= (buf[1][offset] >> 2) & 3; 1336 break; 1337 case V4L2_PIX_FMT_SBGGR12: 1338 buf[0][offset] = odd ? g_u_s << 4 : b_v << 4; 1339 buf[0][offset + 1] = odd ? g_u_s >> 4 : b_v >> 4; 1340 buf[1][offset] = odd ? r_y_h << 4 : g_u_s << 4; 1341 buf[1][offset + 1] = odd ? r_y_h >> 4 : g_u_s >> 4; 1342 buf[0][offset] |= (buf[0][offset] >> 4) & 0xf; 1343 buf[1][offset] |= (buf[1][offset] >> 4) & 0xf; 1344 break; 1345 case V4L2_PIX_FMT_SGBRG12: 1346 buf[0][offset] = odd ? b_v << 4 : g_u_s << 4; 1347 buf[0][offset + 1] = odd ? b_v >> 4 : g_u_s >> 4; 1348 buf[1][offset] = odd ? g_u_s << 4 : r_y_h << 4; 1349 buf[1][offset + 1] = odd ? g_u_s >> 4 : r_y_h >> 4; 1350 buf[0][offset] |= (buf[0][offset] >> 4) & 0xf; 1351 buf[1][offset] |= (buf[1][offset] >> 4) & 0xf; 1352 break; 1353 case V4L2_PIX_FMT_SGRBG12: 1354 buf[0][offset] = odd ? r_y_h << 4 : g_u_s << 4; 1355 buf[0][offset + 1] = odd ? r_y_h >> 4 : g_u_s >> 4; 1356 buf[1][offset] = odd ? g_u_s << 4 : b_v << 4; 1357 buf[1][offset + 1] = odd ? g_u_s >> 4 : b_v >> 4; 1358 buf[0][offset] |= (buf[0][offset] >> 4) & 0xf; 1359 buf[1][offset] |= (buf[1][offset] >> 4) & 0xf; 1360 break; 1361 case V4L2_PIX_FMT_SRGGB12: 1362 buf[0][offset] = odd ? g_u_s << 4 : r_y_h << 4; 1363 buf[0][offset + 1] = odd ? g_u_s >> 4 : r_y_h >> 4; 1364 buf[1][offset] = odd ? b_v << 4 : g_u_s << 4; 1365 buf[1][offset + 1] = odd ? b_v >> 4 : g_u_s >> 4; 1366 buf[0][offset] |= (buf[0][offset] >> 4) & 0xf; 1367 buf[1][offset] |= (buf[1][offset] >> 4) & 0xf; 1368 break; 1369 case V4L2_PIX_FMT_SBGGR16: 1370 buf[0][offset] = buf[0][offset + 1] = odd ? g_u_s : b_v; 1371 buf[1][offset] = buf[1][offset + 1] = odd ? r_y_h : g_u_s; 1372 break; 1373 case V4L2_PIX_FMT_SGBRG16: 1374 buf[0][offset] = buf[0][offset + 1] = odd ? b_v : g_u_s; 1375 buf[1][offset] = buf[1][offset + 1] = odd ? g_u_s : r_y_h; 1376 break; 1377 case V4L2_PIX_FMT_SGRBG16: 1378 buf[0][offset] = buf[0][offset + 1] = odd ? r_y_h : g_u_s; 1379 buf[1][offset] = buf[1][offset + 1] = odd ? g_u_s : b_v; 1380 break; 1381 case V4L2_PIX_FMT_SRGGB16: 1382 buf[0][offset] = buf[0][offset + 1] = odd ? g_u_s : r_y_h; 1383 buf[1][offset] = buf[1][offset + 1] = odd ? b_v : g_u_s; 1384 break; 1385 } 1386 } 1387 1388 unsigned tpg_g_interleaved_plane(const struct tpg_data *tpg, unsigned buf_line) 1389 { 1390 switch (tpg->fourcc) { 1391 case V4L2_PIX_FMT_SBGGR8: 1392 case V4L2_PIX_FMT_SGBRG8: 1393 case V4L2_PIX_FMT_SGRBG8: 1394 case V4L2_PIX_FMT_SRGGB8: 1395 case V4L2_PIX_FMT_SBGGR10: 1396 case V4L2_PIX_FMT_SGBRG10: 1397 case V4L2_PIX_FMT_SGRBG10: 1398 case V4L2_PIX_FMT_SRGGB10: 1399 case V4L2_PIX_FMT_SBGGR12: 1400 case V4L2_PIX_FMT_SGBRG12: 1401 case V4L2_PIX_FMT_SGRBG12: 1402 case V4L2_PIX_FMT_SRGGB12: 1403 case V4L2_PIX_FMT_SBGGR16: 1404 case V4L2_PIX_FMT_SGBRG16: 1405 case V4L2_PIX_FMT_SGRBG16: 1406 case V4L2_PIX_FMT_SRGGB16: 1407 return buf_line & 1; 1408 default: 1409 return 0; 1410 } 1411 } 1412 EXPORT_SYMBOL_GPL(tpg_g_interleaved_plane); 1413 1414 /* Return how many pattern lines are used by the current pattern. */ 1415 static unsigned tpg_get_pat_lines(const struct tpg_data *tpg) 1416 { 1417 switch (tpg->pattern) { 1418 case TPG_PAT_CHECKERS_16X16: 1419 case TPG_PAT_CHECKERS_2X2: 1420 case TPG_PAT_CHECKERS_1X1: 1421 case TPG_PAT_COLOR_CHECKERS_2X2: 1422 case TPG_PAT_COLOR_CHECKERS_1X1: 1423 case TPG_PAT_ALTERNATING_HLINES: 1424 case TPG_PAT_CROSS_1_PIXEL: 1425 case TPG_PAT_CROSS_2_PIXELS: 1426 case TPG_PAT_CROSS_10_PIXELS: 1427 return 2; 1428 case TPG_PAT_100_COLORSQUARES: 1429 case TPG_PAT_100_HCOLORBAR: 1430 return 8; 1431 default: 1432 return 1; 1433 } 1434 } 1435 1436 /* Which pattern line should be used for the given frame line. */ 1437 static unsigned tpg_get_pat_line(const struct tpg_data *tpg, unsigned line) 1438 { 1439 switch (tpg->pattern) { 1440 case TPG_PAT_CHECKERS_16X16: 1441 return (line >> 4) & 1; 1442 case TPG_PAT_CHECKERS_1X1: 1443 case TPG_PAT_COLOR_CHECKERS_1X1: 1444 case TPG_PAT_ALTERNATING_HLINES: 1445 return line & 1; 1446 case TPG_PAT_CHECKERS_2X2: 1447 case TPG_PAT_COLOR_CHECKERS_2X2: 1448 return (line & 2) >> 1; 1449 case TPG_PAT_100_COLORSQUARES: 1450 case TPG_PAT_100_HCOLORBAR: 1451 return (line * 8) / tpg->src_height; 1452 case TPG_PAT_CROSS_1_PIXEL: 1453 return line == tpg->src_height / 2; 1454 case TPG_PAT_CROSS_2_PIXELS: 1455 return (line + 1) / 2 == tpg->src_height / 4; 1456 case TPG_PAT_CROSS_10_PIXELS: 1457 return (line + 10) / 20 == tpg->src_height / 40; 1458 default: 1459 return 0; 1460 } 1461 } 1462 1463 /* 1464 * Which color should be used for the given pattern line and X coordinate. 1465 * Note: x is in the range 0 to 2 * tpg->src_width. 1466 */ 1467 static enum tpg_color tpg_get_color(const struct tpg_data *tpg, 1468 unsigned pat_line, unsigned x) 1469 { 1470 /* Maximum number of bars are TPG_COLOR_MAX - otherwise, the input print code 1471 should be modified */ 1472 static const enum tpg_color bars[3][8] = { 1473 /* Standard ITU-R 75% color bar sequence */ 1474 { TPG_COLOR_CSC_WHITE, TPG_COLOR_75_YELLOW, 1475 TPG_COLOR_75_CYAN, TPG_COLOR_75_GREEN, 1476 TPG_COLOR_75_MAGENTA, TPG_COLOR_75_RED, 1477 TPG_COLOR_75_BLUE, TPG_COLOR_100_BLACK, }, 1478 /* Standard ITU-R 100% color bar sequence */ 1479 { TPG_COLOR_100_WHITE, TPG_COLOR_100_YELLOW, 1480 TPG_COLOR_100_CYAN, TPG_COLOR_100_GREEN, 1481 TPG_COLOR_100_MAGENTA, TPG_COLOR_100_RED, 1482 TPG_COLOR_100_BLUE, TPG_COLOR_100_BLACK, }, 1483 /* Color bar sequence suitable to test CSC */ 1484 { TPG_COLOR_CSC_WHITE, TPG_COLOR_CSC_YELLOW, 1485 TPG_COLOR_CSC_CYAN, TPG_COLOR_CSC_GREEN, 1486 TPG_COLOR_CSC_MAGENTA, TPG_COLOR_CSC_RED, 1487 TPG_COLOR_CSC_BLUE, TPG_COLOR_CSC_BLACK, }, 1488 }; 1489 1490 switch (tpg->pattern) { 1491 case TPG_PAT_75_COLORBAR: 1492 case TPG_PAT_100_COLORBAR: 1493 case TPG_PAT_CSC_COLORBAR: 1494 return bars[tpg->pattern][((x * 8) / tpg->src_width) % 8]; 1495 case TPG_PAT_100_COLORSQUARES: 1496 return bars[1][(pat_line + (x * 8) / tpg->src_width) % 8]; 1497 case TPG_PAT_100_HCOLORBAR: 1498 return bars[1][pat_line]; 1499 case TPG_PAT_BLACK: 1500 return TPG_COLOR_100_BLACK; 1501 case TPG_PAT_WHITE: 1502 return TPG_COLOR_100_WHITE; 1503 case TPG_PAT_RED: 1504 return TPG_COLOR_100_RED; 1505 case TPG_PAT_GREEN: 1506 return TPG_COLOR_100_GREEN; 1507 case TPG_PAT_BLUE: 1508 return TPG_COLOR_100_BLUE; 1509 case TPG_PAT_CHECKERS_16X16: 1510 return (((x >> 4) & 1) ^ (pat_line & 1)) ? 1511 TPG_COLOR_100_BLACK : TPG_COLOR_100_WHITE; 1512 case TPG_PAT_CHECKERS_1X1: 1513 return ((x & 1) ^ (pat_line & 1)) ? 1514 TPG_COLOR_100_WHITE : TPG_COLOR_100_BLACK; 1515 case TPG_PAT_COLOR_CHECKERS_1X1: 1516 return ((x & 1) ^ (pat_line & 1)) ? 1517 TPG_COLOR_100_RED : TPG_COLOR_100_BLUE; 1518 case TPG_PAT_CHECKERS_2X2: 1519 return (((x >> 1) & 1) ^ (pat_line & 1)) ? 1520 TPG_COLOR_100_WHITE : TPG_COLOR_100_BLACK; 1521 case TPG_PAT_COLOR_CHECKERS_2X2: 1522 return (((x >> 1) & 1) ^ (pat_line & 1)) ? 1523 TPG_COLOR_100_RED : TPG_COLOR_100_BLUE; 1524 case TPG_PAT_ALTERNATING_HLINES: 1525 return pat_line ? TPG_COLOR_100_WHITE : TPG_COLOR_100_BLACK; 1526 case TPG_PAT_ALTERNATING_VLINES: 1527 return (x & 1) ? TPG_COLOR_100_WHITE : TPG_COLOR_100_BLACK; 1528 case TPG_PAT_CROSS_1_PIXEL: 1529 if (pat_line || (x % tpg->src_width) == tpg->src_width / 2) 1530 return TPG_COLOR_100_BLACK; 1531 return TPG_COLOR_100_WHITE; 1532 case TPG_PAT_CROSS_2_PIXELS: 1533 if (pat_line || ((x % tpg->src_width) + 1) / 2 == tpg->src_width / 4) 1534 return TPG_COLOR_100_BLACK; 1535 return TPG_COLOR_100_WHITE; 1536 case TPG_PAT_CROSS_10_PIXELS: 1537 if (pat_line || ((x % tpg->src_width) + 10) / 20 == tpg->src_width / 40) 1538 return TPG_COLOR_100_BLACK; 1539 return TPG_COLOR_100_WHITE; 1540 case TPG_PAT_GRAY_RAMP: 1541 return TPG_COLOR_RAMP + ((x % tpg->src_width) * 256) / tpg->src_width; 1542 default: 1543 return TPG_COLOR_100_RED; 1544 } 1545 } 1546 1547 /* 1548 * Given the pixel aspect ratio and video aspect ratio calculate the 1549 * coordinates of a centered square and the coordinates of the border of 1550 * the active video area. The coordinates are relative to the source 1551 * frame rectangle. 1552 */ 1553 static void tpg_calculate_square_border(struct tpg_data *tpg) 1554 { 1555 unsigned w = tpg->src_width; 1556 unsigned h = tpg->src_height; 1557 unsigned sq_w, sq_h; 1558 1559 sq_w = (w * 2 / 5) & ~1; 1560 if (((w - sq_w) / 2) & 1) 1561 sq_w += 2; 1562 sq_h = sq_w; 1563 tpg->square.width = sq_w; 1564 if (tpg->vid_aspect == TPG_VIDEO_ASPECT_16X9_ANAMORPHIC) { 1565 unsigned ana_sq_w = (sq_w / 4) * 3; 1566 1567 if (((w - ana_sq_w) / 2) & 1) 1568 ana_sq_w += 2; 1569 tpg->square.width = ana_sq_w; 1570 } 1571 tpg->square.left = (w - tpg->square.width) / 2; 1572 if (tpg->pix_aspect == TPG_PIXEL_ASPECT_NTSC) 1573 sq_h = sq_w * 10 / 11; 1574 else if (tpg->pix_aspect == TPG_PIXEL_ASPECT_PAL) 1575 sq_h = sq_w * 59 / 54; 1576 tpg->square.height = sq_h; 1577 tpg->square.top = (h - sq_h) / 2; 1578 tpg->border.left = 0; 1579 tpg->border.width = w; 1580 tpg->border.top = 0; 1581 tpg->border.height = h; 1582 switch (tpg->vid_aspect) { 1583 case TPG_VIDEO_ASPECT_4X3: 1584 if (tpg->pix_aspect) 1585 return; 1586 if (3 * w >= 4 * h) { 1587 tpg->border.width = ((4 * h) / 3) & ~1; 1588 if (((w - tpg->border.width) / 2) & ~1) 1589 tpg->border.width -= 2; 1590 tpg->border.left = (w - tpg->border.width) / 2; 1591 break; 1592 } 1593 tpg->border.height = ((3 * w) / 4) & ~1; 1594 tpg->border.top = (h - tpg->border.height) / 2; 1595 break; 1596 case TPG_VIDEO_ASPECT_14X9_CENTRE: 1597 if (tpg->pix_aspect) { 1598 tpg->border.height = tpg->pix_aspect == TPG_PIXEL_ASPECT_NTSC ? 420 : 506; 1599 tpg->border.top = (h - tpg->border.height) / 2; 1600 break; 1601 } 1602 if (9 * w >= 14 * h) { 1603 tpg->border.width = ((14 * h) / 9) & ~1; 1604 if (((w - tpg->border.width) / 2) & ~1) 1605 tpg->border.width -= 2; 1606 tpg->border.left = (w - tpg->border.width) / 2; 1607 break; 1608 } 1609 tpg->border.height = ((9 * w) / 14) & ~1; 1610 tpg->border.top = (h - tpg->border.height) / 2; 1611 break; 1612 case TPG_VIDEO_ASPECT_16X9_CENTRE: 1613 if (tpg->pix_aspect) { 1614 tpg->border.height = tpg->pix_aspect == TPG_PIXEL_ASPECT_NTSC ? 368 : 442; 1615 tpg->border.top = (h - tpg->border.height) / 2; 1616 break; 1617 } 1618 if (9 * w >= 16 * h) { 1619 tpg->border.width = ((16 * h) / 9) & ~1; 1620 if (((w - tpg->border.width) / 2) & ~1) 1621 tpg->border.width -= 2; 1622 tpg->border.left = (w - tpg->border.width) / 2; 1623 break; 1624 } 1625 tpg->border.height = ((9 * w) / 16) & ~1; 1626 tpg->border.top = (h - tpg->border.height) / 2; 1627 break; 1628 default: 1629 break; 1630 } 1631 } 1632 1633 static void tpg_precalculate_line(struct tpg_data *tpg) 1634 { 1635 enum tpg_color contrast; 1636 u8 pix[TPG_MAX_PLANES][8]; 1637 unsigned pat; 1638 unsigned p; 1639 unsigned x; 1640 1641 switch (tpg->pattern) { 1642 case TPG_PAT_GREEN: 1643 contrast = TPG_COLOR_100_RED; 1644 break; 1645 case TPG_PAT_CSC_COLORBAR: 1646 contrast = TPG_COLOR_CSC_GREEN; 1647 break; 1648 default: 1649 contrast = TPG_COLOR_100_GREEN; 1650 break; 1651 } 1652 1653 for (pat = 0; pat < tpg_get_pat_lines(tpg); pat++) { 1654 /* Coarse scaling with Bresenham */ 1655 unsigned int_part = tpg->src_width / tpg->scaled_width; 1656 unsigned fract_part = tpg->src_width % tpg->scaled_width; 1657 unsigned src_x = 0; 1658 unsigned error = 0; 1659 1660 for (x = 0; x < tpg->scaled_width * 2; x += 2) { 1661 unsigned real_x = src_x; 1662 enum tpg_color color1, color2; 1663 1664 real_x = tpg->hflip ? tpg->src_width * 2 - real_x - 2 : real_x; 1665 color1 = tpg_get_color(tpg, pat, real_x); 1666 1667 src_x += int_part; 1668 error += fract_part; 1669 if (error >= tpg->scaled_width) { 1670 error -= tpg->scaled_width; 1671 src_x++; 1672 } 1673 1674 real_x = src_x; 1675 real_x = tpg->hflip ? tpg->src_width * 2 - real_x - 2 : real_x; 1676 color2 = tpg_get_color(tpg, pat, real_x); 1677 1678 src_x += int_part; 1679 error += fract_part; 1680 if (error >= tpg->scaled_width) { 1681 error -= tpg->scaled_width; 1682 src_x++; 1683 } 1684 1685 gen_twopix(tpg, pix, tpg->hflip ? color2 : color1, 0); 1686 gen_twopix(tpg, pix, tpg->hflip ? color1 : color2, 1); 1687 for (p = 0; p < tpg->planes; p++) { 1688 unsigned twopixsize = tpg->twopixelsize[p]; 1689 unsigned hdiv = tpg->hdownsampling[p]; 1690 u8 *pos = tpg->lines[pat][p] + tpg_hdiv(tpg, p, x); 1691 1692 memcpy(pos, pix[p], twopixsize / hdiv); 1693 } 1694 } 1695 } 1696 1697 if (tpg->vdownsampling[tpg->planes - 1] > 1) { 1698 unsigned pat_lines = tpg_get_pat_lines(tpg); 1699 1700 for (pat = 0; pat < pat_lines; pat++) { 1701 unsigned next_pat = (pat + 1) % pat_lines; 1702 1703 for (p = 1; p < tpg->planes; p++) { 1704 unsigned w = tpg_hdiv(tpg, p, tpg->scaled_width * 2); 1705 u8 *pos1 = tpg->lines[pat][p]; 1706 u8 *pos2 = tpg->lines[next_pat][p]; 1707 u8 *dest = tpg->downsampled_lines[pat][p]; 1708 1709 for (x = 0; x < w; x++, pos1++, pos2++, dest++) 1710 *dest = ((u16)*pos1 + (u16)*pos2) / 2; 1711 } 1712 } 1713 } 1714 1715 gen_twopix(tpg, pix, contrast, 0); 1716 gen_twopix(tpg, pix, contrast, 1); 1717 for (p = 0; p < tpg->planes; p++) { 1718 unsigned twopixsize = tpg->twopixelsize[p]; 1719 u8 *pos = tpg->contrast_line[p]; 1720 1721 for (x = 0; x < tpg->scaled_width; x += 2, pos += twopixsize) 1722 memcpy(pos, pix[p], twopixsize); 1723 } 1724 1725 gen_twopix(tpg, pix, TPG_COLOR_100_BLACK, 0); 1726 gen_twopix(tpg, pix, TPG_COLOR_100_BLACK, 1); 1727 for (p = 0; p < tpg->planes; p++) { 1728 unsigned twopixsize = tpg->twopixelsize[p]; 1729 u8 *pos = tpg->black_line[p]; 1730 1731 for (x = 0; x < tpg->scaled_width; x += 2, pos += twopixsize) 1732 memcpy(pos, pix[p], twopixsize); 1733 } 1734 1735 for (x = 0; x < tpg->scaled_width * 2; x += 2) { 1736 gen_twopix(tpg, pix, TPG_COLOR_RANDOM, 0); 1737 gen_twopix(tpg, pix, TPG_COLOR_RANDOM, 1); 1738 for (p = 0; p < tpg->planes; p++) { 1739 unsigned twopixsize = tpg->twopixelsize[p]; 1740 u8 *pos = tpg->random_line[p] + x * twopixsize / 2; 1741 1742 memcpy(pos, pix[p], twopixsize); 1743 } 1744 } 1745 1746 gen_twopix(tpg, tpg->textbg, TPG_COLOR_TEXTBG, 0); 1747 gen_twopix(tpg, tpg->textbg, TPG_COLOR_TEXTBG, 1); 1748 gen_twopix(tpg, tpg->textfg, TPG_COLOR_TEXTFG, 0); 1749 gen_twopix(tpg, tpg->textfg, TPG_COLOR_TEXTFG, 1); 1750 } 1751 1752 /* need this to do rgb24 rendering */ 1753 typedef struct { u16 __; u8 _; } __packed x24; 1754 1755 #define PRINTSTR(PIXTYPE) do { \ 1756 unsigned vdiv = tpg->vdownsampling[p]; \ 1757 unsigned hdiv = tpg->hdownsampling[p]; \ 1758 int line; \ 1759 PIXTYPE fg; \ 1760 PIXTYPE bg; \ 1761 memcpy(&fg, tpg->textfg[p], sizeof(PIXTYPE)); \ 1762 memcpy(&bg, tpg->textbg[p], sizeof(PIXTYPE)); \ 1763 \ 1764 for (line = first; line < 16; line += vdiv * step) { \ 1765 int l = tpg->vflip ? 15 - line : line; \ 1766 PIXTYPE *pos = (PIXTYPE *)(basep[p][(line / vdiv) & 1] + \ 1767 ((y * step + l) / (vdiv * div)) * tpg->bytesperline[p] + \ 1768 (x / hdiv) * sizeof(PIXTYPE)); \ 1769 unsigned s; \ 1770 \ 1771 for (s = 0; s < len; s++) { \ 1772 u8 chr = font8x16[text[s] * 16 + line]; \ 1773 \ 1774 if (hdiv == 2 && tpg->hflip) { \ 1775 pos[3] = (chr & (0x01 << 6) ? fg : bg); \ 1776 pos[2] = (chr & (0x01 << 4) ? fg : bg); \ 1777 pos[1] = (chr & (0x01 << 2) ? fg : bg); \ 1778 pos[0] = (chr & (0x01 << 0) ? fg : bg); \ 1779 } else if (hdiv == 2) { \ 1780 pos[0] = (chr & (0x01 << 7) ? fg : bg); \ 1781 pos[1] = (chr & (0x01 << 5) ? fg : bg); \ 1782 pos[2] = (chr & (0x01 << 3) ? fg : bg); \ 1783 pos[3] = (chr & (0x01 << 1) ? fg : bg); \ 1784 } else if (tpg->hflip) { \ 1785 pos[7] = (chr & (0x01 << 7) ? fg : bg); \ 1786 pos[6] = (chr & (0x01 << 6) ? fg : bg); \ 1787 pos[5] = (chr & (0x01 << 5) ? fg : bg); \ 1788 pos[4] = (chr & (0x01 << 4) ? fg : bg); \ 1789 pos[3] = (chr & (0x01 << 3) ? fg : bg); \ 1790 pos[2] = (chr & (0x01 << 2) ? fg : bg); \ 1791 pos[1] = (chr & (0x01 << 1) ? fg : bg); \ 1792 pos[0] = (chr & (0x01 << 0) ? fg : bg); \ 1793 } else { \ 1794 pos[0] = (chr & (0x01 << 7) ? fg : bg); \ 1795 pos[1] = (chr & (0x01 << 6) ? fg : bg); \ 1796 pos[2] = (chr & (0x01 << 5) ? fg : bg); \ 1797 pos[3] = (chr & (0x01 << 4) ? fg : bg); \ 1798 pos[4] = (chr & (0x01 << 3) ? fg : bg); \ 1799 pos[5] = (chr & (0x01 << 2) ? fg : bg); \ 1800 pos[6] = (chr & (0x01 << 1) ? fg : bg); \ 1801 pos[7] = (chr & (0x01 << 0) ? fg : bg); \ 1802 } \ 1803 \ 1804 pos += (tpg->hflip ? -8 : 8) / (int)hdiv; \ 1805 } \ 1806 } \ 1807 } while (0) 1808 1809 static noinline void tpg_print_str_2(const struct tpg_data *tpg, u8 *basep[TPG_MAX_PLANES][2], 1810 unsigned p, unsigned first, unsigned div, unsigned step, 1811 int y, int x, char *text, unsigned len) 1812 { 1813 PRINTSTR(u8); 1814 } 1815 1816 static noinline void tpg_print_str_4(const struct tpg_data *tpg, u8 *basep[TPG_MAX_PLANES][2], 1817 unsigned p, unsigned first, unsigned div, unsigned step, 1818 int y, int x, char *text, unsigned len) 1819 { 1820 PRINTSTR(u16); 1821 } 1822 1823 static noinline void tpg_print_str_6(const struct tpg_data *tpg, u8 *basep[TPG_MAX_PLANES][2], 1824 unsigned p, unsigned first, unsigned div, unsigned step, 1825 int y, int x, char *text, unsigned len) 1826 { 1827 PRINTSTR(x24); 1828 } 1829 1830 static noinline void tpg_print_str_8(const struct tpg_data *tpg, u8 *basep[TPG_MAX_PLANES][2], 1831 unsigned p, unsigned first, unsigned div, unsigned step, 1832 int y, int x, char *text, unsigned len) 1833 { 1834 PRINTSTR(u32); 1835 } 1836 1837 void tpg_gen_text(const struct tpg_data *tpg, u8 *basep[TPG_MAX_PLANES][2], 1838 int y, int x, char *text) 1839 { 1840 unsigned step = V4L2_FIELD_HAS_T_OR_B(tpg->field) ? 2 : 1; 1841 unsigned div = step; 1842 unsigned first = 0; 1843 unsigned len = strlen(text); 1844 unsigned p; 1845 1846 if (font8x16 == NULL || basep == NULL) 1847 return; 1848 1849 /* Checks if it is possible to show string */ 1850 if (y + 16 >= tpg->compose.height || x + 8 >= tpg->compose.width) 1851 return; 1852 1853 if (len > (tpg->compose.width - x) / 8) 1854 len = (tpg->compose.width - x) / 8; 1855 if (tpg->vflip) 1856 y = tpg->compose.height - y - 16; 1857 if (tpg->hflip) 1858 x = tpg->compose.width - x - 8; 1859 y += tpg->compose.top; 1860 x += tpg->compose.left; 1861 if (tpg->field == V4L2_FIELD_BOTTOM) 1862 first = 1; 1863 else if (tpg->field == V4L2_FIELD_SEQ_TB || tpg->field == V4L2_FIELD_SEQ_BT) 1864 div = 2; 1865 1866 for (p = 0; p < tpg->planes; p++) { 1867 /* Print text */ 1868 switch (tpg->twopixelsize[p]) { 1869 case 2: 1870 tpg_print_str_2(tpg, basep, p, first, div, step, y, x, 1871 text, len); 1872 break; 1873 case 4: 1874 tpg_print_str_4(tpg, basep, p, first, div, step, y, x, 1875 text, len); 1876 break; 1877 case 6: 1878 tpg_print_str_6(tpg, basep, p, first, div, step, y, x, 1879 text, len); 1880 break; 1881 case 8: 1882 tpg_print_str_8(tpg, basep, p, first, div, step, y, x, 1883 text, len); 1884 break; 1885 } 1886 } 1887 } 1888 EXPORT_SYMBOL_GPL(tpg_gen_text); 1889 1890 void tpg_update_mv_step(struct tpg_data *tpg) 1891 { 1892 int factor = tpg->mv_hor_mode > TPG_MOVE_NONE ? -1 : 1; 1893 1894 if (tpg->hflip) 1895 factor = -factor; 1896 switch (tpg->mv_hor_mode) { 1897 case TPG_MOVE_NEG_FAST: 1898 case TPG_MOVE_POS_FAST: 1899 tpg->mv_hor_step = ((tpg->src_width + 319) / 320) * 4; 1900 break; 1901 case TPG_MOVE_NEG: 1902 case TPG_MOVE_POS: 1903 tpg->mv_hor_step = ((tpg->src_width + 639) / 640) * 4; 1904 break; 1905 case TPG_MOVE_NEG_SLOW: 1906 case TPG_MOVE_POS_SLOW: 1907 tpg->mv_hor_step = 2; 1908 break; 1909 case TPG_MOVE_NONE: 1910 tpg->mv_hor_step = 0; 1911 break; 1912 } 1913 if (factor < 0) 1914 tpg->mv_hor_step = tpg->src_width - tpg->mv_hor_step; 1915 1916 factor = tpg->mv_vert_mode > TPG_MOVE_NONE ? -1 : 1; 1917 switch (tpg->mv_vert_mode) { 1918 case TPG_MOVE_NEG_FAST: 1919 case TPG_MOVE_POS_FAST: 1920 tpg->mv_vert_step = ((tpg->src_width + 319) / 320) * 4; 1921 break; 1922 case TPG_MOVE_NEG: 1923 case TPG_MOVE_POS: 1924 tpg->mv_vert_step = ((tpg->src_width + 639) / 640) * 4; 1925 break; 1926 case TPG_MOVE_NEG_SLOW: 1927 case TPG_MOVE_POS_SLOW: 1928 tpg->mv_vert_step = 1; 1929 break; 1930 case TPG_MOVE_NONE: 1931 tpg->mv_vert_step = 0; 1932 break; 1933 } 1934 if (factor < 0) 1935 tpg->mv_vert_step = tpg->src_height - tpg->mv_vert_step; 1936 } 1937 EXPORT_SYMBOL_GPL(tpg_update_mv_step); 1938 1939 /* Map the line number relative to the crop rectangle to a frame line number */ 1940 static unsigned tpg_calc_frameline(const struct tpg_data *tpg, unsigned src_y, 1941 unsigned field) 1942 { 1943 switch (field) { 1944 case V4L2_FIELD_TOP: 1945 return tpg->crop.top + src_y * 2; 1946 case V4L2_FIELD_BOTTOM: 1947 return tpg->crop.top + src_y * 2 + 1; 1948 default: 1949 return src_y + tpg->crop.top; 1950 } 1951 } 1952 1953 /* 1954 * Map the line number relative to the compose rectangle to a destination 1955 * buffer line number. 1956 */ 1957 static unsigned tpg_calc_buffer_line(const struct tpg_data *tpg, unsigned y, 1958 unsigned field) 1959 { 1960 y += tpg->compose.top; 1961 switch (field) { 1962 case V4L2_FIELD_SEQ_TB: 1963 if (y & 1) 1964 return tpg->buf_height / 2 + y / 2; 1965 return y / 2; 1966 case V4L2_FIELD_SEQ_BT: 1967 if (y & 1) 1968 return y / 2; 1969 return tpg->buf_height / 2 + y / 2; 1970 default: 1971 return y; 1972 } 1973 } 1974 1975 static void tpg_recalc(struct tpg_data *tpg) 1976 { 1977 if (tpg->recalc_colors) { 1978 tpg->recalc_colors = false; 1979 tpg->recalc_lines = true; 1980 tpg->real_xfer_func = tpg->xfer_func; 1981 tpg->real_ycbcr_enc = tpg->ycbcr_enc; 1982 tpg->real_hsv_enc = tpg->hsv_enc; 1983 tpg->real_quantization = tpg->quantization; 1984 1985 if (tpg->xfer_func == V4L2_XFER_FUNC_DEFAULT) 1986 tpg->real_xfer_func = 1987 V4L2_MAP_XFER_FUNC_DEFAULT(tpg->colorspace); 1988 1989 if (tpg->ycbcr_enc == V4L2_YCBCR_ENC_DEFAULT) 1990 tpg->real_ycbcr_enc = 1991 V4L2_MAP_YCBCR_ENC_DEFAULT(tpg->colorspace); 1992 1993 if (tpg->quantization == V4L2_QUANTIZATION_DEFAULT) 1994 tpg->real_quantization = 1995 V4L2_MAP_QUANTIZATION_DEFAULT( 1996 tpg->color_enc != TGP_COLOR_ENC_YCBCR, 1997 tpg->colorspace, tpg->real_ycbcr_enc); 1998 1999 tpg_precalculate_colors(tpg); 2000 } 2001 if (tpg->recalc_square_border) { 2002 tpg->recalc_square_border = false; 2003 tpg_calculate_square_border(tpg); 2004 } 2005 if (tpg->recalc_lines) { 2006 tpg->recalc_lines = false; 2007 tpg_precalculate_line(tpg); 2008 } 2009 } 2010 2011 void tpg_calc_text_basep(struct tpg_data *tpg, 2012 u8 *basep[TPG_MAX_PLANES][2], unsigned p, u8 *vbuf) 2013 { 2014 unsigned stride = tpg->bytesperline[p]; 2015 unsigned h = tpg->buf_height; 2016 2017 tpg_recalc(tpg); 2018 2019 basep[p][0] = vbuf; 2020 basep[p][1] = vbuf; 2021 h /= tpg->vdownsampling[p]; 2022 if (tpg->field == V4L2_FIELD_SEQ_TB) 2023 basep[p][1] += h * stride / 2; 2024 else if (tpg->field == V4L2_FIELD_SEQ_BT) 2025 basep[p][0] += h * stride / 2; 2026 if (p == 0 && tpg->interleaved) 2027 tpg_calc_text_basep(tpg, basep, 1, vbuf); 2028 } 2029 EXPORT_SYMBOL_GPL(tpg_calc_text_basep); 2030 2031 static int tpg_pattern_avg(const struct tpg_data *tpg, 2032 unsigned pat1, unsigned pat2) 2033 { 2034 unsigned pat_lines = tpg_get_pat_lines(tpg); 2035 2036 if (pat1 == (pat2 + 1) % pat_lines) 2037 return pat2; 2038 if (pat2 == (pat1 + 1) % pat_lines) 2039 return pat1; 2040 return -1; 2041 } 2042 2043 static const char *tpg_color_enc_str(enum tgp_color_enc 2044 color_enc) 2045 { 2046 switch (color_enc) { 2047 case TGP_COLOR_ENC_HSV: 2048 return "HSV"; 2049 case TGP_COLOR_ENC_YCBCR: 2050 return "Y'CbCr"; 2051 case TGP_COLOR_ENC_LUMA: 2052 return "Luma"; 2053 case TGP_COLOR_ENC_RGB: 2054 default: 2055 return "R'G'B"; 2056 2057 } 2058 } 2059 2060 void tpg_log_status(struct tpg_data *tpg) 2061 { 2062 pr_info("tpg source WxH: %ux%u (%s)\n", 2063 tpg->src_width, tpg->src_height, 2064 tpg_color_enc_str(tpg->color_enc)); 2065 pr_info("tpg field: %u\n", tpg->field); 2066 pr_info("tpg crop: %ux%u@%dx%d\n", tpg->crop.width, tpg->crop.height, 2067 tpg->crop.left, tpg->crop.top); 2068 pr_info("tpg compose: %ux%u@%dx%d\n", tpg->compose.width, tpg->compose.height, 2069 tpg->compose.left, tpg->compose.top); 2070 pr_info("tpg colorspace: %d\n", tpg->colorspace); 2071 pr_info("tpg transfer function: %d/%d\n", tpg->xfer_func, tpg->real_xfer_func); 2072 if (tpg->color_enc == TGP_COLOR_ENC_HSV) 2073 pr_info("tpg HSV encoding: %d/%d\n", 2074 tpg->hsv_enc, tpg->real_hsv_enc); 2075 else if (tpg->color_enc == TGP_COLOR_ENC_YCBCR) 2076 pr_info("tpg Y'CbCr encoding: %d/%d\n", 2077 tpg->ycbcr_enc, tpg->real_ycbcr_enc); 2078 pr_info("tpg quantization: %d/%d\n", tpg->quantization, tpg->real_quantization); 2079 pr_info("tpg RGB range: %d/%d\n", tpg->rgb_range, tpg->real_rgb_range); 2080 } 2081 EXPORT_SYMBOL_GPL(tpg_log_status); 2082 2083 /* 2084 * This struct contains common parameters used by both the drawing of the 2085 * test pattern and the drawing of the extras (borders, square, etc.) 2086 */ 2087 struct tpg_draw_params { 2088 /* common data */ 2089 bool is_tv; 2090 bool is_60hz; 2091 unsigned twopixsize; 2092 unsigned img_width; 2093 unsigned stride; 2094 unsigned hmax; 2095 unsigned frame_line; 2096 unsigned frame_line_next; 2097 2098 /* test pattern */ 2099 unsigned mv_hor_old; 2100 unsigned mv_hor_new; 2101 unsigned mv_vert_old; 2102 unsigned mv_vert_new; 2103 2104 /* extras */ 2105 unsigned wss_width; 2106 unsigned wss_random_offset; 2107 unsigned sav_eav_f; 2108 unsigned left_pillar_width; 2109 unsigned right_pillar_start; 2110 }; 2111 2112 static void tpg_fill_params_pattern(const struct tpg_data *tpg, unsigned p, 2113 struct tpg_draw_params *params) 2114 { 2115 params->mv_hor_old = 2116 tpg_hscale_div(tpg, p, tpg->mv_hor_count % tpg->src_width); 2117 params->mv_hor_new = 2118 tpg_hscale_div(tpg, p, (tpg->mv_hor_count + tpg->mv_hor_step) % 2119 tpg->src_width); 2120 params->mv_vert_old = tpg->mv_vert_count % tpg->src_height; 2121 params->mv_vert_new = 2122 (tpg->mv_vert_count + tpg->mv_vert_step) % tpg->src_height; 2123 } 2124 2125 static void tpg_fill_params_extras(const struct tpg_data *tpg, 2126 unsigned p, 2127 struct tpg_draw_params *params) 2128 { 2129 unsigned left_pillar_width = 0; 2130 unsigned right_pillar_start = params->img_width; 2131 2132 params->wss_width = tpg->crop.left < tpg->src_width / 2 ? 2133 tpg->src_width / 2 - tpg->crop.left : 0; 2134 if (params->wss_width > tpg->crop.width) 2135 params->wss_width = tpg->crop.width; 2136 params->wss_width = tpg_hscale_div(tpg, p, params->wss_width); 2137 params->wss_random_offset = 2138 params->twopixsize * prandom_u32_max(tpg->src_width / 2); 2139 2140 if (tpg->crop.left < tpg->border.left) { 2141 left_pillar_width = tpg->border.left - tpg->crop.left; 2142 if (left_pillar_width > tpg->crop.width) 2143 left_pillar_width = tpg->crop.width; 2144 left_pillar_width = tpg_hscale_div(tpg, p, left_pillar_width); 2145 } 2146 params->left_pillar_width = left_pillar_width; 2147 2148 if (tpg->crop.left + tpg->crop.width > 2149 tpg->border.left + tpg->border.width) { 2150 right_pillar_start = 2151 tpg->border.left + tpg->border.width - tpg->crop.left; 2152 right_pillar_start = 2153 tpg_hscale_div(tpg, p, right_pillar_start); 2154 if (right_pillar_start > params->img_width) 2155 right_pillar_start = params->img_width; 2156 } 2157 params->right_pillar_start = right_pillar_start; 2158 2159 params->sav_eav_f = tpg->field == 2160 (params->is_60hz ? V4L2_FIELD_TOP : V4L2_FIELD_BOTTOM); 2161 } 2162 2163 static void tpg_fill_plane_extras(const struct tpg_data *tpg, 2164 const struct tpg_draw_params *params, 2165 unsigned p, unsigned h, u8 *vbuf) 2166 { 2167 unsigned twopixsize = params->twopixsize; 2168 unsigned img_width = params->img_width; 2169 unsigned frame_line = params->frame_line; 2170 const struct v4l2_rect *sq = &tpg->square; 2171 const struct v4l2_rect *b = &tpg->border; 2172 const struct v4l2_rect *c = &tpg->crop; 2173 2174 if (params->is_tv && !params->is_60hz && 2175 frame_line == 0 && params->wss_width) { 2176 /* 2177 * Replace the first half of the top line of a 50 Hz frame 2178 * with random data to simulate a WSS signal. 2179 */ 2180 u8 *wss = tpg->random_line[p] + params->wss_random_offset; 2181 2182 memcpy(vbuf, wss, params->wss_width); 2183 } 2184 2185 if (tpg->show_border && frame_line >= b->top && 2186 frame_line < b->top + b->height) { 2187 unsigned bottom = b->top + b->height - 1; 2188 unsigned left = params->left_pillar_width; 2189 unsigned right = params->right_pillar_start; 2190 2191 if (frame_line == b->top || frame_line == b->top + 1 || 2192 frame_line == bottom || frame_line == bottom - 1) { 2193 memcpy(vbuf + left, tpg->contrast_line[p], 2194 right - left); 2195 } else { 2196 if (b->left >= c->left && 2197 b->left < c->left + c->width) 2198 memcpy(vbuf + left, 2199 tpg->contrast_line[p], twopixsize); 2200 if (b->left + b->width > c->left && 2201 b->left + b->width <= c->left + c->width) 2202 memcpy(vbuf + right - twopixsize, 2203 tpg->contrast_line[p], twopixsize); 2204 } 2205 } 2206 if (tpg->qual != TPG_QUAL_NOISE && frame_line >= b->top && 2207 frame_line < b->top + b->height) { 2208 memcpy(vbuf, tpg->black_line[p], params->left_pillar_width); 2209 memcpy(vbuf + params->right_pillar_start, tpg->black_line[p], 2210 img_width - params->right_pillar_start); 2211 } 2212 if (tpg->show_square && frame_line >= sq->top && 2213 frame_line < sq->top + sq->height && 2214 sq->left < c->left + c->width && 2215 sq->left + sq->width >= c->left) { 2216 unsigned left = sq->left; 2217 unsigned width = sq->width; 2218 2219 if (c->left > left) { 2220 width -= c->left - left; 2221 left = c->left; 2222 } 2223 if (c->left + c->width < left + width) 2224 width -= left + width - c->left - c->width; 2225 left -= c->left; 2226 left = tpg_hscale_div(tpg, p, left); 2227 width = tpg_hscale_div(tpg, p, width); 2228 memcpy(vbuf + left, tpg->contrast_line[p], width); 2229 } 2230 if (tpg->insert_sav) { 2231 unsigned offset = tpg_hdiv(tpg, p, tpg->compose.width / 3); 2232 u8 *p = vbuf + offset; 2233 unsigned vact = 0, hact = 0; 2234 2235 p[0] = 0xff; 2236 p[1] = 0; 2237 p[2] = 0; 2238 p[3] = 0x80 | (params->sav_eav_f << 6) | 2239 (vact << 5) | (hact << 4) | 2240 ((hact ^ vact) << 3) | 2241 ((hact ^ params->sav_eav_f) << 2) | 2242 ((params->sav_eav_f ^ vact) << 1) | 2243 (hact ^ vact ^ params->sav_eav_f); 2244 } 2245 if (tpg->insert_eav) { 2246 unsigned offset = tpg_hdiv(tpg, p, tpg->compose.width * 2 / 3); 2247 u8 *p = vbuf + offset; 2248 unsigned vact = 0, hact = 1; 2249 2250 p[0] = 0xff; 2251 p[1] = 0; 2252 p[2] = 0; 2253 p[3] = 0x80 | (params->sav_eav_f << 6) | 2254 (vact << 5) | (hact << 4) | 2255 ((hact ^ vact) << 3) | 2256 ((hact ^ params->sav_eav_f) << 2) | 2257 ((params->sav_eav_f ^ vact) << 1) | 2258 (hact ^ vact ^ params->sav_eav_f); 2259 } 2260 } 2261 2262 static void tpg_fill_plane_pattern(const struct tpg_data *tpg, 2263 const struct tpg_draw_params *params, 2264 unsigned p, unsigned h, u8 *vbuf) 2265 { 2266 unsigned twopixsize = params->twopixsize; 2267 unsigned img_width = params->img_width; 2268 unsigned mv_hor_old = params->mv_hor_old; 2269 unsigned mv_hor_new = params->mv_hor_new; 2270 unsigned mv_vert_old = params->mv_vert_old; 2271 unsigned mv_vert_new = params->mv_vert_new; 2272 unsigned frame_line = params->frame_line; 2273 unsigned frame_line_next = params->frame_line_next; 2274 unsigned line_offset = tpg_hscale_div(tpg, p, tpg->crop.left); 2275 bool even; 2276 bool fill_blank = false; 2277 unsigned pat_line_old; 2278 unsigned pat_line_new; 2279 u8 *linestart_older; 2280 u8 *linestart_newer; 2281 u8 *linestart_top; 2282 u8 *linestart_bottom; 2283 2284 even = !(frame_line & 1); 2285 2286 if (h >= params->hmax) { 2287 if (params->hmax == tpg->compose.height) 2288 return; 2289 if (!tpg->perc_fill_blank) 2290 return; 2291 fill_blank = true; 2292 } 2293 2294 if (tpg->vflip) { 2295 frame_line = tpg->src_height - frame_line - 1; 2296 frame_line_next = tpg->src_height - frame_line_next - 1; 2297 } 2298 2299 if (fill_blank) { 2300 linestart_older = tpg->contrast_line[p]; 2301 linestart_newer = tpg->contrast_line[p]; 2302 } else if (tpg->qual != TPG_QUAL_NOISE && 2303 (frame_line < tpg->border.top || 2304 frame_line >= tpg->border.top + tpg->border.height)) { 2305 linestart_older = tpg->black_line[p]; 2306 linestart_newer = tpg->black_line[p]; 2307 } else if (tpg->pattern == TPG_PAT_NOISE || tpg->qual == TPG_QUAL_NOISE) { 2308 linestart_older = tpg->random_line[p] + 2309 twopixsize * prandom_u32_max(tpg->src_width / 2); 2310 linestart_newer = tpg->random_line[p] + 2311 twopixsize * prandom_u32_max(tpg->src_width / 2); 2312 } else { 2313 unsigned frame_line_old = 2314 (frame_line + mv_vert_old) % tpg->src_height; 2315 unsigned frame_line_new = 2316 (frame_line + mv_vert_new) % tpg->src_height; 2317 unsigned pat_line_next_old; 2318 unsigned pat_line_next_new; 2319 2320 pat_line_old = tpg_get_pat_line(tpg, frame_line_old); 2321 pat_line_new = tpg_get_pat_line(tpg, frame_line_new); 2322 linestart_older = tpg->lines[pat_line_old][p] + mv_hor_old; 2323 linestart_newer = tpg->lines[pat_line_new][p] + mv_hor_new; 2324 2325 if (tpg->vdownsampling[p] > 1 && frame_line != frame_line_next) { 2326 int avg_pat; 2327 2328 /* 2329 * Now decide whether we need to use downsampled_lines[]. 2330 * That's necessary if the two lines use different patterns. 2331 */ 2332 pat_line_next_old = tpg_get_pat_line(tpg, 2333 (frame_line_next + mv_vert_old) % tpg->src_height); 2334 pat_line_next_new = tpg_get_pat_line(tpg, 2335 (frame_line_next + mv_vert_new) % tpg->src_height); 2336 2337 switch (tpg->field) { 2338 case V4L2_FIELD_INTERLACED: 2339 case V4L2_FIELD_INTERLACED_BT: 2340 case V4L2_FIELD_INTERLACED_TB: 2341 avg_pat = tpg_pattern_avg(tpg, pat_line_old, pat_line_new); 2342 if (avg_pat < 0) 2343 break; 2344 linestart_older = tpg->downsampled_lines[avg_pat][p] + mv_hor_old; 2345 linestart_newer = linestart_older; 2346 break; 2347 case V4L2_FIELD_NONE: 2348 case V4L2_FIELD_TOP: 2349 case V4L2_FIELD_BOTTOM: 2350 case V4L2_FIELD_SEQ_BT: 2351 case V4L2_FIELD_SEQ_TB: 2352 avg_pat = tpg_pattern_avg(tpg, pat_line_old, pat_line_next_old); 2353 if (avg_pat >= 0) 2354 linestart_older = tpg->downsampled_lines[avg_pat][p] + 2355 mv_hor_old; 2356 avg_pat = tpg_pattern_avg(tpg, pat_line_new, pat_line_next_new); 2357 if (avg_pat >= 0) 2358 linestart_newer = tpg->downsampled_lines[avg_pat][p] + 2359 mv_hor_new; 2360 break; 2361 } 2362 } 2363 linestart_older += line_offset; 2364 linestart_newer += line_offset; 2365 } 2366 if (tpg->field_alternate) { 2367 linestart_top = linestart_bottom = linestart_older; 2368 } else if (params->is_60hz) { 2369 linestart_top = linestart_newer; 2370 linestart_bottom = linestart_older; 2371 } else { 2372 linestart_top = linestart_older; 2373 linestart_bottom = linestart_newer; 2374 } 2375 2376 switch (tpg->field) { 2377 case V4L2_FIELD_INTERLACED: 2378 case V4L2_FIELD_INTERLACED_TB: 2379 case V4L2_FIELD_SEQ_TB: 2380 case V4L2_FIELD_SEQ_BT: 2381 if (even) 2382 memcpy(vbuf, linestart_top, img_width); 2383 else 2384 memcpy(vbuf, linestart_bottom, img_width); 2385 break; 2386 case V4L2_FIELD_INTERLACED_BT: 2387 if (even) 2388 memcpy(vbuf, linestart_bottom, img_width); 2389 else 2390 memcpy(vbuf, linestart_top, img_width); 2391 break; 2392 case V4L2_FIELD_TOP: 2393 memcpy(vbuf, linestart_top, img_width); 2394 break; 2395 case V4L2_FIELD_BOTTOM: 2396 memcpy(vbuf, linestart_bottom, img_width); 2397 break; 2398 case V4L2_FIELD_NONE: 2399 default: 2400 memcpy(vbuf, linestart_older, img_width); 2401 break; 2402 } 2403 } 2404 2405 void tpg_fill_plane_buffer(struct tpg_data *tpg, v4l2_std_id std, 2406 unsigned p, u8 *vbuf) 2407 { 2408 struct tpg_draw_params params; 2409 unsigned factor = V4L2_FIELD_HAS_T_OR_B(tpg->field) ? 2 : 1; 2410 2411 /* Coarse scaling with Bresenham */ 2412 unsigned int_part = (tpg->crop.height / factor) / tpg->compose.height; 2413 unsigned fract_part = (tpg->crop.height / factor) % tpg->compose.height; 2414 unsigned src_y = 0; 2415 unsigned error = 0; 2416 unsigned h; 2417 2418 tpg_recalc(tpg); 2419 2420 params.is_tv = std; 2421 params.is_60hz = std & V4L2_STD_525_60; 2422 params.twopixsize = tpg->twopixelsize[p]; 2423 params.img_width = tpg_hdiv(tpg, p, tpg->compose.width); 2424 params.stride = tpg->bytesperline[p]; 2425 params.hmax = (tpg->compose.height * tpg->perc_fill) / 100; 2426 2427 tpg_fill_params_pattern(tpg, p, ¶ms); 2428 tpg_fill_params_extras(tpg, p, ¶ms); 2429 2430 vbuf += tpg_hdiv(tpg, p, tpg->compose.left); 2431 2432 for (h = 0; h < tpg->compose.height; h++) { 2433 unsigned buf_line; 2434 2435 params.frame_line = tpg_calc_frameline(tpg, src_y, tpg->field); 2436 params.frame_line_next = params.frame_line; 2437 buf_line = tpg_calc_buffer_line(tpg, h, tpg->field); 2438 src_y += int_part; 2439 error += fract_part; 2440 if (error >= tpg->compose.height) { 2441 error -= tpg->compose.height; 2442 src_y++; 2443 } 2444 2445 /* 2446 * For line-interleaved formats determine the 'plane' 2447 * based on the buffer line. 2448 */ 2449 if (tpg_g_interleaved(tpg)) 2450 p = tpg_g_interleaved_plane(tpg, buf_line); 2451 2452 if (tpg->vdownsampling[p] > 1) { 2453 /* 2454 * When doing vertical downsampling the field setting 2455 * matters: for SEQ_BT/TB we downsample each field 2456 * separately (i.e. lines 0+2 are combined, as are 2457 * lines 1+3), for the other field settings we combine 2458 * odd and even lines. Doing that for SEQ_BT/TB would 2459 * be really weird. 2460 */ 2461 if (tpg->field == V4L2_FIELD_SEQ_BT || 2462 tpg->field == V4L2_FIELD_SEQ_TB) { 2463 unsigned next_src_y = src_y; 2464 2465 if ((h & 3) >= 2) 2466 continue; 2467 next_src_y += int_part; 2468 if (error + fract_part >= tpg->compose.height) 2469 next_src_y++; 2470 params.frame_line_next = 2471 tpg_calc_frameline(tpg, next_src_y, tpg->field); 2472 } else { 2473 if (h & 1) 2474 continue; 2475 params.frame_line_next = 2476 tpg_calc_frameline(tpg, src_y, tpg->field); 2477 } 2478 2479 buf_line /= tpg->vdownsampling[p]; 2480 } 2481 tpg_fill_plane_pattern(tpg, ¶ms, p, h, 2482 vbuf + buf_line * params.stride); 2483 tpg_fill_plane_extras(tpg, ¶ms, p, h, 2484 vbuf + buf_line * params.stride); 2485 } 2486 } 2487 EXPORT_SYMBOL_GPL(tpg_fill_plane_buffer); 2488 2489 void tpg_fillbuffer(struct tpg_data *tpg, v4l2_std_id std, unsigned p, u8 *vbuf) 2490 { 2491 unsigned offset = 0; 2492 unsigned i; 2493 2494 if (tpg->buffers > 1) { 2495 tpg_fill_plane_buffer(tpg, std, p, vbuf); 2496 return; 2497 } 2498 2499 for (i = 0; i < tpg_g_planes(tpg); i++) { 2500 tpg_fill_plane_buffer(tpg, std, i, vbuf + offset); 2501 offset += tpg_calc_plane_size(tpg, i); 2502 } 2503 } 2504 EXPORT_SYMBOL_GPL(tpg_fillbuffer); 2505 2506 MODULE_DESCRIPTION("V4L2 Test Pattern Generator"); 2507 MODULE_AUTHOR("Hans Verkuil"); 2508 MODULE_LICENSE("GPL"); 2509