netdev.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Sebastian Haas <haas@ems-wuensche.com>
To: Wolfgang Grandegger <wg@grandegger.com>
Cc: netdev@vger.kernel.org, socketcan-core@lists.berlios.de,
	greg@kroah.com, oliver@hartkopp.net, linux-usb@vger.kernel.org,
	Oliver Neukum <oliver@neukum.org>
Subject: Re: [PATCH 2/2] ems_usb: Added support for EMS CPC-USB/ARM7 CAN/USB interface
Date: Tue, 15 Sep 2009 13:40:55 +0200	[thread overview]
Message-ID: <4AAF7D47.1070005@ems-wuensche.com> (raw)
In-Reply-To: <4AAF7301.60608@grandegger.com>

-----BEGIN PGP SIGNED MESSAGE-----
Hash: SHA1

Wolfgang Grandegger schrieb:
> Hi Sebastian,
> 
> Sebastian Haas wrote:
>> This patch adds support for one channel CAN/USB interace CPC-USB/ARM7 from
>> EMS Dr. Thomas Wuensche (http://www.ems-wuensche.com).
>>
>> Signed-off-by: Sebastian Haas <haas@ems-wuensche.com>
> 
> The driver is almost OK from the Socket-CAN point of view. Just some
> *final* nitpicking:
> 
> [snip]
>> +static void ems_usb_read_interrupt_callback(struct urb *urb)
>> +{
>> +	struct ems_usb *dev = urb->context;
>> +	struct net_device *netdev;
>> +	int err;
>> +
>> +	netdev = dev->netdev;
> 
> Could be done in the declaration part above.
Fixed.

>> +	if (!netif_device_present(netdev))
>> +		return;
>> +
>> +	switch (urb->status) {
>> +	case 0:
>> +		dev->free_slots = dev->intr_in_buffer[1];
>> +		break;
>> +
>> +	case -ECONNRESET: /* unlink */
>> +	case -ENOENT:
>> +	case -ESHUTDOWN:
>> +		return;
>> +
>> +	default:
>> +		dev_info(netdev->dev.parent, "Rx interrupt aborted %d\n",
>> +			 urb->status);
>> +		break;
>> +	}
>> +
>> +	err = usb_submit_urb(urb, GFP_ATOMIC);
>> +
>> +	if (err == -ENODEV)
>> +		netif_device_detach(netdev);
>> +	else if (err)
>> +		dev_err(netdev->dev.parent,
>> +			"failed resubmitting intr urb: %d\n", err);
>> +
>> +	return;
>> +}
>> +
>> +static void ems_usb_rx_can_msg(struct ems_usb *dev, struct ems_cpc_msg *msg)
>> +{
>> +	struct can_frame *cf;
>> +	struct sk_buff *skb;
>> +	int i;
>> +	struct net_device_stats *stats = &dev->netdev->stats;
>> +
>> +	skb = dev_alloc_skb(sizeof(struct can_frame));
> 
> Please use netdev_alloc_skb() ...
Fixed.

>> +	if (skb == NULL)
>> +		return;
>> +
>> +	skb->dev = dev->netdev;
> 
> ... making the line above obsolete.
Fixed.

>> +	skb->protocol = htons(ETH_P_CAN);
>> +
>> +	cf = (struct can_frame *)skb_put(skb, sizeof(struct can_frame));
>> +
>> +	cf->can_id = msg->msg.can_msg.id;
>> +	cf->can_dlc = min_t(u8, msg->msg.can_msg.length, 8);
>> +
>> +	if (msg->type == CPC_MSG_TYPE_EXT_CAN_FRAME
>> +	    || msg->type == CPC_MSG_TYPE_EXT_RTR_FRAME)
>> +		cf->can_id |= CAN_EFF_FLAG;
>> +
>> +	if (msg->type == CPC_MSG_TYPE_RTR_FRAME
>> +	    || msg->type == CPC_MSG_TYPE_EXT_RTR_FRAME) {
>> +		cf->can_id |= CAN_RTR_FLAG;
>> +	} else {
>> +		for (i = 0; i < cf->can_dlc; i++)
>> +			cf->data[i] = msg->msg.can_msg.msg[i];
>> +	}
>> +
>> +	netif_rx(skb);
>> +
>> +	dev->netdev->last_rx = jiffies;
>> +	stats->rx_packets++;
>> +	stats->rx_bytes += cf->can_dlc;
>> +}
>> +
>> +static void ems_usb_rx_err(struct ems_usb *dev, struct ems_cpc_msg *msg)
>> +{
>> +	struct can_frame *cf;
>> +	struct sk_buff *skb;
>> +	struct net_device_stats *stats = &dev->netdev->stats;
>> +
>> +	skb = dev_alloc_skb(sizeof(struct can_frame));
>> +	if (skb == NULL)
>> +		return;
>> +
>> +	skb->dev = dev->netdev;
> 
> Ditto.
> 
> [snip]
>> +static int ems_usb_start_xmit(struct sk_buff *skb, struct net_device *netdev)
>> +{
>> +	struct ems_usb *dev = netdev_priv(netdev);
>> +	struct ems_tx_urb_context *context = NULL;
>> +	struct net_device_stats *stats = &netdev->stats;
>> +	struct can_frame *cf = (struct can_frame *)skb->data;
>> +	struct ems_cpc_msg *msg;
>> +	struct urb *urb;
>> +	u8 *buf;
>> +	int i, err;
>> +	size_t size = CPC_HEADER_SIZE + CPC_MSG_HEADER_LEN
>> +			+ sizeof(struct cpc_can_msg);
>> +
>> +	/* create a URB, and a buffer for it, and copy the data to the URB */
>> +	urb = usb_alloc_urb(0, GFP_ATOMIC);
>> +	if (!urb) {
>> +		dev_err(netdev->dev.parent, "No memory left for URBs\n");
>> +		goto nomem;
>> +	}
>> +
>> +	buf = usb_buffer_alloc(dev->udev, size, GFP_ATOMIC, &urb->transfer_dma);
>> +	if (!buf) {
>> +		dev_err(netdev->dev.parent, "No memory left for USB buffer\n");
>> +		usb_free_urb(urb);
>> +		goto nomem;
>> +	}
>> +
>> +	msg = (struct ems_cpc_msg *)&buf[CPC_HEADER_SIZE];
>> +
>> +	msg->msg.can_msg.id = cf->can_id & 0x1FFFFFFFU;
> 
> Please use CAN_ERR_MASK instead (even if the name is somehow
> misleading). See http://lxr.linux.no/#linux+v2.6.31/include/linux/can.h#L31.
Fixed.

> [snip]
>> +}
>> +
>> +
> 
> Remove one empty line, please.
Fixed.

>> +static void init_params_sja1000(struct ems_cpc_msg *msg)
>> +{
>> +	struct cpc_sja1000_params *sja1000 =
>> +		&msg->msg.can_params.cc_params.sja1000;
>> +
>> +	msg->type = CPC_CMD_TYPE_CAN_PARAMS;
>> +	msg->length = sizeof(struct cpc_can_params);
>> +	msg->msgid = 0;
>> +
>> +	msg->msg.can_params.cc_type = CPC_CC_TYPE_SJA1000;
>> +
>> +	/* Acceptance filter open */
>> +	sja1000->acc_code0 = 0x00;
>> +	sja1000->acc_code1 = 0x00;
>> +	sja1000->acc_code2 = 0x00;
>> +	sja1000->acc_code3 = 0x00;
>> +
>> +	/* Acceptance filter open */
>> +	sja1000->acc_mask0 = 0xFF;
>> +	sja1000->acc_mask1 = 0xFF;
>> +	sja1000->acc_mask2 = 0xFF;
>> +	sja1000->acc_mask3 = 0xFF;
>> +
>> +	sja1000->btr0 = 0;
>> +	sja1000->btr1 = 0;
>> +
>> +	sja1000->outp_contr = SJA1000_DEFAULT_OUTPUT_CONTROL;
>> +	sja1000->mode = SJA1000_MOD_RM;
>> +}
>> +
>> +/*
>> + * probe function for new CPC-USB devices
>> + */
>> +static int ems_usb_probe(struct usb_interface *intf,
>> +			 const struct usb_device_id *id)
>> +{
>> +	struct net_device *netdev;
>> +	struct ems_usb *dev;
>> +	int i, err;
>> +
>> +	netdev = alloc_candev(sizeof(struct ems_usb));
>> +	if (!netdev) {
>> +		dev_err(netdev->dev.parent, "Couldn't alloc candev\n");
>> +		return -ENOMEM;
>> +	}
>> +
>> +	dev = netdev_priv(netdev);
>> +
>> +	dev->udev = interface_to_usbdev(intf);
>> +	dev->netdev = netdev;
>> +
>> +	dev->can.state = CAN_STATE_STOPPED;
>> +	dev->can.bittiming_const = &ems_usb_bittiming_const;
>> +	dev->can.do_set_bittiming = ems_usb_set_bittiming;
>> +	dev->can.do_set_mode = ems_usb_set_mode;
>> +
>> +	netdev->flags |= IFF_ECHO; /* we support local echo */
>> +
>> +	/*
>> +	 * The device actually uses a 16MHz clock to generate the CAN clock
>> +	 * but it expects SJA1000 bit settings based on 8MHz (is internally
>> +	 * converted).
>> +	 */
> 
> Should go up to the macro definition.
Fixed.

>> +	dev->can.clock.freq = EMS_USB_ARM7_CLOCK;
>> +
>> +	netdev->netdev_ops = &ems_usb_netdev_ops;
>> +
>> +	netdev->flags |= IFF_ECHO; /* we support local echo */
>> +
>> +	init_usb_anchor(&dev->rx_submitted);
>> +
>> +	init_usb_anchor(&dev->tx_submitted);
>> +	atomic_set(&dev->active_tx_urbs, 0);
>> +
>> +	for (i = 0; i < MAX_TX_URBS; i++)
>> +		dev->tx_contexts[i].echo_index = MAX_TX_URBS;
>> +
>> +	dev->intr_urb = usb_alloc_urb(0, GFP_KERNEL);
>> +	if (!dev->intr_urb) {
>> +		dev_err(netdev->dev.parent, "Couldn't alloc intr URB\n");
>> +		free_candev(netdev);
> 
> Please use goto's for cleanup to avoid code duplication.
> 
>> +		return -ENOMEM;
>> +	}
>> +
>> +	dev->intr_in_buffer = kzalloc(INTR_IN_BUFFER_SIZE, GFP_KERNEL);
>> +	if (!dev->intr_in_buffer) {
>> +		dev_err(netdev->dev.parent, "Couldn't alloc Intr buffer\n");
>> +		free_candev(netdev);
>> +		usb_free_urb(dev->intr_urb);
>> +		return -ENOMEM;
> 
> Ditto.
> 
>> +	}
>> +
>> +	dev->tx_msg_buffer = kzalloc(CPC_HEADER_SIZE +
>> +				     sizeof(struct ems_cpc_msg), GFP_KERNEL);
>> +	if (!dev->tx_msg_buffer) {
>> +		dev_err(netdev->dev.parent, "Couldn't alloc Tx buffer\n");
>> +		free_candev(netdev);
>> +		usb_free_urb(dev->intr_urb);
>> +		kfree(dev->intr_in_buffer);
>> +		return -ENOMEM;
> 
> Ditto.
> 
>> +	}
>> +
>> +	usb_set_intfdata(intf, dev);
>> +
>> +	SET_NETDEV_DEV(netdev, &intf->dev);
>> +
>> +	init_params_sja1000(&dev->active_params);
>> +
>> +	err = ems_usb_command_msg(dev, &dev->active_params);
>> +	if (err) {
>> +		dev_err(netdev->dev.parent,
>> +			"couldn't initialize controller: %d\n", err);
>> +		free_candev(netdev);
>> +		usb_free_urb(dev->intr_urb);
>> +		return err;
> 
> Ditto. Also kfree(dev->intr_in_buffer) seems to be missing.
Fixed.

> Please add a version number to the next patch, e.g. [PATCH v3 ...].
Okay, have to figure out how to do with stgit ;-)

> It would also be nice, if some USB guru's could have a look as well.
Oliver Neukum already did a quick review. I've fixed serveral
GFP_ATOMIC/KERNEL issues and a problem with DMA coherency on his feedback.

Cheers, Sebastian
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.9 (GNU/Linux)
Comment: Using GnuPG with Mozilla - http://enigmail.mozdev.org

iEYEARECAAYFAkqvfUAACgkQpqRB8PJG7Xxw/ACfT8FUQ5DlmrcDnxotWceIcyNn
fOIAoI/w8NHJg0qGxpF2PvU6Qd5BXaBg
=DV5x
-----END PGP SIGNATURE-----
-- 
EMS Dr. Thomas Wuensche e.K.
Sonnenhang 3
85304 Ilmmuenster
HRA Neuburg a.d. Donau, HR-Nr. 70.106
Phone: +49-8441-490260
Fax  : +49-8441-81860
http://www.ems-wuensche.com

  reply	other threads:[~2009-09-15 11:40 UTC|newest]

Thread overview: 9+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
     [not found] <20090914080945.5441.45509.stgit@localhost.localdomain>
     [not found] ` <20090914082229.5441.16650.stgit@localhost.localdomain>
     [not found]   ` <20090914082229.5441.16650.stgit-bi+AKbBUZKY6gyzm1THtWbp2dZbC/Bob@public.gmane.org>
2009-09-15 10:57     ` [PATCH 2/2] ems_usb: Added support for EMS CPC-USB/ARM7 CAN/USB interface Wolfgang Grandegger
2009-09-15 11:40       ` Sebastian Haas [this message]
     [not found]         ` <4AAF7D47.1070005-zsNKPWJ8Pib6hrUXjxyGrA@public.gmane.org>
2009-09-15 12:05           ` Wolfgang Grandegger
     [not found]             ` <4AAF82F4.6040708-5Yr1BZd7O62+XT7JhA+gdA@public.gmane.org>
2009-09-15 13:33               ` Sebastian Haas
2009-09-15 11:10   ` Eric Dumazet
     [not found]     ` <4AAF762A.5030605-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2009-09-15 11:36       ` Sebastian Haas
2009-09-11 10:54 [PATCH 1/2] cpc-usb: Removed driver from staging tree Sebastian Haas
     [not found] ` <20090911105449.7815.65368.stgit-bi+AKbBUZKY6gyzm1THtWbp2dZbC/Bob@public.gmane.org>
2009-09-11 10:54   ` [PATCH 2/2] ems_usb: Added support for EMS CPC-USB/ARM7 CAN/USB interface Sebastian Haas
     [not found]     ` <20090911105455.7815.12354.stgit-bi+AKbBUZKY6gyzm1THtWbp2dZbC/Bob@public.gmane.org>
2009-09-11 11:36       ` Wolfgang Grandegger
2009-09-11 12:07         ` Oliver Hartkopp

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4AAF7D47.1070005@ems-wuensche.com \
    --to=haas@ems-wuensche.com \
    --cc=greg@kroah.com \
    --cc=linux-usb@vger.kernel.org \
    --cc=netdev@vger.kernel.org \
    --cc=oliver@hartkopp.net \
    --cc=oliver@neukum.org \
    --cc=socketcan-core@lists.berlios.de \
    --cc=wg@grandegger.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).