Blame view

include/media/v4l2-tpg.h 14.1 KB
63881df94   Hans Verkuil   [media] vivid: ad...
1
  /*
e07d46e7e   Helen Mae Koike Fornazier   [media] tpg: Expo...
2
   * v4l2-tpg.h - Test Pattern Generator
63881df94   Hans Verkuil   [media] vivid: ad...
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
   *
   * Copyright 2014 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
   *
   * This program is free software; you may redistribute it and/or modify
   * it under the terms of the GNU General Public License as published by
   * the Free Software Foundation; version 2 of the License.
   *
   * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
   * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
   * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
   * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
   * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
   * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
   * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
   * SOFTWARE.
   */
e07d46e7e   Helen Mae Koike Fornazier   [media] tpg: Expo...
19
20
  #ifndef _V4L2_TPG_H_
  #define _V4L2_TPG_H_
63881df94   Hans Verkuil   [media] vivid: ad...
21

63881df94   Hans Verkuil   [media] vivid: ad...
22
23
24
25
  #include <linux/types.h>
  #include <linux/errno.h>
  #include <linux/random.h>
  #include <linux/slab.h>
5754d0d58   Hans Verkuil   [media] vivid: ad...
26
  #include <linux/vmalloc.h>
63881df94   Hans Verkuil   [media] vivid: ad...
27
  #include <linux/videodev2.h>
e07d46e7e   Helen Mae Koike Fornazier   [media] tpg: Expo...
28
  #include <media/v4l2-tpg-colors.h>
63881df94   Hans Verkuil   [media] vivid: ad...
29
30
31
32
33
34
35
36
37
38
39
40
41
  
  enum tpg_pattern {
  	TPG_PAT_75_COLORBAR,
  	TPG_PAT_100_COLORBAR,
  	TPG_PAT_CSC_COLORBAR,
  	TPG_PAT_100_HCOLORBAR,
  	TPG_PAT_100_COLORSQUARES,
  	TPG_PAT_BLACK,
  	TPG_PAT_WHITE,
  	TPG_PAT_RED,
  	TPG_PAT_GREEN,
  	TPG_PAT_BLUE,
  	TPG_PAT_CHECKERS_16X16,
1a05d3133   Hans Verkuil   [media] vivid: ad...
42
  	TPG_PAT_CHECKERS_2X2,
63881df94   Hans Verkuil   [media] vivid: ad...
43
  	TPG_PAT_CHECKERS_1X1,
1a05d3133   Hans Verkuil   [media] vivid: ad...
44
45
  	TPG_PAT_COLOR_CHECKERS_2X2,
  	TPG_PAT_COLOR_CHECKERS_1X1,
63881df94   Hans Verkuil   [media] vivid: ad...
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
  	TPG_PAT_ALTERNATING_HLINES,
  	TPG_PAT_ALTERNATING_VLINES,
  	TPG_PAT_CROSS_1_PIXEL,
  	TPG_PAT_CROSS_2_PIXELS,
  	TPG_PAT_CROSS_10_PIXELS,
  	TPG_PAT_GRAY_RAMP,
  
  	/* Must be the last pattern */
  	TPG_PAT_NOISE,
  };
  
  extern const char * const tpg_pattern_strings[];
  
  enum tpg_quality {
  	TPG_QUAL_COLOR,
  	TPG_QUAL_GRAY,
  	TPG_QUAL_NOISE
  };
  
  enum tpg_video_aspect {
  	TPG_VIDEO_ASPECT_IMAGE,
  	TPG_VIDEO_ASPECT_4X3,
  	TPG_VIDEO_ASPECT_14X9_CENTRE,
  	TPG_VIDEO_ASPECT_16X9_CENTRE,
  	TPG_VIDEO_ASPECT_16X9_ANAMORPHIC,
  };
  
  enum tpg_pixel_aspect {
  	TPG_PIXEL_ASPECT_SQUARE,
  	TPG_PIXEL_ASPECT_NTSC,
  	TPG_PIXEL_ASPECT_PAL,
  };
  
  enum tpg_move_mode {
  	TPG_MOVE_NEG_FAST,
  	TPG_MOVE_NEG,
  	TPG_MOVE_NEG_SLOW,
  	TPG_MOVE_NONE,
  	TPG_MOVE_POS_SLOW,
  	TPG_MOVE_POS,
  	TPG_MOVE_POS_FAST,
  };
  
  extern const char * const tpg_aspect_strings[];
ba01f673e   Hans Verkuil   [media] vivid-tpg...
90
  #define TPG_MAX_PLANES 3
63881df94   Hans Verkuil   [media] vivid: ad...
91
92
93
94
95
96
97
98
99
100
  #define TPG_MAX_PAT_LINES 8
  
  struct tpg_data {
  	/* Source frame size */
  	unsigned			src_width, src_height;
  	/* Buffer height */
  	unsigned			buf_height;
  	/* Scaled output frame size */
  	unsigned			scaled_width;
  	u32				field;
43047f6b7   Hans Verkuil   [media] vivid: fi...
101
  	bool				field_alternate;
63881df94   Hans Verkuil   [media] vivid: ad...
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
  	/* crop coordinates are frame-based */
  	struct v4l2_rect		crop;
  	/* compose coordinates are format-based */
  	struct v4l2_rect		compose;
  	/* border and square coordinates are frame-based */
  	struct v4l2_rect		border;
  	struct v4l2_rect		square;
  
  	/* Color-related fields */
  	enum tpg_quality		qual;
  	unsigned			qual_offset;
  	u8				alpha_component;
  	bool				alpha_red_only;
  	u8				brightness;
  	u8				contrast;
  	u8				saturation;
  	s16				hue;
  	u32				fourcc;
  	bool				is_yuv;
  	u32				colorspace;
ca5316db0   Hans Verkuil   [media] vivid: ad...
122
  	u32				xfer_func;
481b97a1f   Hans Verkuil   [media] vivid-tpg...
123
124
  	u32				ycbcr_enc;
  	/*
ca5316db0   Hans Verkuil   [media] vivid: ad...
125
126
127
128
129
  	 * Stores the actual transfer function, i.e. will never be
  	 * V4L2_XFER_FUNC_DEFAULT.
  	 */
  	u32				real_xfer_func;
  	/*
481b97a1f   Hans Verkuil   [media] vivid-tpg...
130
131
132
133
134
135
136
137
138
139
  	 * Stores the actual Y'CbCr encoding, i.e. will never be
  	 * V4L2_YCBCR_ENC_DEFAULT.
  	 */
  	u32				real_ycbcr_enc;
  	u32				quantization;
  	/*
  	 * Stores the actual quantization, i.e. will never be
  	 * V4L2_QUANTIZATION_DEFAULT.
  	 */
  	u32				real_quantization;
63881df94   Hans Verkuil   [media] vivid: ad...
140
141
142
143
  	enum tpg_video_aspect		vid_aspect;
  	enum tpg_pixel_aspect		pix_aspect;
  	unsigned			rgb_range;
  	unsigned			real_rgb_range;
06d1f0c2e   Hans Verkuil   [media] vivid-tpg...
144
  	unsigned			buffers;
63881df94   Hans Verkuil   [media] vivid: ad...
145
  	unsigned			planes;
02aa769d9   Hans Verkuil   [media] vivid: ad...
146
  	bool				interleaved;
ba01f673e   Hans Verkuil   [media] vivid-tpg...
147
148
  	u8				vdownsampling[TPG_MAX_PLANES];
  	u8				hdownsampling[TPG_MAX_PLANES];
9991deff1   Hans Verkuil   [media] vivid-tpg...
149
150
151
152
153
  	/*
  	 * horizontal positions must be ANDed with this value to enforce
  	 * correct boundaries for packed YUYV values.
  	 */
  	unsigned			hmask[TPG_MAX_PLANES];
63881df94   Hans Verkuil   [media] vivid: ad...
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
  	/* Used to store the colors in native format, either RGB or YUV */
  	u8				colors[TPG_COLOR_MAX][3];
  	u8				textfg[TPG_MAX_PLANES][8], textbg[TPG_MAX_PLANES][8];
  	/* size in bytes for two pixels in each plane */
  	unsigned			twopixelsize[TPG_MAX_PLANES];
  	unsigned			bytesperline[TPG_MAX_PLANES];
  
  	/* Configuration */
  	enum tpg_pattern		pattern;
  	bool				hflip;
  	bool				vflip;
  	unsigned			perc_fill;
  	bool				perc_fill_blank;
  	bool				show_border;
  	bool				show_square;
  	bool				insert_sav;
  	bool				insert_eav;
  
  	/* Test pattern movement */
  	enum tpg_move_mode		mv_hor_mode;
  	int				mv_hor_count;
  	int				mv_hor_step;
  	enum tpg_move_mode		mv_vert_mode;
  	int				mv_vert_count;
  	int				mv_vert_step;
  
  	bool				recalc_colors;
  	bool				recalc_lines;
  	bool				recalc_square_border;
  
  	/* Used to store TPG_MAX_PAT_LINES lines, each with up to two planes */
  	unsigned			max_line_width;
  	u8				*lines[TPG_MAX_PAT_LINES][TPG_MAX_PLANES];
5d7c539e1   Hans Verkuil   [media] vivid-tpg...
187
  	u8				*downsampled_lines[TPG_MAX_PAT_LINES][TPG_MAX_PLANES];
63881df94   Hans Verkuil   [media] vivid: ad...
188
189
190
191
192
193
194
195
196
197
  	u8				*random_line[TPG_MAX_PLANES];
  	u8				*contrast_line[TPG_MAX_PLANES];
  	u8				*black_line[TPG_MAX_PLANES];
  };
  
  void tpg_init(struct tpg_data *tpg, unsigned w, unsigned h);
  int tpg_alloc(struct tpg_data *tpg, unsigned max_w);
  void tpg_free(struct tpg_data *tpg);
  void tpg_reset_source(struct tpg_data *tpg, unsigned width, unsigned height,
  		       u32 field);
84b76d749   Hans Verkuil   [media] vivid-tpg...
198
  void tpg_log_status(struct tpg_data *tpg);
63881df94   Hans Verkuil   [media] vivid: ad...
199
200
  
  void tpg_set_font(const u8 *f);
dfff04895   Hans Verkuil   [media] vivid-tpg...
201
  void tpg_gen_text(const struct tpg_data *tpg,
63881df94   Hans Verkuil   [media] vivid: ad...
202
203
204
  		u8 *basep[TPG_MAX_PLANES][2], int y, int x, char *text);
  void tpg_calc_text_basep(struct tpg_data *tpg,
  		u8 *basep[TPG_MAX_PLANES][2], unsigned p, u8 *vbuf);
02aa769d9   Hans Verkuil   [media] vivid: ad...
205
  unsigned tpg_g_interleaved_plane(const struct tpg_data *tpg, unsigned buf_line);
dfff04895   Hans Verkuil   [media] vivid-tpg...
206
207
208
209
  void tpg_fill_plane_buffer(struct tpg_data *tpg, v4l2_std_id std,
  			   unsigned p, u8 *vbuf);
  void tpg_fillbuffer(struct tpg_data *tpg, v4l2_std_id std,
  		    unsigned p, u8 *vbuf);
63881df94   Hans Verkuil   [media] vivid: ad...
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
  bool tpg_s_fourcc(struct tpg_data *tpg, u32 fourcc);
  void tpg_s_crop_compose(struct tpg_data *tpg, const struct v4l2_rect *crop,
  		const struct v4l2_rect *compose);
  
  static inline void tpg_s_pattern(struct tpg_data *tpg, enum tpg_pattern pattern)
  {
  	if (tpg->pattern == pattern)
  		return;
  	tpg->pattern = pattern;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_quality(struct tpg_data *tpg,
  				    enum tpg_quality qual, unsigned qual_offset)
  {
  	if (tpg->qual == qual && tpg->qual_offset == qual_offset)
  		return;
  	tpg->qual = qual;
  	tpg->qual_offset = qual_offset;
  	tpg->recalc_colors = true;
  }
  
  static inline enum tpg_quality tpg_g_quality(const struct tpg_data *tpg)
  {
  	return tpg->qual;
  }
  
  static inline void tpg_s_alpha_component(struct tpg_data *tpg,
  					    u8 alpha_component)
  {
  	if (tpg->alpha_component == alpha_component)
  		return;
  	tpg->alpha_component = alpha_component;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_alpha_mode(struct tpg_data *tpg,
  					    bool red_only)
  {
  	if (tpg->alpha_red_only == red_only)
  		return;
  	tpg->alpha_red_only = red_only;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_brightness(struct tpg_data *tpg,
  					u8 brightness)
  {
  	if (tpg->brightness == brightness)
  		return;
  	tpg->brightness = brightness;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_contrast(struct tpg_data *tpg,
  					u8 contrast)
  {
  	if (tpg->contrast == contrast)
  		return;
  	tpg->contrast = contrast;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_saturation(struct tpg_data *tpg,
  					u8 saturation)
  {
  	if (tpg->saturation == saturation)
  		return;
  	tpg->saturation = saturation;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_hue(struct tpg_data *tpg,
  					s16 hue)
  {
  	if (tpg->hue == hue)
  		return;
  	tpg->hue = hue;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_rgb_range(struct tpg_data *tpg,
  					unsigned rgb_range)
  {
  	if (tpg->rgb_range == rgb_range)
  		return;
  	tpg->rgb_range = rgb_range;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_real_rgb_range(struct tpg_data *tpg,
  					unsigned rgb_range)
  {
  	if (tpg->real_rgb_range == rgb_range)
  		return;
  	tpg->real_rgb_range = rgb_range;
  	tpg->recalc_colors = true;
  }
  
  static inline void tpg_s_colorspace(struct tpg_data *tpg, u32 colorspace)
  {
  	if (tpg->colorspace == colorspace)
  		return;
  	tpg->colorspace = colorspace;
  	tpg->recalc_colors = true;
  }
  
  static inline u32 tpg_g_colorspace(const struct tpg_data *tpg)
  {
  	return tpg->colorspace;
  }
481b97a1f   Hans Verkuil   [media] vivid-tpg...
321
322
323
324
325
326
327
328
329
330
331
332
  static inline void tpg_s_ycbcr_enc(struct tpg_data *tpg, u32 ycbcr_enc)
  {
  	if (tpg->ycbcr_enc == ycbcr_enc)
  		return;
  	tpg->ycbcr_enc = ycbcr_enc;
  	tpg->recalc_colors = true;
  }
  
  static inline u32 tpg_g_ycbcr_enc(const struct tpg_data *tpg)
  {
  	return tpg->ycbcr_enc;
  }
ca5316db0   Hans Verkuil   [media] vivid: ad...
333
334
335
336
337
338
339
340
341
342
343
344
  static inline void tpg_s_xfer_func(struct tpg_data *tpg, u32 xfer_func)
  {
  	if (tpg->xfer_func == xfer_func)
  		return;
  	tpg->xfer_func = xfer_func;
  	tpg->recalc_colors = true;
  }
  
  static inline u32 tpg_g_xfer_func(const struct tpg_data *tpg)
  {
  	return tpg->xfer_func;
  }
481b97a1f   Hans Verkuil   [media] vivid-tpg...
345
346
347
348
349
350
351
352
353
354
355
356
  static inline void tpg_s_quantization(struct tpg_data *tpg, u32 quantization)
  {
  	if (tpg->quantization == quantization)
  		return;
  	tpg->quantization = quantization;
  	tpg->recalc_colors = true;
  }
  
  static inline u32 tpg_g_quantization(const struct tpg_data *tpg)
  {
  	return tpg->quantization;
  }
06d1f0c2e   Hans Verkuil   [media] vivid-tpg...
357
358
359
360
  static inline unsigned tpg_g_buffers(const struct tpg_data *tpg)
  {
  	return tpg->buffers;
  }
63881df94   Hans Verkuil   [media] vivid: ad...
361
362
  static inline unsigned tpg_g_planes(const struct tpg_data *tpg)
  {
02aa769d9   Hans Verkuil   [media] vivid: ad...
363
364
365
366
367
368
  	return tpg->interleaved ? 1 : tpg->planes;
  }
  
  static inline bool tpg_g_interleaved(const struct tpg_data *tpg)
  {
  	return tpg->interleaved;
63881df94   Hans Verkuil   [media] vivid: ad...
369
370
371
372
373
374
  }
  
  static inline unsigned tpg_g_twopixelsize(const struct tpg_data *tpg, unsigned plane)
  {
  	return tpg->twopixelsize[plane];
  }
9991deff1   Hans Verkuil   [media] vivid-tpg...
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
  static inline unsigned tpg_hdiv(const struct tpg_data *tpg,
  				  unsigned plane, unsigned x)
  {
  	return ((x / tpg->hdownsampling[plane]) & tpg->hmask[plane]) *
  		tpg->twopixelsize[plane] / 2;
  }
  
  static inline unsigned tpg_hscale(const struct tpg_data *tpg, unsigned x)
  {
  	return (x * tpg->scaled_width) / tpg->src_width;
  }
  
  static inline unsigned tpg_hscale_div(const struct tpg_data *tpg,
  				      unsigned plane, unsigned x)
  {
  	return tpg_hdiv(tpg, plane, tpg_hscale(tpg, x));
  }
63881df94   Hans Verkuil   [media] vivid: ad...
392
393
394
395
396
397
398
  static inline unsigned tpg_g_bytesperline(const struct tpg_data *tpg, unsigned plane)
  {
  	return tpg->bytesperline[plane];
  }
  
  static inline void tpg_s_bytesperline(struct tpg_data *tpg, unsigned plane, unsigned bpl)
  {
4db220418   Hans Verkuil   [media] vivid-tpg...
399
400
401
402
403
404
  	unsigned p;
  
  	if (tpg->buffers > 1) {
  		tpg->bytesperline[plane] = bpl;
  		return;
  	}
02aa769d9   Hans Verkuil   [media] vivid: ad...
405
  	for (p = 0; p < tpg_g_planes(tpg); p++) {
4db220418   Hans Verkuil   [media] vivid-tpg...
406
  		unsigned plane_w = bpl * tpg->twopixelsize[p] / tpg->twopixelsize[0];
ba01f673e   Hans Verkuil   [media] vivid-tpg...
407
  		tpg->bytesperline[p] = plane_w / tpg->hdownsampling[p];
4db220418   Hans Verkuil   [media] vivid-tpg...
408
  	}
3541e349e   Hans Verkuil   [media] vivid: fi...
409
410
  	if (tpg_g_interleaved(tpg))
  		tpg->bytesperline[1] = tpg->bytesperline[0];
4db220418   Hans Verkuil   [media] vivid-tpg...
411
  }
ba01f673e   Hans Verkuil   [media] vivid-tpg...
412

4db220418   Hans Verkuil   [media] vivid-tpg...
413
414
415
416
417
418
419
  static inline unsigned tpg_g_line_width(const struct tpg_data *tpg, unsigned plane)
  {
  	unsigned w = 0;
  	unsigned p;
  
  	if (tpg->buffers > 1)
  		return tpg_g_bytesperline(tpg, plane);
02aa769d9   Hans Verkuil   [media] vivid: ad...
420
  	for (p = 0; p < tpg_g_planes(tpg); p++) {
4db220418   Hans Verkuil   [media] vivid-tpg...
421
  		unsigned plane_w = tpg_g_bytesperline(tpg, p);
ba01f673e   Hans Verkuil   [media] vivid-tpg...
422
  		w += plane_w / tpg->vdownsampling[p];
4db220418   Hans Verkuil   [media] vivid-tpg...
423
424
425
426
427
428
429
430
431
432
433
434
  	}
  	return w;
  }
  
  static inline unsigned tpg_calc_line_width(const struct tpg_data *tpg,
  					   unsigned plane, unsigned bpl)
  {
  	unsigned w = 0;
  	unsigned p;
  
  	if (tpg->buffers > 1)
  		return bpl;
02aa769d9   Hans Verkuil   [media] vivid: ad...
435
  	for (p = 0; p < tpg_g_planes(tpg); p++) {
4db220418   Hans Verkuil   [media] vivid-tpg...
436
  		unsigned plane_w = bpl * tpg->twopixelsize[p] / tpg->twopixelsize[0];
ba01f673e   Hans Verkuil   [media] vivid-tpg...
437
438
  		plane_w /= tpg->hdownsampling[p];
  		w += plane_w / tpg->vdownsampling[p];
4db220418   Hans Verkuil   [media] vivid-tpg...
439
440
441
442
443
444
  	}
  	return w;
  }
  
  static inline unsigned tpg_calc_plane_size(const struct tpg_data *tpg, unsigned plane)
  {
02aa769d9   Hans Verkuil   [media] vivid: ad...
445
  	if (plane >= tpg_g_planes(tpg))
4db220418   Hans Verkuil   [media] vivid-tpg...
446
  		return 0;
ba01f673e   Hans Verkuil   [media] vivid-tpg...
447
448
  	return tpg_g_bytesperline(tpg, plane) * tpg->buf_height /
  	       tpg->vdownsampling[plane];
63881df94   Hans Verkuil   [media] vivid: ad...
449
450
451
452
453
454
  }
  
  static inline void tpg_s_buf_height(struct tpg_data *tpg, unsigned h)
  {
  	tpg->buf_height = h;
  }
43047f6b7   Hans Verkuil   [media] vivid: fi...
455
  static inline void tpg_s_field(struct tpg_data *tpg, unsigned field, bool alternate)
63881df94   Hans Verkuil   [media] vivid: ad...
456
457
  {
  	tpg->field = field;
43047f6b7   Hans Verkuil   [media] vivid: fi...
458
  	tpg->field_alternate = alternate;
63881df94   Hans Verkuil   [media] vivid: ad...
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
  }
  
  static inline void tpg_s_perc_fill(struct tpg_data *tpg,
  				      unsigned perc_fill)
  {
  	tpg->perc_fill = perc_fill;
  }
  
  static inline unsigned tpg_g_perc_fill(const struct tpg_data *tpg)
  {
  	return tpg->perc_fill;
  }
  
  static inline void tpg_s_perc_fill_blank(struct tpg_data *tpg,
  					 bool perc_fill_blank)
  {
  	tpg->perc_fill_blank = perc_fill_blank;
  }
  
  static inline void tpg_s_video_aspect(struct tpg_data *tpg,
  					enum tpg_video_aspect vid_aspect)
  {
  	if (tpg->vid_aspect == vid_aspect)
  		return;
  	tpg->vid_aspect = vid_aspect;
  	tpg->recalc_square_border = true;
  }
  
  static inline enum tpg_video_aspect tpg_g_video_aspect(const struct tpg_data *tpg)
  {
  	return tpg->vid_aspect;
  }
  
  static inline void tpg_s_pixel_aspect(struct tpg_data *tpg,
  					enum tpg_pixel_aspect pix_aspect)
  {
  	if (tpg->pix_aspect == pix_aspect)
  		return;
  	tpg->pix_aspect = pix_aspect;
  	tpg->recalc_square_border = true;
  }
  
  static inline void tpg_s_show_border(struct tpg_data *tpg,
  					bool show_border)
  {
  	tpg->show_border = show_border;
  }
  
  static inline void tpg_s_show_square(struct tpg_data *tpg,
  					bool show_square)
  {
  	tpg->show_square = show_square;
  }
  
  static inline void tpg_s_insert_sav(struct tpg_data *tpg, bool insert_sav)
  {
  	tpg->insert_sav = insert_sav;
  }
  
  static inline void tpg_s_insert_eav(struct tpg_data *tpg, bool insert_eav)
  {
  	tpg->insert_eav = insert_eav;
  }
  
  void tpg_update_mv_step(struct tpg_data *tpg);
  
  static inline void tpg_s_mv_hor_mode(struct tpg_data *tpg,
  				enum tpg_move_mode mv_hor_mode)
  {
  	tpg->mv_hor_mode = mv_hor_mode;
  	tpg_update_mv_step(tpg);
  }
  
  static inline void tpg_s_mv_vert_mode(struct tpg_data *tpg,
  				enum tpg_move_mode mv_vert_mode)
  {
  	tpg->mv_vert_mode = mv_vert_mode;
  	tpg_update_mv_step(tpg);
  }
  
  static inline void tpg_init_mv_count(struct tpg_data *tpg)
  {
  	tpg->mv_hor_count = tpg->mv_vert_count = 0;
  }
  
  static inline void tpg_update_mv_count(struct tpg_data *tpg, bool frame_is_field)
  {
  	tpg->mv_hor_count += tpg->mv_hor_step * (frame_is_field ? 1 : 2);
  	tpg->mv_vert_count += tpg->mv_vert_step * (frame_is_field ? 1 : 2);
  }
  
  static inline void tpg_s_hflip(struct tpg_data *tpg, bool hflip)
  {
  	if (tpg->hflip == hflip)
  		return;
  	tpg->hflip = hflip;
  	tpg_update_mv_step(tpg);
  	tpg->recalc_lines = true;
  }
  
  static inline bool tpg_g_hflip(const struct tpg_data *tpg)
  {
  	return tpg->hflip;
  }
  
  static inline void tpg_s_vflip(struct tpg_data *tpg, bool vflip)
  {
  	tpg->vflip = vflip;
  }
  
  static inline bool tpg_g_vflip(const struct tpg_data *tpg)
  {
  	return tpg->vflip;
  }
  
  static inline bool tpg_pattern_is_static(const struct tpg_data *tpg)
  {
  	return tpg->pattern != TPG_PAT_NOISE &&
  	       tpg->mv_hor_mode == TPG_MOVE_NONE &&
  	       tpg->mv_vert_mode == TPG_MOVE_NONE;
  }
  
  #endif