[media] vivid: add teletext support to VBI capture

This is useful to test teletext capture applications like alevt and mtt.

It also fixes a previously undetected bug where the PAL VBI start line
of the second field was off by one. Using the new field start defines
helps a lot fixing such bugs.

Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
This commit is contained in:
Hans Verkuil 2014-09-20 06:11:44 -03:00 committed by Mauro Carvalho Chehab
parent 625c3442dc
commit 62f28725a8
7 changed files with 117 additions and 25 deletions

View File

@ -422,7 +422,7 @@ generate Closed Caption and XDS data. The closed caption stream will
alternate between "Hello world!" and "Closed captions test" every second. alternate between "Hello world!" and "Closed captions test" every second.
The XDS stream will give the current time once a minute. For 50 Hz standards The XDS stream will give the current time once a minute. For 50 Hz standards
it will generate the Wide Screen Signal which is based on the actual Video it will generate the Wide Screen Signal which is based on the actual Video
Aspect Ratio control setting. Aspect Ratio control setting and teletext pages 100-159, one page per frame.
The VBI device will only work for the S-Video and TV inputs, it will give The VBI device will only work for the S-Video and TV inputs, it will give
back an error if the current input is a webcam or HDMI. back an error if the current input is a webcam or HDMI.
@ -435,8 +435,8 @@ There are three types of VBI output devices: those that only support raw
(undecoded) VBI, those that only support sliced (decoded) VBI and those that (undecoded) VBI, those that only support sliced (decoded) VBI and those that
support both. This is determined by the node_types module option. support both. This is determined by the node_types module option.
The sliced VBI output supports the Wide Screen Signal for 50 Hz standards The sliced VBI output supports the Wide Screen Signal and the teletext signal
and Closed Captioning + XDS for 60 Hz standards. for 50 Hz standards and Closed Captioning + XDS for 60 Hz standards.
The VBI device will only work for the S-Video output, it will give The VBI device will only work for the S-Video output, it will give
back an error if the current output is HDMI. back an error if the current output is HDMI.
@ -910,7 +910,8 @@ capture device.
For VBI looping to work all of the above must be valid and in addition the vbi For VBI looping to work all of the above must be valid and in addition the vbi
output must be configured for sliced VBI. The VBI capture side can be configured output must be configured for sliced VBI. The VBI capture side can be configured
for either raw or sliced VBI. for either raw or sliced VBI. Note that at the moment only CC/XDS (60 Hz formats)
and WSS (50 Hz formats) VBI data is looped. Teletext VBI data is not looped.
Section 10.2: Radio & RDS Looping Section 10.2: Radio & RDS Looping
@ -1090,6 +1091,7 @@ Just as a reminder and in no particular order:
- Add virtual sub-devices and media controller support - Add virtual sub-devices and media controller support
- Some support for testing compressed video - Some support for testing compressed video
- Add support to loop raw VBI output to raw VBI input - Add support to loop raw VBI output to raw VBI input
- Add support to loop teletext sliced VBI output to VBI input
- Fix sequence/field numbering when looping of video with alternate fields - Fix sequence/field numbering when looping of video with alternate fields
- Add support for V4L2_CID_BG_COLOR for video outputs - Add support for V4L2_CID_BG_COLOR for video outputs
- Add ARGB888 overlay support: better testing of the alpha channel - Add ARGB888 overlay support: better testing of the alpha channel

View File

@ -37,25 +37,25 @@ static void vivid_sliced_vbi_cap_fill(struct vivid_dev *dev, unsigned seqnr)
if (!is_60hz) { if (!is_60hz) {
if (dev->loop_video) { if (dev->loop_video) {
if (dev->vbi_out_have_wss) { if (dev->vbi_out_have_wss) {
vbi_gen->data[0].data[0] = dev->vbi_out_wss[0]; vbi_gen->data[12].data[0] = dev->vbi_out_wss[0];
vbi_gen->data[0].data[1] = dev->vbi_out_wss[1]; vbi_gen->data[12].data[1] = dev->vbi_out_wss[1];
} else { } else {
vbi_gen->data[0].id = 0; vbi_gen->data[12].id = 0;
} }
} else { } else {
switch (tpg_g_video_aspect(&dev->tpg)) { switch (tpg_g_video_aspect(&dev->tpg)) {
case TPG_VIDEO_ASPECT_14X9_CENTRE: case TPG_VIDEO_ASPECT_14X9_CENTRE:
vbi_gen->data[0].data[0] = 0x01; vbi_gen->data[12].data[0] = 0x01;
break; break;
case TPG_VIDEO_ASPECT_16X9_CENTRE: case TPG_VIDEO_ASPECT_16X9_CENTRE:
vbi_gen->data[0].data[0] = 0x0b; vbi_gen->data[12].data[0] = 0x0b;
break; break;
case TPG_VIDEO_ASPECT_16X9_ANAMORPHIC: case TPG_VIDEO_ASPECT_16X9_ANAMORPHIC:
vbi_gen->data[0].data[0] = 0x07; vbi_gen->data[12].data[0] = 0x07;
break; break;
case TPG_VIDEO_ASPECT_4X3: case TPG_VIDEO_ASPECT_4X3:
default: default:
vbi_gen->data[0].data[0] = 0x08; vbi_gen->data[12].data[0] = 0x08;
break; break;
} }
} }
@ -83,8 +83,8 @@ static void vivid_g_fmt_vbi_cap(struct vivid_dev *dev, struct v4l2_vbi_format *v
vbi->offset = 24; vbi->offset = 24;
vbi->samples_per_line = 1440; vbi->samples_per_line = 1440;
vbi->sample_format = V4L2_PIX_FMT_GREY; vbi->sample_format = V4L2_PIX_FMT_GREY;
vbi->start[0] = is_60hz ? 10 : 6; vbi->start[0] = is_60hz ? V4L2_VBI_ITU_525_F1_START + 9 : V4L2_VBI_ITU_625_F1_START + 5;
vbi->start[1] = is_60hz ? 273 : 318; vbi->start[1] = is_60hz ? V4L2_VBI_ITU_525_F2_START + 9 : V4L2_VBI_ITU_625_F2_START + 5;
vbi->count[0] = vbi->count[1] = is_60hz ? 12 : 18; vbi->count[0] = vbi->count[1] = is_60hz ? 12 : 18;
vbi->flags = dev->vbi_cap_interlaced ? V4L2_VBI_INTERLACED : 0; vbi->flags = dev->vbi_cap_interlaced ? V4L2_VBI_INTERLACED : 0;
vbi->reserved[0] = 0; vbi->reserved[0] = 0;
@ -125,8 +125,10 @@ void vivid_sliced_vbi_cap_process(struct vivid_dev *dev, struct vivid_buffer *bu
memset(vbuf, 0, vb2_plane_size(&buf->vb, 0)); memset(vbuf, 0, vb2_plane_size(&buf->vb, 0));
if (!VIVID_INVALID_SIGNAL(dev->std_signal_mode)) { if (!VIVID_INVALID_SIGNAL(dev->std_signal_mode)) {
vbuf[0] = dev->vbi_gen.data[0]; unsigned i;
vbuf[1] = dev->vbi_gen.data[1];
for (i = 0; i < 25; i++)
vbuf[i] = dev->vbi_gen.data[i];
} }
v4l2_get_timestamp(&buf->vb.v4l2_buf.timestamp); v4l2_get_timestamp(&buf->vb.v4l2_buf.timestamp);
@ -280,9 +282,15 @@ void vivid_fill_service_lines(struct v4l2_sliced_vbi_format *vbi, u32 service_se
vbi->service_lines[0][21] = V4L2_SLICED_CAPTION_525; vbi->service_lines[0][21] = V4L2_SLICED_CAPTION_525;
vbi->service_lines[1][21] = V4L2_SLICED_CAPTION_525; vbi->service_lines[1][21] = V4L2_SLICED_CAPTION_525;
} }
if (vbi->service_set & V4L2_SLICED_WSS_625) if (vbi->service_set & V4L2_SLICED_WSS_625) {
unsigned i;
for (i = 7; i <= 18; i++)
vbi->service_lines[0][i] =
vbi->service_lines[1][i] = V4L2_SLICED_TELETEXT_B;
vbi->service_lines[0][23] = V4L2_SLICED_WSS_625; vbi->service_lines[0][23] = V4L2_SLICED_WSS_625;
} }
}
int vidioc_g_fmt_sliced_vbi_cap(struct file *file, void *fh, struct v4l2_format *fmt) int vidioc_g_fmt_sliced_vbi_cap(struct file *file, void *fh, struct v4l2_format *fmt)
{ {
@ -306,7 +314,8 @@ int vidioc_try_fmt_sliced_vbi_cap(struct file *file, void *fh, struct v4l2_forma
if (!vivid_is_sdtv_cap(dev) || !dev->has_sliced_vbi_cap) if (!vivid_is_sdtv_cap(dev) || !dev->has_sliced_vbi_cap)
return -EINVAL; return -EINVAL;
service_set &= is_60hz ? V4L2_SLICED_CAPTION_525 : V4L2_SLICED_WSS_625; service_set &= is_60hz ? V4L2_SLICED_CAPTION_525 :
V4L2_SLICED_WSS_625 | V4L2_SLICED_TELETEXT_B;
vivid_fill_service_lines(vbi, service_set); vivid_fill_service_lines(vbi, service_set);
return 0; return 0;
} }
@ -345,11 +354,17 @@ int vidioc_g_sliced_vbi_cap(struct file *file, void *fh, struct v4l2_sliced_vbi_
return -EINVAL; return -EINVAL;
} }
cap->service_set = is_60hz ? V4L2_SLICED_CAPTION_525 : V4L2_SLICED_WSS_625; cap->service_set = is_60hz ? V4L2_SLICED_CAPTION_525 :
V4L2_SLICED_WSS_625 | V4L2_SLICED_TELETEXT_B;
if (is_60hz) { if (is_60hz) {
cap->service_lines[0][21] = V4L2_SLICED_CAPTION_525; cap->service_lines[0][21] = V4L2_SLICED_CAPTION_525;
cap->service_lines[1][21] = V4L2_SLICED_CAPTION_525; cap->service_lines[1][21] = V4L2_SLICED_CAPTION_525;
} else { } else {
unsigned i;
for (i = 7; i <= 18; i++)
cap->service_lines[0][i] =
cap->service_lines[1][i] = V4L2_SLICED_TELETEXT_B;
cap->service_lines[0][23] = V4L2_SLICED_WSS_625; cap->service_lines[0][23] = V4L2_SLICED_WSS_625;
} }
return 0; return 0;

View File

@ -57,6 +57,27 @@ static void vivid_vbi_gen_wss_raw(const struct v4l2_sliced_vbi_data *data,
} }
} }
static void vivid_vbi_gen_teletext_raw(const struct v4l2_sliced_vbi_data *data,
u8 *buf, unsigned sampling_rate)
{
const unsigned rate = 6937500 / 10; /* Teletext has a 6.9375 MHz transmission rate */
u8 teletext[45] = { 0x55, 0x55, 0x27 };
unsigned bit = 0;
int i;
memcpy(teletext + 3, data->data, sizeof(teletext) - 3);
/* prevents 32 bit overflow */
sampling_rate /= 10;
for (i = 0, bit = 0; bit < sizeof(teletext) * 8; bit++) {
unsigned n = ((bit + 1) * sampling_rate) / rate;
u8 val = (teletext[bit / 8] & (1 << (bit & 7))) ? 0xc0 : 0x10;
while (i < n)
buf[i++] = val;
}
}
static void cc_insert(u8 *cc, u8 ch) static void cc_insert(u8 *cc, u8 ch)
{ {
unsigned tot = 0; unsigned tot = 0;
@ -102,7 +123,7 @@ void vivid_vbi_gen_raw(const struct vivid_vbi_gen_data *vbi,
{ {
unsigned idx; unsigned idx;
for (idx = 0; idx < 2; idx++) { for (idx = 0; idx < 25; idx++) {
const struct v4l2_sliced_vbi_data *data = vbi->data + idx; const struct v4l2_sliced_vbi_data *data = vbi->data + idx;
unsigned start_2nd_field; unsigned start_2nd_field;
unsigned line = data->line; unsigned line = data->line;
@ -123,6 +144,8 @@ void vivid_vbi_gen_raw(const struct vivid_vbi_gen_data *vbi,
vivid_vbi_gen_cc_raw(data, linebuf, vbi_fmt->sampling_rate); vivid_vbi_gen_cc_raw(data, linebuf, vbi_fmt->sampling_rate);
else if (data->id == V4L2_SLICED_WSS_625) else if (data->id == V4L2_SLICED_WSS_625)
vivid_vbi_gen_wss_raw(data, linebuf, vbi_fmt->sampling_rate); vivid_vbi_gen_wss_raw(data, linebuf, vbi_fmt->sampling_rate);
else if (data->id == V4L2_SLICED_TELETEXT_B)
vivid_vbi_gen_teletext_raw(data, linebuf, vbi_fmt->sampling_rate);
} }
} }
@ -197,6 +220,41 @@ static void vivid_vbi_gen_set_time_of_day(u8 *packet)
packet[15] = calc_parity(0x100 - checksum); packet[15] = calc_parity(0x100 - checksum);
} }
static const u8 hamming[16] = {
0x15, 0x02, 0x49, 0x5e, 0x64, 0x73, 0x38, 0x2f,
0xd0, 0xc7, 0x8c, 0x9b, 0xa1, 0xb6, 0xfd, 0xea
};
static void vivid_vbi_gen_teletext(u8 *packet, unsigned line, unsigned frame)
{
unsigned offset = 2;
unsigned i;
packet[0] = hamming[1 + ((line & 1) << 3)];
packet[1] = hamming[line >> 1];
memset(packet + 2, 0x20, 40);
if (line == 0) {
/* subcode */
packet[2] = hamming[frame % 10];
packet[3] = hamming[frame / 10];
packet[4] = hamming[0];
packet[5] = hamming[0];
packet[6] = hamming[0];
packet[7] = hamming[0];
packet[8] = hamming[0];
packet[9] = hamming[1];
offset = 10;
}
packet += offset;
memcpy(packet, "Page: 100 Row: 10", 17);
packet[7] = '0' + frame / 10;
packet[8] = '0' + frame % 10;
packet[15] = '0' + line / 10;
packet[16] = '0' + line % 10;
for (i = 0; i < 42 - offset; i++)
packet[i] = calc_parity(packet[i]);
}
void vivid_vbi_gen_sliced(struct vivid_vbi_gen_data *vbi, void vivid_vbi_gen_sliced(struct vivid_vbi_gen_data *vbi,
bool is_60hz, unsigned seqnr) bool is_60hz, unsigned seqnr)
{ {
@ -207,10 +265,26 @@ void vivid_vbi_gen_sliced(struct vivid_vbi_gen_data *vbi,
memset(vbi->data, 0, sizeof(vbi->data)); memset(vbi->data, 0, sizeof(vbi->data));
if (!is_60hz) { if (!is_60hz) {
unsigned i;
for (i = 0; i <= 11; i++) {
data0->id = V4L2_SLICED_TELETEXT_B;
data0->line = 7 + i;
vivid_vbi_gen_teletext(data0->data, i, frame);
data0++;
}
data0->id = V4L2_SLICED_WSS_625; data0->id = V4L2_SLICED_WSS_625;
data0->line = 23; data0->line = 23;
/* 4x3 video aspect ratio */ /* 4x3 video aspect ratio */
data0->data[0] = 0x08; data0->data[0] = 0x08;
data0++;
for (i = 0; i <= 11; i++) {
data0->id = V4L2_SLICED_TELETEXT_B;
data0->field = 1;
data0->line = 7 + i;
vivid_vbi_gen_teletext(data0->data, 12 + i, frame);
data0++;
}
return; return;
} }

View File

@ -21,7 +21,7 @@
#define _VIVID_VBI_GEN_H_ #define _VIVID_VBI_GEN_H_
struct vivid_vbi_gen_data { struct vivid_vbi_gen_data {
struct v4l2_sliced_vbi_data data[2]; struct v4l2_sliced_vbi_data data[25];
u8 time_of_day_packet[16]; u8 time_of_day_packet[16];
}; };

View File

@ -149,8 +149,8 @@ int vidioc_g_fmt_vbi_out(struct file *file, void *priv,
vbi->offset = 24; vbi->offset = 24;
vbi->samples_per_line = 1440; vbi->samples_per_line = 1440;
vbi->sample_format = V4L2_PIX_FMT_GREY; vbi->sample_format = V4L2_PIX_FMT_GREY;
vbi->start[0] = is_60hz ? 10 : 6; vbi->start[0] = is_60hz ? V4L2_VBI_ITU_525_F1_START + 9 : V4L2_VBI_ITU_625_F1_START + 5;
vbi->start[1] = is_60hz ? 273 : 318; vbi->start[1] = is_60hz ? V4L2_VBI_ITU_525_F2_START + 9 : V4L2_VBI_ITU_625_F2_START + 5;
vbi->count[0] = vbi->count[1] = is_60hz ? 12 : 18; vbi->count[0] = vbi->count[1] = is_60hz ? 12 : 18;
vbi->flags = dev->vbi_cap_interlaced ? V4L2_VBI_INTERLACED : 0; vbi->flags = dev->vbi_cap_interlaced ? V4L2_VBI_INTERLACED : 0;
vbi->reserved[0] = 0; vbi->reserved[0] = 0;
@ -195,7 +195,8 @@ int vidioc_try_fmt_sliced_vbi_out(struct file *file, void *fh, struct v4l2_forma
if (!vivid_is_svid_out(dev) || !dev->has_sliced_vbi_out) if (!vivid_is_svid_out(dev) || !dev->has_sliced_vbi_out)
return -EINVAL; return -EINVAL;
service_set &= is_60hz ? V4L2_SLICED_CAPTION_525 : V4L2_SLICED_WSS_625; service_set &= is_60hz ? V4L2_SLICED_CAPTION_525 :
V4L2_SLICED_WSS_625 | V4L2_SLICED_TELETEXT_B;
vivid_fill_service_lines(vbi, service_set); vivid_fill_service_lines(vbi, service_set);
return 0; return 0;
} }

View File

@ -419,7 +419,7 @@ void vivid_update_format_cap(struct vivid_dev *dev, bool keep_controls)
} else { } else {
dev->src_rect.height = 576; dev->src_rect.height = 576;
dev->timeperframe_vid_cap = (struct v4l2_fract) { 1000, 25000 }; dev->timeperframe_vid_cap = (struct v4l2_fract) { 1000, 25000 };
dev->service_set_cap = V4L2_SLICED_WSS_625; dev->service_set_cap = V4L2_SLICED_WSS_625 | V4L2_SLICED_TELETEXT_B;
} }
tpg_s_rgb_range(&dev->tpg, V4L2_DV_RGB_RANGE_AUTO); tpg_s_rgb_range(&dev->tpg, V4L2_DV_RGB_RANGE_AUTO);
break; break;

View File

@ -234,7 +234,7 @@ void vivid_update_format_out(struct vivid_dev *dev)
} else { } else {
dev->sink_rect.height = 576; dev->sink_rect.height = 576;
dev->timeperframe_vid_out = (struct v4l2_fract) { 1000, 25000 }; dev->timeperframe_vid_out = (struct v4l2_fract) { 1000, 25000 };
dev->service_set_out = V4L2_SLICED_WSS_625; dev->service_set_out = V4L2_SLICED_WSS_625 | V4L2_SLICED_TELETEXT_B;
} }
dev->colorspace_out = V4L2_COLORSPACE_SMPTE170M; dev->colorspace_out = V4L2_COLORSPACE_SMPTE170M;
break; break;