Re: [PATCH 2/2] seco-cec: add Consumer-IR support

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

 



Hi Ettore,

On Thu, Oct 04, 2018 at 11:46:45PM +0200, ektor5 wrote:
> Hi Sean,
> 
> On Thu, Oct 04, 2018 at 02:49:27PM +0100, Sean Young wrote:
> > On Tue, Oct 02, 2018 at 06:59:56PM +0200, ektor5 wrote:
> > > From: Ettore Chimenti <ek5.chimenti@xxxxxxxxx>
> > > 
> > > 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.chimenti@xxxxxxxxx>
> > > ---
> > >  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.
> 
> Ops. Yes, will fix.
> 
> > 
> > > +
> > >  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).
> 
> Yes, I figured out that in the middle of developing. I got rid of most
> of the "irda" references, but I have accidentally left some of them.
> Will finish the work.
> 
> > 
> > >  	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.
> 
> Ok, will remove.
> 
> > 
> > > +		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.
> 
> I took that from CEC RC Passthrough device name.

Thanks for pointing that out, I've just submitted a patch to fix that one
as well:

	https://patchwork.linuxtv.org/patch/52376/

Hans, what do you think?

> 
> > 
> > > +	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.
> 
> Ok.
> 
> > 
> > > +	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.
> 
> I wondered why the address wasn't used, it had to be together. Thanks. :)
> 
> > 
> > > +
> > > +	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
> 
> Thanks a lot,
> 	Ettore

Thanks,
Sean



[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