Re: [PATCH 0/3] Improve CEC autorepeat handling

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

 



On 11/27/2017 10:47 AM, Sean Young wrote:
> Hi Hans,
> 
> On Mon, Nov 27, 2017 at 10:13:51AM +0100, Hans Verkuil wrote:
>> On 11/26/2017 12:47 AM, Dmitry Torokhov wrote:
>>> On Fri, Nov 24, 2017 at 11:43:58AM +0000, Sean Young wrote:
>>>> Due to the slowness of the CEC bus, autorepeat handling rather special
>>>> on CEC. If the repeated user control pressed message is received, a 
>>>> keydown repeat should be sent immediately.
>>>
>>> This sounds like you want to have hardware autorepeat combined with
>>> software one. This seems fairly specific to CEC and I do not think that
>>> this should be in input core; but stay in the driver.
>>>
>>> Another option just to decide what common delay for CEC autorepeat is
>>> and rely on the standard autorepeat handling. The benefit is that users
>>> can control the delay before autorepeat kicks in.
>>
>> They are not allowed to. Autorepeat is only allowed to start when a second
>> keydown message arrives within 550 ms as per the spec. After that autorepeat
>> continues as long as keydown messages are received within 550ms from the
>> previous one. The actual REP_PERIOD time is unrelated to the frequency of
>> the CEC messages but should be that of the local system.
>>
>> The thing to remember here is that CEC is slooow (400 bits/s) so you cannot
>> send messages at REP_PERIOD rate. You should see it as messages that tell
>> you to enter/stay in autorepeat mode. Not as actual autorepeat messages.
>>
>>>
>>>>
>>>> By handling this in the input layer, we can remove some ugly code from
>>>> cec, which also sends a keyup event after the first keydown, to prevent
>>>> autorepeat.
>>>
>>> If driver does not want input core to handle autorepeat (but handle
>>> autorepeat by themselves) they should indicate it by setting appropriate
>>> dev->rep[REP_DELAY] and dev->rep[REP_PERIOD] before calling
>>> input_register_device(). This will let input core know that it should
>>> not setup its autorepeat timer.
>>
>> That only means that I have to setup the autorepeat timer myself, there
>> is no benefit in that :-)
>>
>> Sean, I kind of agree with Dmitry here. The way autorepeat works for CEC
>> is pretty specific to that protocol and unlikely to be needed for other
>> protocols.
> 
> That's a valid point, I guess. The only downside is special case handling
> outside the input layer, which would be much simpler in the input layer.
> 
>> It is also no big deal to keep knowledge of that within cec-adap.c.
> 
> So first of all, the sii8620 uses the CEC protocol as well (see commit
> e25f1f7c94e1 drm/bridge/sii8620: add remote control support), so this
> will have to go into rc-core, not cec-adap.c. There was a discussion about
> some time ago.
> 
> The current implementation has an ugly key up event which would be nice
> to do without.
> 
>> The only thing that would be nice to have control over is that with CEC
>> userspace shouldn't be able to change REP_DELAY and that REP_DELAY should
>> always be identical to REP_PERIOD. If this can be done easily, then that
>> would be nice, but it's a nice-to-have in my opinion.
> 
> The REP_DELAY must be equal to REP_PERIOD seems a bit odd to me. In fact,
> I propose something different. If REP_DELAY != 0 then the input layer
> produces autorepeats as normal. If REP_DELAY == 0, then generate repeats
> on the second key down event.
> 
> See patch below.
> 
> Thanks,
> Sean
> ----
> From 3f439e326888a0ab8688d73c4276ac87b4225b1c Mon Sep 17 00:00:00 2001
> From: Sean Young <sean@xxxxxxxx>
> Date: Thu, 23 Nov 2017 22:37:10 +0000
> Subject: [PATCH] media: cec: move cec autorepeat handling to rc-core
> 
> CEC autorepeat is different than other protocols. Autorepeat is triggered
> by the first repeated user control pressed CEC message, rather than a
> fixed REP_DELAY.
> 
> This change also does away with the KEY_UP event directly after the first
> KEY_DOWN event, which was used to stop autorepeat from starting.
> 
> See commit a9a249a2c997 ("media: cec: fix remote control passthrough")
> for the original change.
> 
> Signed-off-by: Sean Young <sean@xxxxxxxx>
> ---
>  drivers/media/cec/cec-adap.c | 56 ++++----------------------------------------
>  drivers/media/cec/cec-core.c | 12 ----------
>  drivers/media/rc/rc-main.c   | 45 +++++++++++++++++++++++++++++++++--
>  include/media/cec.h          |  5 ----
>  include/media/rc-core.h      |  3 +++
>  5 files changed, 51 insertions(+), 70 deletions(-)
> 
> diff --git a/drivers/media/cec/cec-adap.c b/drivers/media/cec/cec-adap.c
> index 98f88c43f62c..258a91800963 100644
> --- a/drivers/media/cec/cec-adap.c
> +++ b/drivers/media/cec/cec-adap.c
> @@ -1788,9 +1788,6 @@ static int cec_receive_notify(struct cec_adapter *adap, struct cec_msg *msg,
>  	int la_idx = cec_log_addr2idx(adap, dest_laddr);
>  	bool from_unregistered = init_laddr == 0xf;
>  	struct cec_msg tx_cec_msg = { };
> -#ifdef CONFIG_MEDIA_CEC_RC
> -	int scancode;
> -#endif
>  
>  	dprintk(2, "%s: %*ph\n", __func__, msg->len, msg->msg);
>  
> @@ -1886,9 +1883,11 @@ static int cec_receive_notify(struct cec_adapter *adap, struct cec_msg *msg,
>  		 */
>  		case 0x60:
>  			if (msg->len == 2)
> -				scancode = msg->msg[2];
> +				rc_keydown(adap->rc, RC_PROTO_CEC,
> +					   msg->msg[2], 0);
>  			else
> -				scancode = msg->msg[2] << 8 | msg->msg[3];
> +				rc_keydown(adap->rc, RC_PROTO_CEC,
> +					   msg->msg[2] << 8 | msg->msg[3], 0);
>  			break;
>  		/*
>  		 * Other function messages that are not handled.
> @@ -1901,54 +1900,11 @@ static int cec_receive_notify(struct cec_adapter *adap, struct cec_msg *msg,
>  		 */
>  		case 0x56: case 0x57:
>  		case 0x67: case 0x68: case 0x69: case 0x6a:
> -			scancode = -1;
>  			break;
>  		default:
> -			scancode = msg->msg[2];
> -			break;
> -		}
> -
> -		/* Was repeating, but keypress timed out */
> -		if (adap->rc_repeating && !adap->rc->keypressed) {
> -			adap->rc_repeating = false;
> -			adap->rc_last_scancode = -1;
> -		}
> -		/* Different keypress from last time, ends repeat mode */
> -		if (adap->rc_last_scancode != scancode) {
> -			rc_keyup(adap->rc);
> -			adap->rc_repeating = false;
> -		}
> -		/* We can't handle this scancode */
> -		if (scancode < 0) {
> -			adap->rc_last_scancode = scancode;
> -			break;
> -		}
> -
> -		/* Send key press */
> -		rc_keydown(adap->rc, RC_PROTO_CEC, scancode, 0);
> -
> -		/* When in repeating mode, we're done */
> -		if (adap->rc_repeating)
> -			break;
> -
> -		/*
> -		 * We are not repeating, but the new scancode is
> -		 * the same as the last one, and this second key press is
> -		 * within 550 ms (the 'Follower Safety Timeout') from the
> -		 * previous key press, so we now enable the repeating mode.
> -		 */
> -		if (adap->rc_last_scancode == scancode &&
> -		    msg->rx_ts - adap->rc_last_keypress < 550 * NSEC_PER_MSEC) {
> -			adap->rc_repeating = true;
> +			rc_keydown(adap->rc, RC_PROTO_CEC, msg->msg[2], 0);
>  			break;
>  		}
> -		/*
> -		 * Not in repeating mode, so avoid triggering repeat mode
> -		 * by calling keyup.
> -		 */
> -		rc_keyup(adap->rc);
> -		adap->rc_last_scancode = scancode;
> -		adap->rc_last_keypress = msg->rx_ts;
>  #endif
>  		break;
>  
> @@ -1958,8 +1914,6 @@ static int cec_receive_notify(struct cec_adapter *adap, struct cec_msg *msg,
>  			break;
>  #ifdef CONFIG_MEDIA_CEC_RC
>  		rc_keyup(adap->rc);
> -		adap->rc_repeating = false;
> -		adap->rc_last_scancode = -1;
>  #endif
>  		break;
>  
> diff --git a/drivers/media/cec/cec-core.c b/drivers/media/cec/cec-core.c
> index 5870da6a567f..3c67af511c66 100644
> --- a/drivers/media/cec/cec-core.c
> +++ b/drivers/media/cec/cec-core.c
> @@ -280,7 +280,6 @@ struct cec_adapter *cec_allocate_adapter(const struct cec_adap_ops *ops,
>  	adap->rc->priv = adap;
>  	adap->rc->map_name = RC_MAP_CEC;
>  	adap->rc->timeout = MS_TO_NS(100);
> -	adap->rc_last_scancode = -1;
>  #endif
>  	return adap;
>  }
> @@ -312,17 +311,6 @@ int cec_register_adapter(struct cec_adapter *adap,
>  			adap->rc = NULL;
>  			return res;
>  		}
> -		/*
> -		 * The REP_DELAY for CEC is really the time between the initial
> -		 * 'User Control Pressed' message and the second. The first
> -		 * keypress is always seen as non-repeating, the second
> -		 * (provided it has the same UI Command) will start the 'Press
> -		 * and Hold' (aka repeat) behavior. By setting REP_DELAY to the
> -		 * same value as REP_PERIOD the expected CEC behavior is
> -		 * reproduced.
> -		 */
> -		adap->rc->input_dev->rep[REP_DELAY] =
> -			adap->rc->input_dev->rep[REP_PERIOD];
>  	}
>  #endif
>  
> diff --git a/drivers/media/rc/rc-main.c b/drivers/media/rc/rc-main.c
> index 17950e29d4e3..7860ed50bc7f 100644
> --- a/drivers/media/rc/rc-main.c
> +++ b/drivers/media/rc/rc-main.c
> @@ -599,6 +599,7 @@ static void ir_do_keyup(struct rc_dev *dev, bool sync)
>  		return;
>  
>  	IR_dprintk(1, "keyup key 0x%04x\n", dev->last_keycode);
> +	del_timer_sync(&dev->timer_repeat);
>  	input_report_key(dev->input_dev, dev->last_keycode, 0);
>  	led_trigger_event(led_feedback, LED_OFF);
>  	if (sync)
> @@ -625,7 +626,6 @@ EXPORT_SYMBOL_GPL(rc_keyup);
>  
>  /**
>   * ir_timer_keyup() - generates a keyup event after a timeout
> - * @cookie:	a pointer to the struct rc_dev for the device
>   *
>   * This routine will generate a keyup event some time after a keydown event
>   * is generated when no further activity has been detected.
> @@ -651,6 +651,26 @@ static void ir_timer_keyup(struct timer_list *t)
>  	spin_unlock_irqrestore(&dev->keylock, flags);
>  }
>  
> +/**
> + * ir_timer_repeat() - generates a repeat event after a timeout
> + *
> + * This routine will generate a soft repeat event every REP_PERIOD
> + * milliseconds.
> + */
> +static void ir_timer_repeat(struct timer_list *t)
> +{
> +	struct rc_dev *dev = from_timer(dev, t, timer_repeat);
> +	unsigned long flags;
> +
> +	spin_lock_irqsave(&dev->keylock, flags);
> +	if (dev->keypressed) {
> +		input_event(dev->input_dev, EV_KEY, dev->last_keycode, 2);
> +		mod_timer(&dev->timer_repeat,
> +			  jiffies + dev->input_dev->rep[REP_PERIOD]);
> +	}
> +	spin_unlock_irqrestore(&dev->keylock, flags);
> +}
> +
>  /**
>   * rc_repeat() - signals that a key is still pressed
>   * @dev:	the struct rc_dev descriptor of the device
> @@ -719,6 +739,22 @@ static void ir_do_keydown(struct rc_dev *dev, enum rc_proto protocol,
>  		led_trigger_event(led_feedback, LED_FULL);
>  	}
>  
> +	/*
> +	 * For CEC, start sending repeat messages as soon as the first
> +	 * repeated message is sent, as long as REP_DELAY = 0 and REP_PERIOD
> +	 * is non-zero. Otherwise, the input layer will generate repeat
> +	 * messages.
> +	 */
> +	if (!new_event && keycode != KEY_RESERVED &&
> +	    dev->allowed_protocols == RC_PROTO_BIT_CEC &&
> +	    !timer_pending(&dev->timer_repeat) &&
> +	    dev->input_dev->rep[REP_PERIOD] &&
> +	    !dev->input_dev->rep[REP_DELAY]) {
> +		input_event(dev->input_dev, EV_KEY, keycode, 2);
> +		mod_timer(&dev->timer_repeat, jiffies +
> +			  msecs_to_jiffies(dev->input_dev->rep[REP_PERIOD]));
> +	}
> +
>  	input_sync(dev->input_dev);
>  }
>  
> @@ -1571,6 +1607,7 @@ struct rc_dev *rc_allocate_device(enum rc_driver_type type)
>  		input_set_drvdata(dev->input_dev, dev);
>  
>  		timer_setup(&dev->timer_keyup, ir_timer_keyup, 0);
> +		timer_setup(&dev->timer_repeat, ir_timer_repeat, 0);
>  
>  		spin_lock_init(&dev->rc_map.lock);
>  		spin_lock_init(&dev->keylock);
> @@ -1704,7 +1741,10 @@ static int rc_setup_rx_device(struct rc_dev *dev)
>  	 * to avoid wrong repetition of the keycodes. Note that this must be
>  	 * set after the call to input_register_device().
>  	 */
> -	dev->input_dev->rep[REP_DELAY] = 500;
> +	if (dev->allowed_protocols == RC_PROTO_BIT_CEC)
> +		dev->input_dev->rep[REP_DELAY] = 0;
> +	else
> +		dev->input_dev->rep[REP_DELAY] = 500;
>  
>  	/*
>  	 * As a repeat event on protocols like RC-5 and NEC take as long as
> @@ -1846,6 +1886,7 @@ void rc_unregister_device(struct rc_dev *dev)
>  		return;
>  
>  	del_timer_sync(&dev->timer_keyup);
> +	del_timer_sync(&dev->timer_repeat);
>  
>  	if (dev->driver_type == RC_DRIVER_IR_RAW)
>  		ir_raw_event_unregister(dev);
> diff --git a/include/media/cec.h b/include/media/cec.h
> index 16341210d3ba..5db31bbdf133 100644
> --- a/include/media/cec.h
> +++ b/include/media/cec.h
> @@ -191,11 +191,6 @@ struct cec_adapter {
>  
>  	u32 tx_timeouts;
>  
> -#ifdef CONFIG_MEDIA_CEC_RC
> -	bool rc_repeating;
> -	int rc_last_scancode;
> -	u64 rc_last_keypress;
> -#endif
>  #ifdef CONFIG_CEC_NOTIFIER
>  	struct cec_notifier *notifier;
>  #endif
> diff --git a/include/media/rc-core.h b/include/media/rc-core.h
> index 314a1edb6189..6ba4be82ef73 100644
> --- a/include/media/rc-core.h
> +++ b/include/media/rc-core.h
> @@ -106,6 +106,8 @@ enum rc_filter_type {
>   * @keypressed: whether a key is currently pressed
>   * @keyup_jiffies: time (in jiffies) when the current keypress should be released
>   * @timer_keyup: timer for releasing a keypress
> + * @timer_repeat: timer for autorepeat events. This is needed for CEC, which
> + *	has non-standard repeats.
>   * @last_keycode: keycode of last keypress
>   * @last_protocol: protocol of last keypress
>   * @last_scancode: scancode of last keypress
> @@ -165,6 +167,7 @@ struct rc_dev {
>  	bool				keypressed;
>  	unsigned long			keyup_jiffies;
>  	struct timer_list		timer_keyup;
> +	struct timer_list		timer_repeat;
>  	u32				last_keycode;
>  	enum rc_proto			last_protocol;
>  	u32				last_scancode;
> 

Acked-by: Hans Verkuil <hans.verkuil@xxxxxxxxx>

Thanks!

	Hans



[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