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 <s...@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 <s...@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;
-- 
2.14.3


Reply via email to