diff mbox

[0/3] Improve CEC autorepeat handling

Message ID 20171127094724.6cjl6zex46y6wgfd@gofer.mess.org (mailing list archive)
State New, archived
Headers show

Commit Message

Sean Young Nov. 27, 2017, 9:47 a.m. UTC
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@mess.org>
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@mess.org>
---
 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(-)

Comments

Hans Verkuil Nov. 27, 2017, 10:01 a.m. UTC | #1
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.

Ah yes, that's true. But it still is specific to the CEC protocol, the code
handling that would just move to rc-core from cec-adap.c

> 
> 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.

That looks nice! I'm fine with that.

Regards,

	Hans

> 
> Thanks,
> Sean
> ----
> From 3f439e326888a0ab8688d73c4276ac87b4225b1c Mon Sep 17 00:00:00 2001
> From: Sean Young <sean@mess.org>
> 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@mess.org>
> ---
>  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;
> 

--
To unsubscribe from this list: send the line "unsubscribe linux-input" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Dmitry Torokhov Nov. 27, 2017, 10:11 a.m. UTC | #2
On Mon, Nov 27, 2017 at 09:47:24AM +0000, 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.

Not allowed by whom? If I, as a user, want my remote to start
autorepeating after 400 msec instead of 550, will the police come and
fine me? Please do not confuse the default behavior with allowed one.
The only restriction is that if you have not seen messages for longer
than 550 msecs you should "release" the key.

> > 
> > 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.

Right, and they do not have to match autorepeat timings, just control
whether we should continue repeating or generate release event.

> > 
> > > 
> > >>
> > >> 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.

You could do that by catching EV_REP events in your driver, but I do not
see why you want to remove this flexibility from users. Again, if I want
to wait for autorepeat to start for 10 seconds, or 10 msecs, it is my
choice. You are to provide the default, I am to override it if I want.

> 
> 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.

I have no idea why you simply not rely on current input core autorepeat
handling, set REP_DELAY to be 550 msec and call it a day.

Thanks.
Hans Verkuil Nov. 27, 2017, 3:43 p.m. UTC | #3
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@mess.org>
> 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@mess.org>
> ---
>  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@cisco.com>

Thanks!

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

Patch

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;