Received: by 2002:ac0:a582:0:0:0:0:0 with SMTP id m2-v6csp1001913imm; Thu, 4 Oct 2018 06:50:17 -0700 (PDT) X-Google-Smtp-Source: ACcGV62bzkaQe0OYcUBfZ8RHJiB7iGhFaEvSHAMLBt+6xP6pVN9Ppql0pjBdK+oI5oO0sq7o8klJ X-Received: by 2002:a17:902:82c9:: with SMTP id u9-v6mr6700604plz.320.1538661017865; Thu, 04 Oct 2018 06:50:17 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1538661017; cv=none; d=google.com; s=arc-20160816; b=dDp7gSIBGGntl+2IIa9DrWLrgFpK2Xm/JB4QEp03rco0gRBzP51X+Lyti0fsOPfE1N HzS9FONandKxT35quqRbiwR5uyNDqjM6m1xXZXdbTf5iHquM/TjkGa035vvk+8fK5gkF PeSbNwBom4mJoYtJhUe2L24Hs3Q2vA6SMOV4pL6fcW+6GcDAEXgcwk/Zd7sENRZMNnnq dmHYJPQywhS3tJMhk2nyRaQMaTwbgZYsNxtTDw/rz5N3ADkSu8D5ObqPGbGYq9kB0nNK zpVuGvJn8Y0Soa3YCb4VqIZppVMRSvY+46Qzsx+DRwWVO3ifig/aBt8M5ywRWSlEb6sR H5Vg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-disposition:mime-version:references:message-id:subject:cc :to:from:date; bh=wsVbRyANWTMIRF1A6YtNCq9Uq4OSF6e8kVzmzapxGiE=; b=GOQbrW0+JKs89bD+028ZqdrBtf3AXIvKG6bdBihYN3KQ23YwsBtb/M4LKNQw5clNRp k3kmSs3bQber7J4mp09zuQp0LSFyCNNWtVMkyhxb8OGkZr6kgch7iwsPIn4sMJ14yqPT RIS+yqsLEmNh914aVolg3OXovVZP/qF6nSPreuo0S5Y1P1+/FHcmr/UB74MxzaBpq+Aq 0O3Z1bhkwMrVa58cyqRaRgZTJdXo3HdPPhLY4GEmLiQnqG1cngahchMBtbpCY2gA7N31 FUujsQ1mCK6MlVAOU3yMDJEvfM4Ui5Wm6AVpv9v33fBEkUeMBAMGygGcuWoBW/+XAePC y4aw== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id f88-v6si5812715pfh.33.2018.10.04.06.50.02; Thu, 04 Oct 2018 06:50:17 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727644AbeJDUmz (ORCPT + 99 others); Thu, 4 Oct 2018 16:42:55 -0400 Received: from gofer.mess.org ([88.97.38.141]:49247 "EHLO gofer.mess.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727354AbeJDUmy (ORCPT ); Thu, 4 Oct 2018 16:42:54 -0400 Received: by gofer.mess.org (Postfix, from userid 1000) id EED92601C8; Thu, 4 Oct 2018 14:49:27 +0100 (BST) Date: Thu, 4 Oct 2018 14:49:27 +0100 From: Sean Young To: ektor5 Cc: hverkuil@xs4all.nl, luca.pisani@udoo.org, jose.abreu@synopsys.com, sakari.ailus@linux.intel.com, Mauro Carvalho Chehab , Greg Kroah-Hartman , "David S. Miller" , Andrew Morton , Arnd Bergmann , Hans Verkuil , Laurent Pinchart , Geert Uytterhoeven , Jacob Chen , Todor Tomov , Kate Stewart , Jacopo Mondi , Neil Armstrong , linux-kernel@vger.kernel.org, linux-media@vger.kernel.org Subject: Re: [PATCH 2/2] seco-cec: add Consumer-IR support Message-ID: <20181004134927.ox7alorufq56f2ux@gofer.mess.org> References: MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: User-Agent: NeoMutt/20170113 (1.7.2) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Tue, Oct 02, 2018 at 06:59:56PM +0200, ektor5 wrote: > From: Ettore Chimenti > > 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 > --- > drivers/media/platform/Kconfig | 10 ++ > drivers/media/platform/seco-cec/seco-cec.c | 136 ++++++++++++++++++++- > drivers/media/platform/seco-cec/seco-cec.h | 11 ++ > 3 files changed, 154 insertions(+), 3 deletions(-) > > diff --git a/drivers/media/platform/Kconfig b/drivers/media/platform/Kconfig > index f477764b902a..5833f488eef8 100644 > --- a/drivers/media/platform/Kconfig > +++ b/drivers/media/platform/Kconfig > @@ -624,6 +624,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. Strange mixture of spaces/tabs. > + > 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 ba3b7c144a87..ee1949395cf4 100644 > --- a/drivers/media/platform/seco-cec/seco-cec.c > +++ b/drivers/media/platform/seco-cec/seco-cec.c > @@ -28,6 +28,9 @@ struct secocec_data { > struct platform_device *pdev; > struct cec_adapter *cec_adap; > struct cec_notifier *notifier; > + struct rc_dev *irda_rc; > + char irda_input_name[32]; > + char irda_input_phys[32]; IrDA is a completely different encoding than RC-5, CIR or anything rc-core supports; RC-5 is much lower transmission speed. Please do not conflate the two, and rename it either ir_input_phys or rc_input_phys (same for the rest of the functions/members in the rest of the file). > int irq; > }; > > @@ -383,6 +386,119 @@ struct cec_adap_ops secocec_cec_adap_ops = { > .adap_transmit = secocec_adap_transmit, > }; > > +#ifdef CONFIG_VIDEO_SECO_RC > +static int secocec_irda_probe(void *priv) > +{ > + struct secocec_data *cec = priv; > + struct device *dev = cec->dev; > + int status; > + u16 val; > + > + /* Prepare the RC input device */ > + cec->irda_rc = devm_rc_allocate_device(dev, RC_DRIVER_SCANCODE); > + if (!cec->irda_rc) { > + dev_err(dev, "Failed to allocate memory for rc_dev"); No need to dev_err() here, kmalloc() will have already reported the error. > + return -ENOMEM; > + } > + > + snprintf(cec->irda_input_name, sizeof(cec->irda_input_name), > + "IrDA RC for %s", dev_name(dev)); Since it's an RC device there is no need to put RC in the name. Just use dev_name() as the device_name. > + snprintf(cec->irda_input_phys, sizeof(cec->irda_input_phys), > + "%s/input0", dev_name(dev)); > + > + cec->irda_rc->device_name = cec->irda_input_name; > + cec->irda_rc->input_phys = cec->irda_input_phys; > + cec->irda_rc->input_id.bustype = BUS_HOST; > + cec->irda_rc->input_id.vendor = 0; > + cec->irda_rc->input_id.product = 0; > + cec->irda_rc->input_id.version = 1; > + cec->irda_rc->driver_name = SECOCEC_DEV_NAME; > + cec->irda_rc->allowed_protocols = RC_PROTO_BIT_RC5; > + cec->irda_rc->enabled_protocols = RC_PROTO_BIT_RC5; No need to set enabled_protocols. > + cec->irda_rc->priv = cec; > + cec->irda_rc->map_name = RC_MAP_HAUPPAUGE; > + cec->irda_rc->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->irda_rc); > + > + if (status) { > + dev_err(dev, "Failed to prepare input device"); > + cec->irda_rc = 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_irda_rx(struct secocec_data *priv) > +{ > + struct secocec_data *cec = priv; > + struct device *dev = cec->dev; > + u16 val, status, key, addr, toggle; > + > + if (!cec->irda_rc) > + 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->irda_rc, RC_PROTO_RC5, key, toggle); Here you are just reported the key, not the address. Please use: rc_keydown(cec->rc, RC_PROTO_RC5, RC_SCANCODE_RC5(addr, key), toggle); In fact, you could do: rc_keydown(cec->rc, RC_PROTO_RC5, val & 0x1f7f, toggle); I presume the compile is clever enough to fold those shift instructions. > + > + 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_irda_rx(struct secocec_data *priv) > +{ > +} > + > +static int secocec_irda_probe(void *priv) > +{ > + return 0; > +} > +#endif > + > static irqreturn_t secocec_irq_handler(int irq, void *priv) > { > struct secocec_data *cec = priv; > @@ -420,7 +536,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_irda_rx(cec); > } > > /* Reset status register */ > @@ -595,6 +712,10 @@ static int secocec_probe(struct platform_device *pdev) > if (secocec->notifier) > cec_register_cec_notifier(secocec->cec_adap, secocec->notifier); > > + ret = secocec_irda_probe(secocec); > + if (ret) > + goto err_delete_adapter; > + > platform_set_drvdata(pdev, secocec); > > dev_dbg(dev, "Device registered"); > @@ -614,7 +735,16 @@ 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->irda_rc) { > + smb_rd16(SECOCEC_ENABLE_REG_1, &val); > > + smb_wr16(SECOCEC_ENABLE_REG_1, > + val & ~SECOCEC_ENABLE_REG_1_IR); Those two fit on one line. > + > + dev_dbg(&pdev->dev, "IR disabled"); > + } > cec_unregister_adapter(secocec->cec_adap); > > if (secocec->notifier) > @@ -632,8 +762,8 @@ static int secocec_remove(struct platform_device *pdev) > #ifdef CONFIG_PM_SLEEP > static int secocec_suspend(struct device *dev) > { > - u16 val; > int status; > + u16 val; > > dev_dbg(dev, "Device going to suspend, disabling"); > > @@ -665,8 +795,8 @@ static int secocec_suspend(struct device *dev) > > static int secocec_resume(struct device *dev) > { > - u16 val; > int status; > + u16 val; > > dev_dbg(dev, "Resuming device from suspend"); > > diff --git a/drivers/media/platform/seco-cec/seco-cec.h b/drivers/media/platform/seco-cec/seco-cec.h > index cc7f0cba8e9e..c00660104a3e 100644 > --- a/drivers/media/platform/seco-cec/seco-cec.h > +++ b/drivers/media/platform/seco-cec/seco-cec.h > @@ -101,6 +101,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 Thanks, Sean