Re: [PATCH 3/3] USB: EXYNOS USB3.0 device controller driver

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

 



Hi,

On Mon, Feb 06, 2012 at 05:13:28PM +0900, Anton Tikhomirov wrote:
> Cc: Kukjin Kim <kgene.kim <at> samsung.com>
> Cc: Greg Kroah-Hartman <gregkh <at> suse.de>
> Cc: Felipe Balbi <balbi <at> ti.com>

you can add the correct email, actually, but they should follow your
Signed-off-by line.

> 
> Adds Exynos USB3.0 device controller driver to support USB peripherals
> on Exynos5 chips.
> 
> Signed-off-by: Anton Tikhomirov <av.tikhomirov@xxxxxxxxxxx>
> ---
>  drivers/usb/gadget/Kconfig         |   13 +
>  drivers/usb/gadget/Makefile        |    1 +
>  drivers/usb/gadget/exynos_ss_udc.c | 2511 ++++++++++++++++++++++++++++++++++++
>  drivers/usb/gadget/exynos_ss_udc.h |  342 +++++
>  4 files changed, 2867 insertions(+), 0 deletions(-)
>  create mode 100644 drivers/usb/gadget/exynos_ss_udc.c
>  create mode 100644 drivers/usb/gadget/exynos_ss_udc.h
> 
> diff --git a/drivers/usb/gadget/Kconfig b/drivers/usb/gadget/Kconfig
> index 7ecb68a..8d3bc6e 100644
> --- a/drivers/usb/gadget/Kconfig
> +++ b/drivers/usb/gadget/Kconfig
> @@ -268,6 +268,19 @@ config USB_S3C_HSOTG
>  	  The Samsung S3C64XX USB2.0 high-speed gadget controller
>  	  integrated into the S3C64XX series SoC.
>  
> +config USB_EXYNOS_SS_UDC
> +	tristate "EXYNOS SuperSpeed USB 3.0 Device controller"
> +	depends on EXYNOS_DEV_SS_UDC
> +	select USB_GADGET_DUALSPEED
> +	select USB_GADGET_SUPERSPEED
> +	help
> +	  The Samsung Exynos SuperSpeed USB 3.0 device controller
> +	  integrated into the Exynos5 series SoC.
> +
> +	  Say "y" to link the driver statically, or "m" to build a
> +	  dynamically linked module called "exynos_ss_udc" and force all
> +	  gadget drivers to also be dynamically linked.
> +
>  config USB_IMX
>  	tristate "Freescale i.MX1 USB Peripheral Controller"
>  	depends on ARCH_MXC
> diff --git a/drivers/usb/gadget/Makefile b/drivers/usb/gadget/Makefile
> index b7f6eef..092bc76 100644
> --- a/drivers/usb/gadget/Makefile
> +++ b/drivers/usb/gadget/Makefile
> @@ -25,6 +25,7 @@ obj-$(CONFIG_USB_FSL_QE)	+= fsl_qe_udc.o
>  obj-$(CONFIG_USB_CI13XXX_PCI)	+= ci13xxx_pci.o
>  obj-$(CONFIG_USB_S3C_HSOTG)	+= s3c-hsotg.o
>  obj-$(CONFIG_USB_S3C_HSUDC)	+= s3c-hsudc.o
> +obj-$(CONFIG_USB_EXYNOS_SS_UDC) += exynos_ss_udc.o
>  obj-$(CONFIG_USB_LANGWELL)	+= langwell_udc.o
>  obj-$(CONFIG_USB_EG20T)		+= pch_udc.o
>  obj-$(CONFIG_USB_MV_UDC)	+= mv_udc.o
> diff --git a/drivers/usb/gadget/exynos_ss_udc.c b/drivers/usb/gadget/exynos_ss_udc.c
> new file mode 100644
> index 0000000..aaad33c
> --- /dev/null
> +++ b/drivers/usb/gadget/exynos_ss_udc.c
> @@ -0,0 +1,2511 @@
> +/* linux/drivers/usb/gadget/exynos_ss_udc.c
> + *
> + * Copyright 2011 Samsung Electronics Co., Ltd.
> + *	http://www.samsung.com/
> + *
> + * EXYNOS SuperSpeed USB 3.0 Device Controlle driver
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#include <linux/kernel.h>
> +#include <linux/module.h>
> +#include <linux/spinlock.h>
> +#include <linux/interrupt.h>
> +#include <linux/platform_device.h>
> +#include <linux/dma-mapping.h>
> +#include <linux/delay.h>
> +#include <linux/io.h>
> +#include <linux/slab.h>
> +#include <linux/clk.h>
> +
> +#include <linux/usb/ch9.h>
> +#include <linux/usb/gadget.h>
> +
> +#include <asm/byteorder.h>
> +
> +#include <mach/map.h>
> +
> +#include <plat/regs-usb3-exynos-drd.h>
> +#include <plat/udc-ss.h>
> +#include <plat/usb-phy.h>

do _NOT_ depend on any arch-specific crap please. Registers can be
defined on the C source code and other stuff could sit on <linux/usb/>
or <linux/platform_data/> depending on what it is. I want to be able to
compile this with whatever (cross) compiler I have available.

> +#include "exynos_ss_udc.h"
> +
> +static void exynos_ss_udc_kill_all_requests(struct exynos_ss_udc *udc,
> +					    struct exynos_ss_udc_ep *udc_ep,
> +					    int result);
> +static void exynos_ss_udc_complete_setup(struct usb_ep *ep,
> +					 struct usb_request *req);
> +static void exynos_ss_udc_complete_request(struct exynos_ss_udc *udc,
> +					   struct exynos_ss_udc_ep *udc_ep,
> +					   struct exynos_ss_udc_req *udc_req,
> +					   int result);
> +static void exynos_ss_udc_start_req(struct exynos_ss_udc *udc,
> +				    struct exynos_ss_udc_ep *udc_ep,
> +				    struct exynos_ss_udc_req *udc_req,
> +				    bool continuing);
> +static void exynos_ss_udc_ep_activate(struct exynos_ss_udc *udc,
> +				      struct exynos_ss_udc_ep *udc_ep);
> +static void exynos_ss_udc_ep_deactivate(struct exynos_ss_udc *udc,
> +					struct exynos_ss_udc_ep *udc_ep);
> +static int exynos_ss_udc_start(struct usb_gadget *gadget,
> +			       struct usb_gadget_driver *driver);
> +static int exynos_ss_udc_stop(struct usb_gadget *gadget,
> +			      struct usb_gadget_driver *driver);

if you reorder the driver, you will not need to prototype everything
here.

> +
> +

one blank line is enough.

> +/**
> + * on_list - check request is on the given endpoint
> + * @ep: The endpoint to check.
> + * @test: The request to test if it is on the endpoint.
> + */
> +static bool on_list(struct exynos_ss_udc_ep *udc_ep,
> +		    struct exynos_ss_udc_req *test)
> +{
> +	struct exynos_ss_udc_req *udc_req, *treq;
> +
> +	list_for_each_entry_safe(udc_req, treq, &udc_ep->queue, queue) {
> +		if (udc_req == test)
> +			return true;
> +	}
> +
> +	return false;
> +}

why don't you just hold a pointer to the endpoint on your
exynos_ss_udc_req structure ? Set that pointer on usb_ep_alloc() time
and all you have to do is compare a pointer, rather than traversing the
entire ep list.

> +/**
> + * exynos_ss_udc_map_dma - map the DMA memory being used for the request
> + * @udc: The device state.
> + * @udc_ep: The endpoint the request is on.
> + * @req: The request being processed.
> + *
> + * We've been asked to queue a request, so ensure that the memory buffer
> + * is correctly setup for DMA. If we've been passed an extant DMA address
> + * then ensure the buffer has been synced to memory. If our buffer has no
> + * DMA memory, then we map the memory and mark our request to allow us to
> + * cleanup on completion.
> + */
> +static int exynos_ss_udc_map_dma(struct exynos_ss_udc *udc,
> +				 struct exynos_ss_udc_ep *udc_ep,
> +				 struct usb_request *req)
> +{
> +	enum dma_data_direction dir;
> +	struct exynos_ss_udc_req *udc_req = our_req(req);
> +
> +	dir = udc_ep->dir_in ? DMA_TO_DEVICE : DMA_FROM_DEVICE;
> +
> +	/* if the length is zero, ignore the DMA data */
> +	if (udc_req->req.length == 0)
> +		return 0;
> +
> +	if (req->dma == DMA_ADDR_INVALID) {

NAK, we want to remove this. Gadget drivers should _never_ map buffer by
themselves. Just go ahead and map it.

> +		dma_addr_t dma;
> +
> +		dma = dma_map_single(udc->dev,
> +				     req->buf, req->length, dir);
> +
> +		if (unlikely(dma_mapping_error(udc->dev, dma)))
> +			goto dma_error;
> +
> +		udc_req->mapped = 1;
> +		req->dma = dma;
> +	} else
> +		dma_sync_single_for_device(udc->dev,
> +					   req->dma, req->length, dir);

if one branch has {}, both should have, but this will be dropped because
of reason above.

> +
> +	return 0;

beware that we will be providing generic map/unmap routines soonish.

> +dma_error:
> +	dev_err(udc->dev, "%s: failed to map buffer %p, %d bytes\n",
> +			  __func__, req->buf, req->length);
> +
> +	return -EIO;
> +}
> +
> +/**
> + * exynos_ss_udc_unmap_dma - unmap the DMA memory being used for the request
> + * @udc: The device state.
> + * @udc_ep: The endpoint for the request.
> + * @udc_req: The request being processed.
> + *
> + * This is the reverse of exynos_ss_udc_map_dma(), called for the completion
> + * of a request to ensure the buffer is ready for access by the caller.
> + */
> +static void exynos_ss_udc_unmap_dma(struct exynos_ss_udc *udc,
> +				    struct exynos_ss_udc_ep *udc_ep,
> +				    struct exynos_ss_udc_req *udc_req)
> +{
> +	struct usb_request *req = &udc_req->req;
> +	enum dma_data_direction dir;
> +
> +	dir = udc_ep->dir_in ? DMA_TO_DEVICE : DMA_FROM_DEVICE;
> +
> +	/* ignore this if we're not moving any data */
> +	if (udc_req->req.length == 0)
> +		return;
> +
> +	if (udc_req->mapped) {
> +		/* we mapped this, so unmap and remove the dma */
> +
> +		dma_unmap_single(udc->dev, req->dma, req->length, dir);
> +
> +		req->dma = DMA_ADDR_INVALID;
> +		udc_req->mapped = 0;
> +	}
> +}
> +
> +/**
> + * exynos_ss_udc_issue_epcmd - issue physical endpoint-specific command
> + * @udc: The device state.
> + * @epcmd: The command to issue.
> + */
> +static int exynos_ss_udc_issue_epcmd(struct exynos_ss_udc *udc,
> +				     struct exynos_ss_udc_ep_command *epcmd)
> +{
> +	int res;
> +	u32 depcmd;
> +
> +	/* If some of parameters are not in use, we will write it anyway
> +	   for simplification */
> +	writel(epcmd->param0, udc->regs + EXYNOS_USB3_DEPCMDPAR0(epcmd->ep));
> +	writel(epcmd->param1, udc->regs + EXYNOS_USB3_DEPCMDPAR1(epcmd->ep));
> +	writel(epcmd->param2, udc->regs + EXYNOS_USB3_DEPCMDPAR2(epcmd->ep));
> +
> +	depcmd = epcmd->cmdtyp | epcmd->cmdflags;
> +	writel(depcmd, udc->regs + EXYNOS_USB3_DEPCMD(epcmd->ep));
> +
> +	res = poll_bit_clear(udc->regs + EXYNOS_USB3_DEPCMD(epcmd->ep),
> +			     EXYNOS_USB3_DEPCMDx_CmdAct,
> +			     1000);
> +	return res;
> +}
> +
> +/**
> + * exynos_ss_udc_run_stop - start/stop the device controller operation
> + * @udc: The device state.
> + * @is_on: The action to take (1 - start, 0 - stop).
> + */
> +static void exynos_ss_udc_run_stop(struct exynos_ss_udc *udc, int is_on)
> +{
> +	int res;
> +
> +	if (is_on) {
> +		__orr32(udc->regs + EXYNOS_USB3_DCTL,
> +			EXYNOS_USB3_DCTL_Run_Stop);
> +		res = poll_bit_clear(udc->regs + EXYNOS_USB3_DSTS,
> +				     EXYNOS_USB3_DSTS_DevCtrlHlt,
> +				     1000);
> +	} else {
> +		__bic32(udc->regs + EXYNOS_USB3_DCTL,
> +			EXYNOS_USB3_DCTL_Run_Stop);
> +		res = poll_bit_set(udc->regs + EXYNOS_USB3_DSTS,
> +				   EXYNOS_USB3_DSTS_DevCtrlHlt,
> +				   1000);
> +	}
> +
> +	if (res < 0)
> +		dev_dbg(udc->dev, "Failed to %sconnect by software\n",
> +				  is_on ? "" : "dis");
> +}
> +
> +/**
> + * exynos_ss_udc_pullup - software-controlled connect/disconnect to USB host
> + * @gadget: The peripheral being connected/disconnected.
> + * @is_on: The action to take (1 - connect, 0 - disconnect).
> + */
> +static int exynos_ss_udc_pullup(struct usb_gadget *gadget, int is_on)
> +{
> +	struct exynos_ss_udc *udc = our_udc(gadget);
> +
> +	exynos_ss_udc_run_stop(udc, is_on);
> +
> +	return 0;
> +}

Ok, I will stop reviewing here. This looks VERY similar to Synopsys DWC3
controller. Why are you trying to add yet another driver ? Before
reviewing anything else, I want confirmation that you're _NOT_ using
Synopsys' controller.

Anton, please confirm the information. All commands are the same as
Synopsys' controller, I cannot let you add another driver for the same
controller. Just re-use the dwc3 controller.

-- 
balbi

Attachment: signature.asc
Description: Digital signature


[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux