Re: [PATCH] cx18, cx2341x, ivtv: Add AC-3 audio encoding control to cx18

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sat, 2009-01-03 at 10:36 +0100, Hans Verkuil wrote: 
> Hi Andy,
> 
> I've done a quick review:
> 
> On Thursday 01 January 2009 23:20:08 Andy Walls wrote:
> > The patch in line below adds a control to the cx18 driver to request
> > AC-3 audio instead of MPEG Layer II.  It doesn't quite work yet due to
> > cx18 firmware issues.
> >
> > However, I think I've got the basic control work done and need a review
> > to make sure I didn't muck anything up with the cx2341x or ivtv modules.
> >
> > Of particular concern to me is
> >
> > a) changing the cx2341x "audio_properties" from a u16 to a u32, as this
> > is what rippled down in source code to the to ivtv driver.
> >
> > b) accidentally adding a bogus options or controls to ivtv.
> >
> >
> > The change can also be found at
> >
> > http://linuxtv.org/hg/~awalls/v4l-dvb
> >
> > Regards,
> > Andy
> >
> >
> > # HG changeset patch
> > # User Andy Walls <awalls@xxxxxxxxx>
> > # Date 1230847351 18000
> > # Node ID e9cf344a6749de5d3778fac3c7114476f7f0b647
> > # Parent  41242777b3d8bb162c65c2d0b2c542417b72d946
> > cx18, cx2341x, ivtv: Add AC-3 audio encoding control to cx18
> >
> > From: Andy Walls <awalls@xxxxxxxxx>
> >
> > Initial addition of controls to set AC-3 audio encoding for the CX23418 -
> > it does not work yet due to firmware or cx18 driver issues.  This change
> > affects the common cx2341x and ivtv modules due to shared structures and
> > common functions.
> >
> > Priority: normal
> >
> > Signed-off-by: Andy Walls <awalls@xxxxxxxxx>
> >
> > diff -r 41242777b3d8 -r e9cf344a6749
> > linux/drivers/media/video/cx18/cx18-driver.c ---
> > a/linux/drivers/media/video/cx18/cx18-driver.c	Thu Jan 01 10:35:06 2009
> > -0500 +++ b/linux/drivers/media/video/cx18/cx18-driver.c	Thu Jan 01
> > 17:02:31 2009 -0500 @@ -592,7 +592,8 @@ static int __devinit
> > cx18_init_struct1(s
> >  		(cx->params.video_temporal_filter_mode << 1) |
> >  		(cx->params.video_median_filter_type << 2);
> >  	cx->params.port = CX2341X_PORT_MEMORY;
> > -	cx->params.capabilities = CX2341X_CAP_HAS_TS;
> > +	cx->params.capabilities = CX2341X_CAP_HAS_TS   | CX2341X_CAP_HAS_AC3 |
> > +				  CX2341X_CAP_HAS_LPCM;
> >  	init_waitqueue_head(&cx->cap_w);
> >  	init_waitqueue_head(&cx->mb_apu_waitq);
> >  	init_waitqueue_head(&cx->mb_cpu_waitq);
> > diff -r 41242777b3d8 -r e9cf344a6749
> > linux/drivers/media/video/cx18/cx18-driver.h ---
> > a/linux/drivers/media/video/cx18/cx18-driver.h	Thu Jan 01 10:35:06 2009
> > -0500 +++ b/linux/drivers/media/video/cx18/cx18-driver.h	Thu Jan 01
> > 17:02:31 2009 -0500 @@ -413,7 +413,7 @@ struct cx18 {
> >
> >  	/* dualwatch */
> >  	unsigned long dualwatch_jiffies;
> > -	u16 dualwatch_stereo_mode;
> > +	u32 dualwatch_stereo_mode;
> >
> >  	/* Digitizer type */
> >  	int digitizer;		/* 0x00EF = saa7114 0x00FO = saa7115 0x0106 = mic */
> > diff -r 41242777b3d8 -r e9cf344a6749
> > linux/drivers/media/video/cx18/cx18-fileops.c ---
> > a/linux/drivers/media/video/cx18/cx18-fileops.c	Thu Jan 01 10:35:06 2009
> > -0500 +++ b/linux/drivers/media/video/cx18/cx18-fileops.c	Thu Jan 01
> > 17:02:31 2009 -0500 @@ -128,10 +128,10 @@ static void
> > cx18_dualwatch(struct cx18 *
> >  static void cx18_dualwatch(struct cx18 *cx)
> >  {
> >  	struct v4l2_tuner vt;
> > -	u16 new_bitmap;
> > -	u16 new_stereo_mode;
> > -	const u16 stereo_mask = 0x0300;
> > -	const u16 dual = 0x0200;
> > +	u32 new_bitmap;
> > +	u32 new_stereo_mode;
> > +	const u32 stereo_mask = 0x0300;
> > +	const u32 dual = 0x0200;
> >  	u32 h;
> >
> >  	new_stereo_mode = cx->params.audio_properties & stereo_mask;
> > diff -r 41242777b3d8 -r e9cf344a6749 linux/drivers/media/video/cx2341x.c
> > --- a/linux/drivers/media/video/cx2341x.c	Thu Jan 01 10:35:06 2009 -0500
> > +++ b/linux/drivers/media/video/cx2341x.c	Thu Jan 01 17:02:31 2009 -0500
> > @@ -1,5 +1,5 @@
> >  /*
> > - * cx2341x - generic code for cx23415/6 based devices
> > + * cx2341x - generic code for cx23415/6/8 based devices
> >   *
> >   * Copyright (C) 2006 Hans Verkuil <hverkuil@xxxxxxxxx>
> >   *
> > @@ -31,7 +31,7 @@
> >  #include <media/v4l2-common.h>
> >  #include "compat.h"
> >
> > -MODULE_DESCRIPTION("cx23415/6 driver");
> > +MODULE_DESCRIPTION("cx23415/6/8 driver");
> >  MODULE_AUTHOR("Hans Verkuil");
> >  MODULE_LICENSE("GPL");
> >
> > @@ -46,6 +46,7 @@ const u32 cx2341x_mpeg_ctrls[] = {
> >  	V4L2_CID_MPEG_AUDIO_SAMPLING_FREQ,
> >  	V4L2_CID_MPEG_AUDIO_ENCODING,
> >  	V4L2_CID_MPEG_AUDIO_L2_BITRATE,
> > +	V4L2_CID_MPEG_AUDIO_AC3_BITRATE,
> >  	V4L2_CID_MPEG_AUDIO_MODE,
> >  	V4L2_CID_MPEG_AUDIO_MODE_EXTENSION,
> >  	V4L2_CID_MPEG_AUDIO_EMPHASIS,
> > @@ -95,6 +96,7 @@ static const struct cx2341x_mpeg_params
> >  	.audio_sampling_freq = V4L2_MPEG_AUDIO_SAMPLING_FREQ_48000,
> >  	.audio_encoding = V4L2_MPEG_AUDIO_ENCODING_LAYER_2,
> >  	.audio_l2_bitrate = V4L2_MPEG_AUDIO_L2_BITRATE_224K,
> > +	.audio_ac3_bitrate = V4L2_MPEG_AUDIO_AC3_BITRATE_224K,
> >  	.audio_mode = V4L2_MPEG_AUDIO_MODE_STEREO,
> >  	.audio_mode_extension = V4L2_MPEG_AUDIO_MODE_EXTENSION_BOUND_4,
> >  	.audio_emphasis = V4L2_MPEG_AUDIO_EMPHASIS_NONE,
> > @@ -149,6 +151,9 @@ static int cx2341x_get_ctrl(const struct
> >  	case V4L2_CID_MPEG_AUDIO_L2_BITRATE:
> >  		ctrl->value = params->audio_l2_bitrate;
> >  		break;
> > +	case V4L2_CID_MPEG_AUDIO_AC3_BITRATE:
> > +		ctrl->value = params->audio_ac3_bitrate;
> > +		break;
> >  	case V4L2_CID_MPEG_AUDIO_MODE:
> >  		ctrl->value = params->audio_mode;
> >  		break;
> > @@ -257,12 +262,23 @@ static int cx2341x_set_ctrl(struct cx234
> >  		params->audio_sampling_freq = ctrl->value;
> >  		break;
> >  	case V4L2_CID_MPEG_AUDIO_ENCODING:
> > +		if (busy)
> > +			return -EBUSY;
> > +		if (params->capabilities & CX2341X_CAP_HAS_AC3 &&
> > +		    ctrl->value != V4L2_MPEG_AUDIO_ENCODING_LAYER_2 &&
> > +		    ctrl->value != V4L2_MPEG_AUDIO_ENCODING_AC3)
> > +			return -EINVAL;
> 
> This can't be right: if CAP_HAS_AC3 is not set, then it will always 
> return -EINVAL.

Ack!  You're right.  I shouldn't have done this stuff on New Year's Day
- I needed more sleep.


> >  		params->audio_encoding = ctrl->value;
> >  		break;
> >  	case V4L2_CID_MPEG_AUDIO_L2_BITRATE:
> >  		if (busy)
> >  			return -EBUSY;
> >  		params->audio_l2_bitrate = ctrl->value;
> > +		break;
> > +	case V4L2_CID_MPEG_AUDIO_AC3_BITRATE:
> > +		if (busy)
> > +			return -EBUSY;
> > +		params->audio_ac3_bitrate = ctrl->value;
> >  		break;
> 
> This should test for CAP_HAS_AC3 as well.

I wondered about this, since I added V4L2_CID_MPEG_AUDIO_AC3_BITRATE in
the cx2341x_mpeg_ctrls[] list of controls for CX2341x devices, I wasn't
sure of the user experience for always getting -EINVAL for a listed
control.  There is no harm in letting the user app set a control value
that will never be used for CX23415/6 devices.

I didn't want to make an alternate cx2341x_mpeg_ctrls[] list.  I'll put
in a check for the CAP_HAS_AC3 and make the control inactive or
read-only, if the CAP isn't set.


> >  	case V4L2_CID_MPEG_AUDIO_MODE:
> >  		params->audio_mode = ctrl->value;
> > @@ -483,6 +499,12 @@ int cx2341x_ctrl_query(const struct cx23
> >
> >  	switch (qctrl->id) {
> >  	case V4L2_CID_MPEG_AUDIO_ENCODING:
> > +		if (params->capabilities & CX2341X_CAP_HAS_AC3)
> > +			return v4l2_ctrl_query_fill(qctrl,
> > +					V4L2_MPEG_AUDIO_ENCODING_LAYER_2,
> > +					V4L2_MPEG_AUDIO_ENCODING_AC3, 1,
> > +					default_params.audio_encoding);
> > +
> >  		return v4l2_ctrl_query_fill(qctrl,
> >  				V4L2_MPEG_AUDIO_ENCODING_LAYER_2,
> >  				V4L2_MPEG_AUDIO_ENCODING_LAYER_2, 1,
> > @@ -497,6 +519,12 @@ int cx2341x_ctrl_query(const struct cx23
> >  	case V4L2_CID_MPEG_AUDIO_L1_BITRATE:
> >  	case V4L2_CID_MPEG_AUDIO_L3_BITRATE:
> >  		return -EINVAL;
> > +
> > +	case V4L2_CID_MPEG_AUDIO_AC3_BITRATE:
> > +		return v4l2_ctrl_query_fill(qctrl,
> > +				V4L2_MPEG_AUDIO_AC3_BITRATE_48K,
> > +				V4L2_MPEG_AUDIO_AC3_BITRATE_448K, 1,
> > +				default_params.audio_ac3_bitrate);
> 
> Also needs a test.

OK.  Can do.

BTW, The old code fell through to the default case.  I note that the L1
& L3 bitrate controls aren't valid, are trapped here, but aren't in the
cx2341x_mpeg_ctrls[] list.  I was unsure if a control not in the
cx2341x_mpeg_ctrls[] list would ever come through here.

> >  	case V4L2_CID_MPEG_AUDIO_MODE_EXTENSION:
> >  		err = v4l2_ctrl_query_fill_std(qctrl);
> > @@ -672,6 +700,15 @@ const char **cx2341x_ctrl_get_menu(const
> >  		NULL
> >  	};
> >
> > +	static const char *mpeg_audio_encoding_l2_ac3[] = {
> > +		"",
> > +		"MPEG-1/2 Layer II",
> > +		"",
> > +		"",
> > +		"AC-3",
> > +		NULL
> > +	};
> > +
> >  	static const char *cx2341x_video_spatial_filter_mode_menu[] = {
> >  		"Manual",
> >  		"Auto",
> > @@ -712,6 +749,9 @@ const char **cx2341x_ctrl_get_menu(const
> >  	case V4L2_CID_MPEG_STREAM_TYPE:
> >  		return (p->capabilities & CX2341X_CAP_HAS_TS) ?
> >  			mpeg_stream_type_with_ts : mpeg_stream_type_without_ts;
> > +	case V4L2_CID_MPEG_AUDIO_ENCODING:
> > +		return (p->capabilities & CX2341X_CAP_HAS_AC3) ?
> > +			mpeg_audio_encoding_l2_ac3 : v4l2_ctrl_get_menu(id);
> >  	case V4L2_CID_MPEG_AUDIO_L1_BITRATE:
> >  	case V4L2_CID_MPEG_AUDIO_L3_BITRATE:
> >  		return NULL;
> > @@ -731,16 +771,36 @@ const char **cx2341x_ctrl_get_menu(const
> >  }
> >  EXPORT_SYMBOL(cx2341x_ctrl_get_menu);
> >
> > +/* definitions for audio properties bits 29-28 */
> > +#define CX2341X_AUDIO_ENCDING_METHOD_MPEG	0
> > +#define CX2341X_AUDIO_ENCDING_METHOD_AC3	1
> > +#define CX2341X_AUDIO_ENCDING_METHOD_LPCM	2
> 
> ENCDING? You mean ENCODING.

Yes, thanks.  Sleep deprivation and not wearing my glasses. 8-)


> > +
> >  static void cx2341x_calc_audio_properties(struct cx2341x_mpeg_params
> > *params) {
> > -	params->audio_properties = (params->audio_sampling_freq << 0) |
> > -		((3 - params->audio_encoding) << 2) |
> > -		((1 + params->audio_l2_bitrate) << 4) |
> > +	params->audio_properties =
> > +		(params->audio_sampling_freq << 0) |
> >  		(params->audio_mode << 8) |
> >  		(params->audio_mode_extension << 10) |
> >  		(((params->audio_emphasis == V4L2_MPEG_AUDIO_EMPHASIS_CCITT_J17)
> >  		  ? 3 : params->audio_emphasis) << 12) |
> >  		(params->audio_crc << 14);
> > +
> > +	if ((params->capabilities & CX2341X_CAP_HAS_AC3) &&
> > +	    params->audio_encoding == V4L2_MPEG_AUDIO_ENCODING_AC3) {
> > +		params->audio_properties |=
> > +#if 1
> > +			/* Not sure if this MPEG Layer II setting is required */
> > +			((3 - V4L2_MPEG_AUDIO_ENCODING_LAYER_2) << 2) |
> > +#endif
> > +			(params->audio_ac3_bitrate << 4) |
> > +			(CX2341X_AUDIO_ENCDING_METHOD_AC3 << 28);
> > +	} else {
> > +		/* Assuming MPEG Layer II */
> > +		params->audio_properties |=
> > +			((3 - params->audio_encoding) << 2) |
> > +			((1 + params->audio_l2_bitrate) << 4);
> > +	}
> >  }
> >
> >  int cx2341x_ext_ctrls(struct cx2341x_mpeg_params *params, int busy,
> > @@ -1023,7 +1083,10 @@ void cx2341x_log_status(const struct cx2
> >  		prefix,
> >  		cx2341x_menu_item(p, V4L2_CID_MPEG_AUDIO_SAMPLING_FREQ),
> >  		cx2341x_menu_item(p, V4L2_CID_MPEG_AUDIO_ENCODING),
> > -		cx2341x_menu_item(p, V4L2_CID_MPEG_AUDIO_L2_BITRATE),
> > +		cx2341x_menu_item(p,
> > +			   p->audio_encoding == V4L2_MPEG_AUDIO_ENCODING_AC3
> > +					      ? V4L2_CID_MPEG_AUDIO_AC3_BITRATE
> > +					      : V4L2_CID_MPEG_AUDIO_L2_BITRATE),
> >  		cx2341x_menu_item(p, V4L2_CID_MPEG_AUDIO_MODE),
> >  		p->audio_mute ? " (muted)" : "");
> >  	if (p->audio_mode == V4L2_MPEG_AUDIO_MODE_JOINT_STEREO)
> > diff -r 41242777b3d8 -r e9cf344a6749
> > linux/drivers/media/video/ivtv/ivtv-driver.h ---
> > a/linux/drivers/media/video/ivtv/ivtv-driver.h	Thu Jan 01 10:35:06 2009
> > -0500 +++ b/linux/drivers/media/video/ivtv/ivtv-driver.h	Thu Jan 01
> > 17:02:31 2009 -0500 @@ -697,7 +697,7 @@ struct ivtv {
> >  	u64 vbi_data_inserted;          /* number of VBI bytes inserted into
> > the MPEG stream */ u32 last_dec_timing[3];         /* cache last
> > retrieved pts/scr/frame values */ unsigned long dualwatch_jiffies;/*
> > jiffies value of the previous dualwatch check */ -	u16
> > dualwatch_stereo_mode;      /* current detected dualwatch stereo mode */
> > +	u32 dualwatch_stereo_mode;      /* current detected dualwatch stereo
> > mode */
> >
> >
> >  	/* VBI state info */
> > diff -r 41242777b3d8 -r e9cf344a6749
> > linux/drivers/media/video/ivtv/ivtv-fileops.c ---
> > a/linux/drivers/media/video/ivtv/ivtv-fileops.c	Thu Jan 01 10:35:06 2009
> > -0500 +++ b/linux/drivers/media/video/ivtv/ivtv-fileops.c	Thu Jan 01
> > 17:02:31 2009 -0500 @@ -148,10 +148,10 @@ static void
> > ivtv_dualwatch(struct ivtv *
> >  static void ivtv_dualwatch(struct ivtv *itv)
> >  {
> >  	struct v4l2_tuner vt;
> > -	u16 new_bitmap;
> > -	u16 new_stereo_mode;
> > -	const u16 stereo_mask = 0x0300;
> > -	const u16 dual = 0x0200;
> > +	u32 new_bitmap;
> > +	u32 new_stereo_mode;
> > +	const u32 stereo_mask = 0x0300;
> > +	const u32 dual = 0x0200;
> >
> >  	new_stereo_mode = itv->params.audio_properties & stereo_mask;
> >  	memset(&vt, 0, sizeof(vt));
> > diff -r 41242777b3d8 -r e9cf344a6749 linux/include/media/cx2341x.h
> > --- a/linux/include/media/cx2341x.h	Thu Jan 01 10:35:06 2009 -0500
> > +++ b/linux/include/media/cx2341x.h	Thu Jan 01 17:02:31 2009 -0500
> > @@ -1,5 +1,5 @@
> >  /*
> > -    cx23415/6 header containing common defines.
> > +    cx23415/6/8 header containing common defines.
> >
> >      This program is free software; you can redistribute it and/or modify
> >      it under the terms of the GNU General Public License as published by
> > @@ -28,6 +28,8 @@ enum cx2341x_cap {
> >  enum cx2341x_cap {
> >  	CX2341X_CAP_HAS_SLICED_VBI = 1 << 0,
> >  	CX2341X_CAP_HAS_TS 	   = 1 << 1,
> > +	CX2341X_CAP_HAS_AC3 	   = 1 << 2,
> > +	CX2341X_CAP_HAS_LPCM	   = 1 << 3,
> >  };
> >
> >  struct cx2341x_mpeg_params {
> > @@ -47,11 +49,12 @@ struct cx2341x_mpeg_params {
> >  	enum v4l2_mpeg_audio_sampling_freq audio_sampling_freq;
> >  	enum v4l2_mpeg_audio_encoding audio_encoding;
> >  	enum v4l2_mpeg_audio_l2_bitrate audio_l2_bitrate;
> > +	enum v4l2_mpeg_audio_ac3_bitrate audio_ac3_bitrate;
> >  	enum v4l2_mpeg_audio_mode audio_mode;
> >  	enum v4l2_mpeg_audio_mode_extension audio_mode_extension;
> >  	enum v4l2_mpeg_audio_emphasis audio_emphasis;
> >  	enum v4l2_mpeg_audio_crc audio_crc;
> > -	u16 audio_properties;
> > +	u32 audio_properties;
> >  	u16 audio_mute;
> >
> >  	/* video */
> 
> Regards,
> 
> 	Hans

Thanks for heading off that disaster of a patch.

Regards,
Andy



--
To unsubscribe from this list: send the line "unsubscribe linux-media" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux Input]     [Video for Linux]     [Gstreamer Embedded]     [Mplayer Users]     [Linux USB Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Yosemite Backpacking]
  Powered by Linux