On Fri, Oct 05, 2018 at 07:33:59PM +0200, ektor5 wrote:
> From: Ettore Chimenti <ek5.chime...@gmail.com>
> 
> Introduce support for Consumer-IR into seco-cec driver, as it shares the
> same interrupt for receiving messages.
> The device decodes RC5 signals only, defaults to hauppauge mapping.
> It will spawn an input interface using the RC framework (like CEC
> device).
> 
> Signed-off-by: Ettore Chimenti <ek5.chime...@gmail.com>

Looks great, thanks!

Reviewed-by: Sean Young <s...@mess.org>

> ---
>  drivers/media/platform/Kconfig             |  10 ++
>  drivers/media/platform/seco-cec/seco-cec.c | 125 ++++++++++++++++++++-
>  drivers/media/platform/seco-cec/seco-cec.h |  11 ++
>  3 files changed, 145 insertions(+), 1 deletion(-)
> 
> diff --git a/drivers/media/platform/Kconfig b/drivers/media/platform/Kconfig
> index 51cd1fd005e3..e6b45da2af6d 100644
> --- a/drivers/media/platform/Kconfig
> +++ b/drivers/media/platform/Kconfig
> @@ -625,6 +625,16 @@ config VIDEO_SECO_CEC
>         CEC bus is present in the HDMI connector and enables communication
>         between compatible devices.
>  
> +config VIDEO_SECO_RC
> +     bool "SECO Boards IR RC5 support"
> +     depends on VIDEO_SECO_CEC
> +     select RC_CORE
> +     help
> +       If you say yes here you will get support for the
> +       SECO Boards Consumer-IR in seco-cec driver.
> +       The embedded controller supports RC5 protocol only, default mapping
> +       is set to rc-hauppauge.
> +
>  endif #CEC_PLATFORM_DRIVERS
>  
>  menuconfig SDR_PLATFORM_DRIVERS
> diff --git a/drivers/media/platform/seco-cec/seco-cec.c 
> b/drivers/media/platform/seco-cec/seco-cec.c
> index 990e88f979a2..6ae42fc855ef 100644
> --- a/drivers/media/platform/seco-cec/seco-cec.c
> +++ b/drivers/media/platform/seco-cec/seco-cec.c
> @@ -26,6 +26,8 @@ struct secocec_data {
>       struct platform_device *pdev;
>       struct cec_adapter *cec_adap;
>       struct cec_notifier *notifier;
> +     struct rc_dev *ir;
> +     char ir_input_phys[32];
>       int irq;
>  };
>  
> @@ -371,6 +373,114 @@ struct cec_adap_ops secocec_cec_adap_ops = {
>       .adap_transmit = secocec_adap_transmit,
>  };
>  
> +#ifdef CONFIG_VIDEO_SECO_RC
> +static int secocec_ir_probe(void *priv)
> +{
> +     struct secocec_data *cec = priv;
> +     struct device *dev = cec->dev;
> +     int status;
> +     u16 val;
> +
> +     /* Prepare the RC input device */
> +     cec->ir = devm_rc_allocate_device(dev, RC_DRIVER_SCANCODE);
> +     if (!cec->ir)
> +             return -ENOMEM;
> +
> +     snprintf(cec->ir_input_phys, sizeof(cec->ir_input_phys),
> +              "%s/input0", dev_name(dev));
> +
> +     cec->ir->device_name = dev_name(dev);
> +     cec->ir->input_phys = cec->ir_input_phys;
> +     cec->ir->input_id.bustype = BUS_HOST;
> +     cec->ir->input_id.vendor = 0;
> +     cec->ir->input_id.product = 0;
> +     cec->ir->input_id.version = 1;
> +     cec->ir->driver_name = SECOCEC_DEV_NAME;
> +     cec->ir->allowed_protocols = RC_PROTO_BIT_RC5;
> +     cec->ir->priv = cec;
> +     cec->ir->map_name = RC_MAP_HAUPPAUGE;
> +     cec->ir->timeout = MS_TO_NS(100);
> +
> +     /* Clear the status register */
> +     status = smb_rd16(SECOCEC_STATUS_REG_1, &val);
> +     if (status != 0)
> +             goto err;
> +
> +     status = smb_wr16(SECOCEC_STATUS_REG_1, val);
> +     if (status != 0)
> +             goto err;
> +
> +     /* Enable the interrupts */
> +     status = smb_rd16(SECOCEC_ENABLE_REG_1, &val);
> +     if (status != 0)
> +             goto err;
> +
> +     status = smb_wr16(SECOCEC_ENABLE_REG_1,
> +                       val | SECOCEC_ENABLE_REG_1_IR);
> +     if (status != 0)
> +             goto err;
> +
> +     dev_dbg(dev, "IR enabled");
> +
> +     status = devm_rc_register_device(dev, cec->ir);
> +
> +     if (status) {
> +             dev_err(dev, "Failed to prepare input device");
> +             cec->ir = NULL;
> +             goto err;
> +     }
> +
> +     return 0;
> +
> +err:
> +     smb_rd16(SECOCEC_ENABLE_REG_1, &val);
> +
> +     smb_wr16(SECOCEC_ENABLE_REG_1,
> +              val & ~SECOCEC_ENABLE_REG_1_IR);
> +
> +     dev_dbg(dev, "IR disabled");
> +     return status;
> +}
> +
> +static int secocec_ir_rx(struct secocec_data *priv)
> +{
> +     struct secocec_data *cec = priv;
> +     struct device *dev = cec->dev;
> +     u16 val, status, key, addr, toggle;
> +
> +     if (!cec->ir)
> +             return -ENODEV;
> +
> +     status = smb_rd16(SECOCEC_IR_READ_DATA, &val);
> +     if (status != 0)
> +             goto err;
> +
> +     key = val & SECOCEC_IR_COMMAND_MASK;
> +     addr = (val & SECOCEC_IR_ADDRESS_MASK) >> SECOCEC_IR_ADDRESS_SHL;
> +     toggle = (val & SECOCEC_IR_TOGGLE_MASK) >> SECOCEC_IR_TOGGLE_SHL;
> +
> +     rc_keydown(cec->ir, RC_PROTO_RC5, RC_SCANCODE_RC5(addr, key), toggle);
> +
> +     dev_dbg(dev, "IR key pressed: 0x%02x addr 0x%02x toggle 0x%02x", key,
> +             addr, toggle);
> +
> +     return 0;
> +
> +err:
> +     dev_err(dev, "IR Receive message failed (%d)", status);
> +     return -EIO;
> +}
> +#else
> +static void secocec_ir_rx(struct secocec_data *priv)
> +{
> +}
> +
> +static int secocec_ir_probe(void *priv)
> +{
> +     return 0;
> +}
> +#endif
> +
>  static irqreturn_t secocec_irq_handler(int irq, void *priv)
>  {
>       struct secocec_data *cec = priv;
> @@ -406,7 +516,8 @@ static irqreturn_t secocec_irq_handler(int irq, void 
> *priv)
>       if (status_val & SECOCEC_STATUS_REG_1_IR) {
>               dev_dbg(dev, "IR RC5 Interrupt Caught");
>               val |= SECOCEC_STATUS_REG_1_IR;
> -             /* TODO IRDA RX */
> +
> +             secocec_ir_rx(cec);
>       }
>  
>       /*  Reset status register */
> @@ -576,6 +687,10 @@ static int secocec_probe(struct platform_device *pdev)
>       if (secocec->notifier)
>               cec_register_cec_notifier(secocec->cec_adap, secocec->notifier);
>  
> +     ret = secocec_ir_probe(secocec);
> +     if (ret)
> +             goto err_delete_adapter;
> +
>       platform_set_drvdata(pdev, secocec);
>  
>       dev_dbg(dev, "Device registered");
> @@ -593,7 +708,15 @@ static int secocec_probe(struct platform_device *pdev)
>  static int secocec_remove(struct platform_device *pdev)
>  {
>       struct secocec_data *secocec = platform_get_drvdata(pdev);
> +     u16 val;
> +
> +     if (secocec->ir) {
> +             smb_rd16(SECOCEC_ENABLE_REG_1, &val);
>  
> +             smb_wr16(SECOCEC_ENABLE_REG_1, val & ~SECOCEC_ENABLE_REG_1_IR);
> +
> +             dev_dbg(&pdev->dev, "IR disabled");
> +     }
>       cec_unregister_adapter(secocec->cec_adap);
>  
>       if (secocec->notifier)
> diff --git a/drivers/media/platform/seco-cec/seco-cec.h 
> b/drivers/media/platform/seco-cec/seco-cec.h
> index 93020900935e..3f1aad89b073 100644
> --- a/drivers/media/platform/seco-cec/seco-cec.h
> +++ b/drivers/media/platform/seco-cec/seco-cec.h
> @@ -99,6 +99,17 @@
>  
>  #define SECOCEC_IR_READ_DATA         0x3e
>  
> +/*
> + * IR
> + */
> +
> +#define SECOCEC_IR_COMMAND_MASK              0x007F
> +#define SECOCEC_IR_COMMAND_SHL               0
> +#define SECOCEC_IR_ADDRESS_MASK              0x1F00
> +#define SECOCEC_IR_ADDRESS_SHL               7
> +#define SECOCEC_IR_TOGGLE_MASK               0x8000
> +#define SECOCEC_IR_TOGGLE_SHL                15
> +
>  /*
>   * Enabling register
>   */
> -- 
> 2.18.0

Reply via email to