RE: [PATCH v2 2/2] dmaengine: usb-dmac: Add Renesas USB DMA Controller (USB-DMAC) driver

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

 




Hi Vinod,

Thank you for your review!

> On Wed, Mar 11, 2015 at 02:39:54PM +0900, Yoshihiro Shimoda wrote:
> > +static struct dma_async_tx_descriptor *
> > +usb_dmac_prep_slave_sg(struct dma_chan *chan, struct scatterlist *sgl,
> > +		       unsigned int sg_len, enum dma_transfer_direction dir,
> > +		       unsigned long dma_flags, void *context)
> > +{
> > +	struct usb_dmac_chan *uchan = to_usb_dmac_chan(chan);
> > +	struct usb_dmac_desc *desc;
> > +
> > +	/* A client driver will use dmaengine_prep_slave_single() */
> > +	if (sg_len != 1) {
> and why is that?

This is because that a usb peripheral driver (this is one of a client driver)
will handle a single buffer to transfer data. So, the usb peripheral driver will
use dmaengine_prep_slave_single().
Also, if this usb-dmac driver handles sg_len == 1 only, this driver code becomes more simple.

> > +static enum dma_status usb_dmac_tx_status(struct dma_chan *chan,
> > +					  dma_cookie_t cookie,
> > +					  struct dma_tx_state *txstate)
> > +{
> > +	struct usb_dmac_chan *uchan = to_usb_dmac_chan(chan);
> > +	enum dma_status status;
> > +	unsigned long flags;
> > +	unsigned int residue;
> > +
> > +	status = dma_cookie_status(chan, cookie, txstate);
> > +	/* a client driver will get residue after DMA_COMPLETE */
> can you explain this comment?

I'm sorry for the lack comment. This is related to the following:
http://thread.gmane.org/gmane.linux.drivers.devicetree/109578/focus=44468
======= extract from the email ===========
This USB-DMAC has a function to detect a USB specific packet (called short-length-packet).
If the USB-DMAC detects it, the USB-DMAC assumes the USB-DMAC completes the transfer.
==========================================

So, a usb peripheral driver (this is one of a client driver) will call
dmaengine_tx_status() after the USB-DMAC completes the transfer.

> > +	if (!txstate)
> > +		return status;
> > +
> > +	spin_lock_irqsave(&uchan->vc.lock, flags);
> > +	residue = uchan->residue;
> how is this related to the cookie for which request is made?

This implementation is not related to the cookie...
This is because that this driver has no chance to get the descriptor
after the driver calls vchan_cookie_complete().

> > +static void usb_dmac_chan_tasklet(unsigned long data)
> > +{
> > +	struct usb_dmac_chan *chan = (struct usb_dmac_chan *)data;
> > +	struct usb_dmac_desc *desc = chan->desc;
> > +
> > +	spin_lock_irq(&chan->vc.lock);
> > +
> > +	/* This driver assumes a transfer finishes if desc is not NULL */
> > +	if (desc) {
> > +		chan->residue = usb_dmac_chan_get_last_residue(chan, desc);
> why should this be for channel and not the descriptor which is completed, so
> you overwrite this before user has chance to query ?

This is related to my comment above. After the driver calls vchan_cookie_complete(),
the descriptor is freed. So, should this driver has "done" descriptors or something
to store the residue?

> Looking at this, so you issue a transaction of 100bytes, will you get
> completion even if 100bytes have not been transfered, If so how does dma
> decide theat transfer is complete even if 100bytes are not transferred yet?

If the USB-DMAC detects a special interrupt status (short-length-packet),
this driver decides that transfer is complete even if 100bytes are not transferred yet.

> > +		vchan_cookie_complete(&desc->vd);
> > +	}
> > +
> > +	/* (Re)start the next transfer if this driver has a next desc */
> > +	usb_dmac_chan_start_xfer(chan);
> why not do this in irq :)

Oops, I assumed that a dmaengine driver must use a tasklet.
However, since I think this function is not heavy, I will move this in irq.

> > +#ifdef CONFIG_PM
> > +static int usb_dmac_runtime_suspend(struct device *dev)
> > +{
> > +	struct usb_dmac *dmac = dev_get_drvdata(dev);
> > +	int i;
> > +
> > +	for (i = 0; i < dmac->n_channels; ++i)
> > +		usb_dmac_chan_halt(&dmac->channels[i]);
> > +
> > +	return 0;
> > +}
> > +
> > +static int usb_dmac_runtime_resume(struct device *dev)
> > +{
> > +	struct usb_dmac *dmac = dev_get_drvdata(dev);
> > +
> > +	return usb_dmac_init(dmac);
> > +}
> > +#endif
> > +
> > +static const struct dev_pm_ops usb_dmac_pm = {
> > +	SET_RUNTIME_PM_OPS(usb_dmac_runtime_suspend, usb_dmac_runtime_resume,
> since you are using SET_RUNTIME_PM_OPS, you dont need to wrap this with
> ifdef CONFIG_PM

Thank you for the point. I will fix it.

> > +
> > +static void usb_dmac_chan_remove(struct usb_dmac_chan *uchan)
> > +{
> > +	tasklet_kill(&uchan->task);
> that part is good, but how about disabling irq? you can still get insterrupt

Thank you for the point! I will add calling usb_dmac_chan_halt() to disable the interrupt.

Best regards,
Yoshihiro Shimoda

> --
> ~Vinod

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




[Index of Archives]     [Device Tree Compilter]     [Device Tree Spec]     [Linux Driver Backports]     [Video for Linux]     [Linux USB Devel]     [Linux PCI Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Yosemite Backpacking]
  Powered by Linux