All of lore.kernel.org
 help / color / mirror / Atom feed
* [RFC] CAN FD support part 2 - ideas and commented source
       [not found] <4FA242D0.2070604@hartkopp.net>
@ 2012-05-03 11:14 ` Oliver Hartkopp
  2012-05-03 12:37   ` Kurt Van Dijck
  0 siblings, 1 reply; 10+ messages in thread
From: Oliver Hartkopp @ 2012-05-03 11:14 UTC (permalink / raw)
  To: linux-can

Hi all,

in this mail i would like to document the ideas for CAN FD support pointing
directly to the changes inside the source.

The several steps and commits can be followed here:

https://gitorious.org/~hartkopp/linux-can/hartkopps-linux-can-next/commits

Generally CAN and CAN FD has four different possibilities we need to cover.

We have the high data rate (HDR) in the data section of the frame and we have
the extended data length (EDL) which allows more than 8 bytes payload:

See this table:

HDR EDL

 0   0  -> standard CAN frame and standard speed *1
 1   0  -> standard CAN frame and high data rate *2
 0   1  -> CAN FD frame and standard speed
 1   1  -> CAN FD frame and high data rate *3

*1 This is our standard SocketCAN behaviour we don't touch at all

*2 This is the estimated first occurrence of CAN FD in the real world which
uses the higher data rate for the payload but has no impact to userspace apps.
This means that the CAN driver itself needs to support the two bitrate
settings but still handles standard size CAN frames

*3 This is the final CAN FD support case using the CAN FD frames with 64 byte
data section

What are the major ideas of this CAN FD support?

#1. introduce a new struct canfd_frame for EDL support

#2. introduce a new ethernet protocol ETH_P_CANFD to make socket buffers
(skbuff) containing a struct canfd_frame

#3. use the maximum transfer unit (MTU) to distinguish EDL enabled CAN
netdevices supporting either can_frames or canfd_frames:

CAN_MTU = sizeof(struct can_frame) -> in standard CAN mode (-> without EDL)
CANFD_MTU = sizeof(struct canfd_frame) -> in CAN FD mode (with or w/o EDL)

E.g. when a CAN FD capable controller is configured without EDL it presents
itself as dev->mtu == CAN_MTU and processes struct can_frames. This is
independently whether it uses the high data rate in the data section or not.

#4. the CAN_RAW sockets get a new socket option CAN_RAW_FD_FRAMES which is
disabled by default to behave like a standard CAN RAW socket. When
CAN_RAW_FD_FRAMES is enabled the socket expects and provides struct
canfd_frames instead of struct can_frames.

But let's go through some code:
>

> diff --git a/drivers/net/can/dev.c b/drivers/net/can/dev.c
> index f03d7a4..1b7e843 100644
> --- a/drivers/net/can/dev.c
> +++ b/drivers/net/can/dev.c
> @@ -454,7 +454,7 @@ EXPORT_SYMBOL_GPL(can_bus_off);
>  static void can_setup(struct net_device *dev)
>  {
>  	dev->type = ARPHRD_CAN;
> -	dev->mtu = sizeof(struct can_frame);
> +	dev->mtu = CAN_MTU;


use the new CAN_MTU definition.

>  	dev->hard_header_len = 0;
>  	dev->addr_len = 0;
>  	dev->tx_queue_len = 10;
> diff --git a/drivers/net/can/vcan.c b/drivers/net/can/vcan.c
> index ea2d942..bddbafb 100644
> --- a/drivers/net/can/vcan.c
> +++ b/drivers/net/can/vcan.c
> @@ -74,9 +74,8 @@ static void vcan_rx(struct sk_buff *skb, struct net_device *dev)
>  	struct net_device_stats *stats = &dev->stats;
>  
>  	stats->rx_packets++;
> -	stats->rx_bytes += cf->can_dlc;
> +	stats->rx_bytes += can_dlc2len(cf->can_dlc);


calculate the real data length from the given dlc.

>  
> -	skb->protocol  = htons(ETH_P_CAN);


This was not needed here.

>  	skb->pkt_type  = PACKET_BROADCAST;
>  	skb->dev       = dev;
>  	skb->ip_summed = CHECKSUM_UNNECESSARY;
> @@ -94,7 +93,7 @@ static netdev_tx_t vcan_tx(struct sk_buff *skb, struct net_device *dev)
>  		return NETDEV_TX_OK;
>  
>  	stats->tx_packets++;
> -	stats->tx_bytes += cf->can_dlc;
> +	stats->tx_bytes += can_dlc2len(cf->can_dlc);


calculate the real data length from the given dlc.

>  
>  	/* set flag whether this packet has to be looped back */
>  	loop = skb->pkt_type == PACKET_LOOPBACK;
> @@ -108,7 +107,7 @@ static netdev_tx_t vcan_tx(struct sk_buff *skb, struct net_device *dev)
>  			 * CAN core already did the echo for us
>  			 */
>  			stats->rx_packets++;
> -			stats->rx_bytes += cf->can_dlc;
> +			stats->rx_bytes += can_dlc2len(cf->can_dlc);


dito.

>  		}
>  		kfree_skb(skb);
>  		return NETDEV_TX_OK;
> @@ -133,14 +132,25 @@ static netdev_tx_t vcan_tx(struct sk_buff *skb, struct net_device *dev)
>  	return NETDEV_TX_OK;
>  }
>  
> +static int vcan_change_mtu(struct net_device *dev, int new_mtu)
> +{
> +	if (new_mtu == CAN_MTU || new_mtu == CANFD_MTU) {
> +		dev->mtu = new_mtu;
> +		return 0;
> +	}
> +
> +	return -EINVAL;
> +}
> +
>  static const struct net_device_ops vcan_netdev_ops = {
>  	.ndo_start_xmit = vcan_tx,
> +	.ndo_change_mtu = vcan_change_mtu,
>  };


When changing the mtu it's only allowed to switch between CAN and CANFD MTUs.

>  
>  static void vcan_setup(struct net_device *dev)
>  {
>  	dev->type		= ARPHRD_CAN;
> -	dev->mtu		= sizeof(struct can_frame);
> +	dev->mtu		= CAN_MTU;


see above

>  	dev->hard_header_len	= 0;
>  	dev->addr_len		= 0;
>  	dev->tx_queue_len	= 0;
> diff --git a/include/linux/can.h b/include/linux/can.h
> index 9a19bcb..74052da 100644
> --- a/include/linux/can.h
> +++ b/include/linux/can.h
> @@ -46,6 +46,9 @@ typedef __u32 canid_t;
>   */
>  typedef __u32 can_err_mask_t;
>  
> +#define CAN_MAX_DLC 8
> +#define CANFD_MAX_DLC 15
> +


some useful defines to avoid hardcoded values.

>  /**
>   * struct can_frame - basic CAN frame structure
>   * @can_id:  the CAN ID of the frame and CAN_*_FLAG flags, see above.
> @@ -58,6 +61,65 @@ struct can_frame {
>  	__u8    data[8] __attribute__((aligned(8)));
>  };
>  
> +/*
> + * defined bits for canfd_frame.flags
> + *
> + * RX: HDR/EDL - info about received CAN FD frame
> + *     ESI     - bit from originating CAN controller
> + * TX: HDR/EDL - control per-frame settings if supported by CAN controller
> + *     ESI     - bit is set by local CAN controller
> + */
> +#define CANFD_HDR 0x01 /* high data rate */
> +#define CANFD_EDL 0x02 /* extended data length */
> +#define CANFD_ESI 0x04 /* error state indicator */
> +


New flags for ingoing and outgoing CAN FD frames.

It allows to set the flags per frame, which can make the CAN controller
transmit different frames upon request.

> +/**
> + * struct canfd_frame - CAN flexible data rate frame structure
> + * @can_id:  the CAN ID of the frame and CAN_*_FLAG flags, see above.
> + * @can_dlc: the data length field of the CAN frame (data length code)
> + * @flags:   additional flags for CAN FD
> + * @__res0:  reserved / padding
> + * @__res1:  reserved / padding
> + * @data:    the CAN FD frame payload (up to 64 byte).
> + */
> +struct canfd_frame {
> +	canid_t can_id;  /* 32 bit CAN_ID + EFF/RTR/ERR flags */
> +	__u8    can_dlc; /* CAN FD data length code: 0 .. 0xF */
> +	__u8    flags;   /* additional flags for CAN FD */
> +	__u8    __res0;  /* reserved / padding */
> +	__u8    __res1;  /* reserved / padding */
> +	__u8    data[64] __attribute__((aligned(8)));
> +};


Added some reasonable padding. The can_dlc is at the same place in can_frame
and canfd_frame to ease the access for sanity checks.

See BUILD_BUG_ON() statement below.

> +
> +static const u8 dlc2len[16] = {0, 1, 2, 3, 4, 5, 6, 7, 8,
> +			       12, 16, 20, 24, 32, 48, 64};
> +
> +/* get data length from can_dlc with sanitized can_dlc */
> +static inline u8 can_dlc2len (u8 can_dlc)
> +{
> +	return dlc2len[can_dlc & 0x0F];
> +}
> +
> +static const u8 len2dlc[65] = {0, 1, 2, 3, 4, 5, 6, 7, 8,	/* 0 - 8 */
> +			       9, 9, 9, 9,			/* 9 - 12 */
> +			       10, 10, 10, 10,			/* 13 - 16 */
> +			       11, 11, 11, 11,			/* 17 - 20 */
> +			       12, 12, 12, 12,			/* 21 - 24 */
> +			       13, 13, 13, 13, 13, 13, 13, 13,	/* 25 - 32 */
> +			       14, 14, 14, 14, 14, 14, 14, 14,	/* 33 - 40 */
> +			       14, 14, 14, 14, 14, 14, 14, 14,	/* 41 - 48 */
> +			       15, 15, 15, 15, 15, 15, 15, 15,	/* 49 - 56 */
> +			       15, 15, 15, 15, 15, 15, 15, 15}; /* 57 - 64 */
> +
> +/* map the sanitized data length to an appropriate data length code */ 
> +static inline u8 can_len2dlc(u8 len)
> +{
> +	if (unlikely(len > 64))
> +		return 0xF;
> +
> +	return len2dlc[len];
> +}
> +


Some helpers to swap length and dlc values.

>  /* particular protocols of the protocol family PF_CAN */
>  #define CAN_RAW		1 /* RAW sockets */
>  #define CAN_BCM		2 /* Broadcast Manager */
> diff --git a/include/linux/can/core.h b/include/linux/can/core.h
> index 0ccc1cd..e6a3126 100644
> --- a/include/linux/can/core.h
> +++ b/include/linux/can/core.h
> @@ -17,10 +17,10 @@
>  #include <linux/skbuff.h>
>  #include <linux/netdevice.h>
>  
> -#define CAN_VERSION "20090105"
> +#define CAN_VERSION "20120413"
>  
>  /* increment this number each time you change some user-space interface */
> -#define CAN_ABI_VERSION "8"
> +#define CAN_ABI_VERSION "9"
>  
>  #define CAN_VERSION_STRING "rev " CAN_VERSION " abi " CAN_ABI_VERSION
>  


When supporting CAN FD we should probably indicate the new possibilities.

> diff --git a/include/linux/can/dev.h b/include/linux/can/dev.h
> index 5d2efe7..082b664 100644
> --- a/include/linux/can/dev.h
> +++ b/include/linux/can/dev.h
> @@ -61,7 +61,11 @@ struct can_priv {
>   * To be used in the CAN netdriver receive path to ensure conformance with
>   * ISO 11898-1 Chapter 8.4.2.3 (DLC field)
>   */
> -#define get_can_dlc(i)	(min_t(__u8, (i), 8))
> +#define get_can_dlc(i)		(min_t(__u8, (i), CAN_MAX_DLC))
> +#define get_canfd_dlc(i)	(min_t(__u8, (i), CANFD_MAX_DLC))


trivial.

> +
> +#define CAN_MTU		(sizeof(struct can_frame))
> +#define CANFD_MTU	(sizeof(struct canfd_frame))


Add new defines for better code readability.

>  
>  /* Drop a given socketbuffer if it does not contain a valid CAN frame. */
>  static inline int can_dropped_invalid_skb(struct net_device *dev,
> @@ -69,13 +73,23 @@ static inline int can_dropped_invalid_skb(struct net_device *dev,
>  {
>  	const struct can_frame *cf = (struct can_frame *)skb->data;
>  
> -	if (unlikely(skb->len != sizeof(*cf) || cf->can_dlc > 8)) {
> -		kfree_skb(skb);
> -		dev->stats.tx_dropped++;
> -		return 1;
> -	}
> +	if (skb->protocol == htons(ETH_P_CAN)) {
> +		if (skb->len != sizeof(struct can_frame) ||
> +		    cf->can_dlc > CAN_MAX_DLC)
> +			goto inval_skb;
> +	} else if (skb->protocol == htons(ETH_P_CANFD)) {
> +		if (skb->len != sizeof(struct canfd_frame) ||
> +		    cf->can_dlc > CANFD_MAX_DLC)
> +			goto inval_skb;
> +	} else
> +		goto inval_skb;
>  
>  	return 0;
> +
> +inval_skb:
> +	kfree_skb(skb);
> +	dev->stats.tx_dropped++;
> +	return 1;
>  }


extend the checks for CAN FD.

>  
>  struct net_device *alloc_candev(int sizeof_priv, unsigned int echo_skb_max);
> diff --git a/include/linux/can/raw.h b/include/linux/can/raw.h
> index 781f3a3..5448c0f 100644
> --- a/include/linux/can/raw.h
> +++ b/include/linux/can/raw.h
> @@ -23,7 +23,8 @@ enum {
>  	CAN_RAW_FILTER = 1,	/* set 0 .. n can_filter(s)          */
>  	CAN_RAW_ERR_FILTER,	/* set filter for error frames       */
>  	CAN_RAW_LOOPBACK,	/* local loopback (default:on)       */
> -	CAN_RAW_RECV_OWN_MSGS	/* receive my own msgs (default:off) */
> +	CAN_RAW_RECV_OWN_MSGS,	/* receive my own msgs (default:off) */
> +	CAN_RAW_FD_FRAMES,	/* use struct canfd_frame (default:off) */
>  };


new sockopt for CAN_RAW.

Could be used for other struct can_frame depended protocols too.

>  
>  #endif
> diff --git a/include/linux/if_ether.h b/include/linux/if_ether.h
> index 56d907a..5c9c2b4 100644
> --- a/include/linux/if_ether.h
> +++ b/include/linux/if_ether.h
> @@ -119,6 +119,7 @@
>  #define ETH_P_PHONET	0x00F5		/* Nokia Phonet frames          */
>  #define ETH_P_IEEE802154 0x00F6		/* IEEE802.15.4 frame		*/
>  #define ETH_P_CAIF	0x00F7		/* ST-Ericsson CAIF protocol	*/
> +#define ETH_P_CANFD	0x00FD		/* Controller Area Network FD   */


to indicate the skbuff content as struct can_frame

>  
>  /*
>   *	This is an Ethernet frame header.
> diff --git a/net/can/af_can.c b/net/can/af_can.c
> index 0ce2ad0..1b7f1f8 100644
> --- a/net/can/af_can.c
> +++ b/net/can/af_can.c
> @@ -41,6 +41,7 @@
>   */
>  
>  #include <linux/module.h>
> +#include <linux/stddef.h>
>  #include <linux/init.h>
>  #include <linux/kmod.h>
>  #include <linux/slab.h>
> @@ -228,10 +229,20 @@ int can_send(struct sk_buff *skb, int loop)
>  	struct can_frame *cf = (struct can_frame *)skb->data;
>  	int err;
>  
> -	if (skb->len != sizeof(struct can_frame) || cf->can_dlc > 8) {
> -		kfree_skb(skb);
> -		return -EINVAL;
> -	}
> +	if (skb->protocol == htons(ETH_P_CAN)) {
> +		if (skb->len != sizeof(struct can_frame) ||
> +		    cf->can_dlc > CAN_MAX_DLC)
> +			goto inval_skb;
> +	} else if (skb->protocol == htons(ETH_P_CANFD)) {
> +		if (skb->len != sizeof(struct canfd_frame) ||
> +		    cf->can_dlc > CANFD_MAX_DLC)
> +			goto inval_skb;
> +	} else
> +		goto inval_skb;
> +


extend sanity check.

> +	/* make sure the CAN frame can pass the selected CAN netdevice */
> +	if (skb->len > skb->dev->mtu)
> +		goto inval_skb;


see comment itself.

>  
>  	if (skb->dev->type != ARPHRD_CAN) {
>  		kfree_skb(skb);
> @@ -243,7 +254,6 @@ int can_send(struct sk_buff *skb, int loop)
>  		return -ENETDOWN;
>  	}
>  
> -	skb->protocol = htons(ETH_P_CAN);


remove this from can_send() as we don't know this anymore.
The protocol has to take care of this setting now (see e.g. bcm.c)

>  	skb_reset_network_header(skb);
>  	skb_reset_transport_header(skb);
>  
> @@ -300,6 +310,10 @@ int can_send(struct sk_buff *skb, int loop)
>  	can_stats.tx_frames_delta++;
>  
>  	return 0;
> +
> +inval_skb:
> +	kfree_skb(skb);
> +	return -EINVAL;
>  }
>  EXPORT_SYMBOL(can_send);
>  
> @@ -632,24 +646,11 @@ static int can_rcv_filter(struct dev_rcv_lists *d, struct sk_buff *skb)
>  	return matches;
>  }
>  
> -static int can_rcv(struct sk_buff *skb, struct net_device *dev,
> -		   struct packet_type *pt, struct net_device *orig_dev)
> +static void can_receive(struct sk_buff *skb, struct net_device *dev)
>  {
>  	struct dev_rcv_lists *d;
> -	struct can_frame *cf = (struct can_frame *)skb->data;
>  	int matches;
>  
> -	if (!net_eq(dev_net(dev), &init_net))
> -		goto drop;
> -
> -	if (WARN_ONCE(dev->type != ARPHRD_CAN ||
> -		      skb->len != sizeof(struct can_frame) ||
> -		      cf->can_dlc > 8,
> -		      "PF_CAN: dropped non conform skbuf: "
> -		      "dev type %d, len %d, can_dlc %d\n",
> -		      dev->type, skb->len, cf->can_dlc))
> -		goto drop;
> -
>  	/* update statistics */
>  	can_stats.rx_frames++;
>  	can_stats.rx_frames_delta++;
> @@ -673,7 +674,49 @@ static int can_rcv(struct sk_buff *skb, struct net_device *dev,
>  		can_stats.matches++;
>  		can_stats.matches_delta++;
>  	}
> +}
> +
> +static int can_rcv(struct sk_buff *skb, struct net_device *dev,
> +		   struct packet_type *pt, struct net_device *orig_dev)
> +{
> +	struct can_frame *cf = (struct can_frame *)skb->data;
>  
> +	if (!net_eq(dev_net(dev), &init_net))
> +		goto drop;
> +
> +	if (WARN_ONCE(dev->type != ARPHRD_CAN ||
> +		      skb->len != sizeof(struct can_frame) ||
> +		      cf->can_dlc > CAN_MAX_DLC,
> +		      "PF_CAN: dropped non conform CAN skbuf: "
> +		      "dev type %d, len %d, can_dlc %d\n",
> +		      dev->type, skb->len, cf->can_dlc))
> +		goto drop;
> +
> +	can_receive(skb, dev);
> +	return NET_RX_SUCCESS;
> +
> +drop:
> +	kfree_skb(skb);
> +	return NET_RX_DROP;
> +}
> +
> +static int canfd_rcv(struct sk_buff *skb, struct net_device *dev,
> +		   struct packet_type *pt, struct net_device *orig_dev)
> +{
> +	struct can_frame *cf = (struct can_frame *)skb->data;
> +
> +	if (!net_eq(dev_net(dev), &init_net))
> +		goto drop;
> +
> +	if (WARN_ONCE(dev->type != ARPHRD_CAN ||
> +		      skb->len != sizeof(struct canfd_frame) ||
> +		      cf->can_dlc > CANFD_MAX_DLC,
> +		      "PF_CAN: dropped non conform CAN FD skbuf: "
> +		      "dev type %d, len %d, can_dlc %d\n",
> +		      dev->type, skb->len, cf->can_dlc))
> +		goto drop;
> +
> +	can_receive(skb, dev);
>  	return NET_RX_SUCCESS;
>  


Add two different callback functions to skip the skb->type testing every time.

>  drop:
> @@ -811,6 +854,12 @@ static struct packet_type can_packet __read_mostly = {
>  	.func = can_rcv,
>  };
>  
> +static struct packet_type canfd_packet __read_mostly = {
> +	.type = cpu_to_be16(ETH_P_CANFD),
> +	.dev  = NULL,
> +	.func = canfd_rcv,
> +};
> +


register for CAN FD frames

>  static const struct net_proto_family can_family_ops = {
>  	.family = PF_CAN,
>  	.create = can_create,
> @@ -824,6 +873,10 @@ static struct notifier_block can_netdev_notifier __read_mostly = {
>  
>  static __init int can_init(void)
>  {
> +	/* check for correct padding that can_dlc owns always the same position */
> +	BUILD_BUG_ON(offsetof(struct can_frame, can_dlc) !=
> +		     offsetof(struct canfd_frame, can_dlc));
> +


Check if can_dlc always has the same position to access the value by the same
offset reference.

Otherwise you would need
struct can_frame *cf = (struct can_frame *)skb->data;
struct canfd_frame *cfd = (struct canfd_frame *)skb->data;

to implement
if (cf->can_dlc > 8)

which doubles code.

>  	printk(banner);
>  
>  	memset(&can_rx_alldev_list, 0, sizeof(can_rx_alldev_list));
> @@ -846,6 +899,7 @@ static __init int can_init(void)
>  	sock_register(&can_family_ops);
>  	register_netdevice_notifier(&can_netdev_notifier);
>  	dev_add_pack(&can_packet);
> +	dev_add_pack(&canfd_packet);


register for CAN FD frames

>  
>  	return 0;
>  }
> @@ -861,6 +915,7 @@ static __exit void can_exit(void)
>  
>  	/* protocol unregister */
>  	dev_remove_pack(&can_packet);
> +	dev_remove_pack(&canfd_packet);


unregister for CAN FD frames

>  	unregister_netdevice_notifier(&can_netdev_notifier);
>  	sock_unregister(PF_CAN);
>  
> diff --git a/net/can/bcm.c b/net/can/bcm.c
> index 151b773..2060945 100644
> --- a/net/can/bcm.c
> +++ b/net/can/bcm.c
> @@ -265,6 +265,7 @@ static void bcm_can_tx(struct bcm_op *op)
>  	/* send with loopback */
>  	skb->dev = dev;
>  	skb->sk = op->sk;
> +	skb->protocol = htons(ETH_P_CAN);


take care yourself.

>  	can_send(skb, 1);
>  
>  	/* update statistics */
> @@ -1215,6 +1216,7 @@ static int bcm_tx_send(struct msghdr *msg, int ifindex, struct sock *sk)
>  
>  	skb->dev = dev;
>  	skb->sk  = sk;
> +	skb->protocol = htons(ETH_P_CAN);


dito.

>  	err = can_send(skb, 1); /* send with loopback */
>  	dev_put(dev);
>  
> diff --git a/net/can/raw.c b/net/can/raw.c
> index cde1b4a..7d3864e 100644
> --- a/net/can/raw.c
> +++ b/net/can/raw.c
> @@ -82,6 +82,7 @@ struct raw_sock {
>  	struct notifier_block notifier;
>  	int loopback;
>  	int recv_own_msgs;
> +	int fd_frames;


the switch.

>  	int count;                 /* number of active filters */
>  	struct can_filter dfilter; /* default/single filter */
>  	struct can_filter *filter; /* pointer to filter(s) */
> @@ -119,6 +120,15 @@ static void raw_rcv(struct sk_buff *oskb, void *data)
>  	if (!ro->recv_own_msgs && oskb->sk == sk)
>  		return;
>  
> +	/* only pass correct frames to the socket */
> +	if (ro->fd_frames) {
> +		if (oskb->protocol != htons(ETH_P_CANFD))
> +			return;
> +	} else {
> +		if (oskb->protocol != htons(ETH_P_CAN))
> +			return;
> +	}
> +


sanity check

>  	/* clone the given skb to be able to enqueue it into the rcv queue */
>  	skb = skb_clone(oskb, GFP_ATOMIC);
>  	if (!skb)
> @@ -291,6 +301,7 @@ static int raw_init(struct sock *sk)
>  	/* set default loopback behaviour */
>  	ro->loopback         = 1;
>  	ro->recv_own_msgs    = 0;
> +	ro->fd_frames        = 0;
>  
>  	/* set notifier */
>  	ro->notifier.notifier_call = raw_notifier;
> @@ -569,6 +580,15 @@ static int raw_setsockopt(struct socket *sock, int level, int optname,
>  
>  		break;
>  
> +	case CAN_RAW_FD_FRAMES:
> +		if (optlen != sizeof(ro->fd_frames))
> +			return -EINVAL;
> +
> +		if (copy_from_user(&ro->fd_frames, optval, optlen))
> +			return -EFAULT;
> +
> +		break;
> +
>  	default:
>  		return -ENOPROTOOPT;
>  	}
> @@ -627,6 +647,12 @@ static int raw_getsockopt(struct socket *sock, int level, int optname,
>  		val = &ro->recv_own_msgs;
>  		break;
>  
> +	case CAN_RAW_FD_FRAMES:
> +		if (len > sizeof(int))
> +			len = sizeof(int);
> +		val = &ro->fd_frames;
> +		break;
> +
>  	default:
>  		return -ENOPROTOOPT;
>  	}


sockopt settings

> @@ -662,13 +688,24 @@ static int raw_sendmsg(struct kiocb *iocb, struct socket *sock,
>  	} else
>  		ifindex = ro->ifindex;
>  
> -	if (size != sizeof(struct can_frame))
> -		return -EINVAL;
> +	if (ro->fd_frames) {
> +		if (size != sizeof(struct canfd_frame))
> +			return -EINVAL;
> +	} else {
> +		if (size != sizeof(struct can_frame))
> +			return -EINVAL;
> +	}
>  
>  	dev = dev_get_by_index(&init_net, ifindex);
>  	if (!dev)
>  		return -ENXIO;
>  
> +	/* make sure the created CAN frame can pass the CAN netdevice */
> +	if (size > dev->mtu) {
> +		err = -EINVAL;
> +		goto put_dev;
> +	}
> +


some sanity checks.

>  	skb = sock_alloc_send_skb(sk, size, msg->msg_flags & MSG_DONTWAIT,
>  				  &err);
>  	if (!skb)
> @@ -684,6 +721,11 @@ static int raw_sendmsg(struct kiocb *iocb, struct socket *sock,
>  	/* to be able to check the received tx sock reference in raw_rcv() */
>  	skb_shinfo(skb)->tx_flags |= SKBTX_DRV_NEEDS_SK_REF;
>  
> +	if (ro->fd_frames)
> +		skb->protocol = htons(ETH_P_CANFD);
> +	else
> +		skb->protocol = htons(ETH_P_CAN);
> +
>  	skb->dev = dev;
>  	skb->sk  = sk;
>  


take care to set the correct protocol type.

That's it.

Any objections / better ideas?

Regards,
Oliver



^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 11:14 ` [RFC] CAN FD support part 2 - ideas and commented source Oliver Hartkopp
@ 2012-05-03 12:37   ` Kurt Van Dijck
  2012-05-03 12:50     ` Oliver Hartkopp
  0 siblings, 1 reply; 10+ messages in thread
From: Kurt Van Dijck @ 2012-05-03 12:37 UTC (permalink / raw)
  To: Oliver Hartkopp; +Cc: linux-can

Oliver,

Your proposal looks more elaborate than my first one :-)

In this email, I'll restrict myself to discussing ETH_P_CANFD.

On Thu, May 03, 2012 at 01:14:58PM +0200, Oliver Hartkopp wrote:
> Hi all,
> 
> in this mail i would like to document the ideas for CAN FD support pointing
> directly to the changes inside the source.
> 

[........]
> 
> That's it.

using dev's MTU is a very good idea!

Restricting the API to use one of 2 MTU's makes life easier.

> 
> Any objections / better ideas?
I don't see benefit in introducing ETH_P_CANFD.
Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
Why not using that info. IMHO, ETH_P_CANFD duplicates the info.

This part proves my point:
@@ -69,13 +73,23 @@ static inline int can_dropped_invalid_skb(struct net_device *dev,
 {
 	const struct can_frame *cf = (struct can_frame *)skb->data;
 
-	if (unlikely(skb->len != sizeof(*cf) || cf->can_dlc > 8)) {
This check should have become:
+	if (unlikely(skb->len != dev->mut) || (cf->can_dlc > mtu_to_max_can_dlc(dev->mtu))

Instead of all the if then else flows here.
A lookup table to use by mtu_to_max_can_dlc is quite straight-forward:
static const int mtu_to_max_can_dlc_table[] = {
	[sizeof(struct can_frame) = CAN_MAX_DLC,
	[sizeof(struct canfd_frame) = CANFD_MAX_DLC,
};

The remainder of the function needs no change, so all this patching
was not necessary.

-		kfree_skb(skb);
-		dev->stats.tx_dropped++;
-		return 1;
-	}
+	if (skb->protocol == htons(ETH_P_CAN)) {
+		if (skb->len != sizeof(struct can_frame) ||
+		    cf->can_dlc > CAN_MAX_DLC)
+			goto inval_skb;
+	} else if (skb->protocol == htons(ETH_P_CANFD)) {
+		if (skb->len != sizeof(struct canfd_frame) ||
+		    cf->can_dlc > CANFD_MAX_DLC)
+			goto inval_skb;
+	} else
+		goto inval_skb;
	}
 
 	return 0;

Similarly, can_rcv() & can_send() can access the device's mtu

Do you still think introducing ETH_P_CANFD is necessary?


Kurt

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 12:37   ` Kurt Van Dijck
@ 2012-05-03 12:50     ` Oliver Hartkopp
  2012-05-03 13:22       ` Kurt Van Dijck
  2012-05-03 18:01       ` Wireshark / libpcap - was " Oliver Hartkopp
  0 siblings, 2 replies; 10+ messages in thread
From: Oliver Hartkopp @ 2012-05-03 12:50 UTC (permalink / raw)
  To: linux-can

On 03.05.2012 14:37, Kurt Van Dijck wrote:

> Oliver,
> 
> Your proposal looks more elaborate than my first one :-)
> 
> In this email, I'll restrict myself to discussing ETH_P_CANFD.
> 
> On Thu, May 03, 2012 at 01:14:58PM +0200, Oliver Hartkopp wrote:
>> Hi all,
>>
>> in this mail i would like to document the ideas for CAN FD support pointing
>> directly to the changes inside the source.
>>
> 
> [........]
>>
>> That's it.
> 
> using dev's MTU is a very good idea!


Tnx!

> 
> Restricting the API to use one of 2 MTU's makes life easier.
> 
>>
>> Any objections / better ideas?
> I don't see benefit in introducing ETH_P_CANFD.
> Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
> Why not using that info. IMHO, ETH_P_CANFD duplicates the info.


This has a big advantage when thinking about wireshark & friends.

You can look into the eth protocol and know the skb data layout.
The length information is some kind of implicit knowledge.

> 
> This part proves my point:
> @@ -69,13 +73,23 @@ static inline int can_dropped_invalid_skb(struct net_device *dev,
>  {
>  	const struct can_frame *cf = (struct can_frame *)skb->data;
>  
> -	if (unlikely(skb->len != sizeof(*cf) || cf->can_dlc > 8)) {
> This check should have become:
> +	if (unlikely(skb->len != dev->mut) || (cf->can_dlc > mtu_to_max_can_dlc(dev->mtu))
> 
> Instead of all the if then else flows here.
> A lookup table to use by mtu_to_max_can_dlc is quite straight-forward:
> static const int mtu_to_max_can_dlc_table[] = {
> 	[sizeof(struct can_frame) = CAN_MAX_DLC,
> 	[sizeof(struct canfd_frame) = CANFD_MAX_DLC,
> };


Nice idea.

How big is mtu_to_max_can_dlc_table ??
sizeof(struct canfd_frame) * sizeof(int)

Regards,
Oliver

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 12:50     ` Oliver Hartkopp
@ 2012-05-03 13:22       ` Kurt Van Dijck
  2012-05-07 11:51         ` Felix Obenhuber
  2012-05-03 18:01       ` Wireshark / libpcap - was " Oliver Hartkopp
  1 sibling, 1 reply; 10+ messages in thread
From: Kurt Van Dijck @ 2012-05-03 13:22 UTC (permalink / raw)
  To: Oliver Hartkopp; +Cc: linux-can

On Thu, May 03, 2012 at 02:50:32PM +0200, Oliver Hartkopp wrote:
> On 03.05.2012 14:37, Kurt Van Dijck wrote:
> 
> > Oliver,
> > 
> > Your proposal looks more elaborate than my first one :-)
> > 
> > In this email, I'll restrict myself to discussing ETH_P_CANFD.
> > 
> > On Thu, May 03, 2012 at 01:14:58PM +0200, Oliver Hartkopp wrote:
> >> Hi all,
> >>
> >> in this mail i would like to document the ideas for CAN FD support pointing
> >> directly to the changes inside the source.
> >>
> > 
> > [........]
> >>
> >> That's it.
> > 
> > using dev's MTU is a very good idea!
> 
> 
> Tnx!
> 
> > 
> > Restricting the API to use one of 2 MTU's makes life easier.
> > 
> >>
> >> Any objections / better ideas?
> > I don't see benefit in introducing ETH_P_CANFD.
> > Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
> > Why not using that info. IMHO, ETH_P_CANFD duplicates the info.
> 
> 
> This has a big advantage when thinking about wireshark & friends.
I must have missed something here. ETH_P_CANFD value somewhere in struct skb
is something kernel specific. Does wireshark look inside kernel?
> 
> You can look into the eth protocol and know the skb data layout.
Both skb data layouts are compatible!
> The length information is some kind of implicit knowledge.
There are quite some checks that force the explicit use of 2 lengths

> 
> > 
> > This part proves my point:
> > @@ -69,13 +73,23 @@ static inline int can_dropped_invalid_skb(struct net_device *dev,
> >  {
> >  	const struct can_frame *cf = (struct can_frame *)skb->data;
> >  
> > -	if (unlikely(skb->len != sizeof(*cf) || cf->can_dlc > 8)) {
> > This check should have become:
> > +	if (unlikely(skb->len != dev->mut) || (cf->can_dlc > mtu_to_max_can_dlc(dev->mtu))
> > 
> > Instead of all the if then else flows here.
> > A lookup table to use by mtu_to_max_can_dlc is quite straight-forward:
> > static const int mtu_to_max_can_dlc_table[] = {
> > 	[sizeof(struct can_frame) = CAN_MAX_DLC,
> > 	[sizeof(struct canfd_frame) = CANFD_MAX_DLC,
> > };
> 
> 
> Nice idea.
> 
> How big is mtu_to_max_can_dlc_table ??
> sizeof(struct canfd_frame) * sizeof(int)
yep. We could argue using uint8_t for it.
Even then, the amount of saved code may compensate?

I wasn't planning to put such table static in the header ...
> 
> Regards,
> Oliver

Kurt

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Wireshark / libpcap - was Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 12:50     ` Oliver Hartkopp
  2012-05-03 13:22       ` Kurt Van Dijck
@ 2012-05-03 18:01       ` Oliver Hartkopp
  2012-05-04  9:29         ` Kurt Van Dijck
  1 sibling, 1 reply; 10+ messages in thread
From: Oliver Hartkopp @ 2012-05-03 18:01 UTC (permalink / raw)
  To: Kurt Van Dijck, Felix Obenhuber; +Cc: linux-can

On 03.05.2012 14:50, Oliver Hartkopp wrote:

> On 03.05.2012 14:37, Kurt Van Dijck wrote:


>> I don't see benefit in introducing ETH_P_CANFD.
>> Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
>> Why not using that info. IMHO, ETH_P_CANFD duplicates the info.
> 
> 
> This has a big advantage when thinking about wireshark & friends.
> 
> You can look into the eth protocol and know the skb data layout.
> The length information is some kind of implicit knowledge.


Hi Kurt,

i just wanted to pick up the ETH_P_CANFD topic you asked for.

As you can see in packet_sendmsg_spkt() here

http://lxr.linux.no/#linux+v3.3.4/net/packet/af_packet.c#L1447

the 'proto' can be set directly when creating a skb with packet sockets:

http://lxr.linux.no/#linux+v3.3.4/net/packet/af_packet.c#L1466

At this point the value is put into the skb:

http://lxr.linux.no/#linux+v3.3.4/net/packet/af_packet.c#L1542


So e.g. the packet socket is definitely aware of the ETH_P_xxx values, try
'man packet' :

(..)
protocol is the IEEE 802.3 protocol number in network order.  See the
<linux/if_ether.h> include file for a list  of allowed  protocols.
When  protocol  is  set  to htons(ETH_P_ALL) then all protocols are received.
All incoming packets of that protocol type will be passed to the packet socket
before they are passed to the protocols implemented in the kernel.
(..)

For that reason and because of the reference to <linux/if_ether.h> i think
it's pretty relevant to wireshark and libpcap - that's why i added Felix in CC.

http://sourceforge.net/tracker/?func=detail&aid=2872132&group_id=53067&atid=469579

Regards,
Oliver


^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: Wireshark / libpcap - was Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 18:01       ` Wireshark / libpcap - was " Oliver Hartkopp
@ 2012-05-04  9:29         ` Kurt Van Dijck
  2012-05-04  9:59           ` Oliver Hartkopp
  0 siblings, 1 reply; 10+ messages in thread
From: Kurt Van Dijck @ 2012-05-04  9:29 UTC (permalink / raw)
  To: Oliver Hartkopp; +Cc: Felix Obenhuber, linux-can

On Thu, May 03, 2012 at 08:01:32PM +0200, Oliver Hartkopp wrote:
> On 03.05.2012 14:50, Oliver Hartkopp wrote:
> 
> > On 03.05.2012 14:37, Kurt Van Dijck wrote:
> 
> 
> >> I don't see benefit in introducing ETH_P_CANFD.
> >> Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
> >> Why not using that info. IMHO, ETH_P_CANFD duplicates the info.
> > 
> > 
> > This has a big advantage when thinking about wireshark & friends.
> > 
> > You can look into the eth protocol and know the skb data layout.
> > The length information is some kind of implicit knowledge.
> 
Oliver,

Thanks for the explanation.
It shows how ETH_P_XXX values escape from kernel space.
It thus promotes to a part of the ABI, which I did not yet realize :-).

But you did not address my other concern, why you use
different values for CAN & CANFD.

> > You can look into the eth protocol and know the skb data layout.
Again, both layouts are compatible.

> > The length information is some kind of implicit knowledge.
Again, there are so many checks to fix the length to one of 2 available
MTU's that I doubt the implicit character of the lenght information
for the CAN situation.

I appreciate your effort in digging up Felix's original posts.
His opinions on this may be very usefull.

Kurt

[...]
> 
> For that reason and because of the reference to <linux/if_ether.h> i think
> it's pretty relevant to wireshark and libpcap - that's why i added Felix in CC.
> 
> http://sourceforge.net/tracker/?func=detail&aid=2872132&group_id=53067&atid=469579
> 
> Regards,
> Oliver
> 

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: Wireshark / libpcap - was Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-04  9:29         ` Kurt Van Dijck
@ 2012-05-04  9:59           ` Oliver Hartkopp
  2012-05-07 12:19             ` Felix Obenhuber
  0 siblings, 1 reply; 10+ messages in thread
From: Oliver Hartkopp @ 2012-05-04  9:59 UTC (permalink / raw)
  To: Kurt Van Dijck; +Cc: Felix Obenhuber, linux-can

On 04.05.2012 11:29, Kurt Van Dijck wrote:

> On Thu, May 03, 2012 at 08:01:32PM +0200, Oliver Hartkopp wrote:
>> On 03.05.2012 14:50, Oliver Hartkopp wrote:
>>
>>> On 03.05.2012 14:37, Kurt Van Dijck wrote:
>>
>>
>>>> I don't see benefit in introducing ETH_P_CANFD.
>>>> Using the skb's payload length differentiates enough between CAN_MTU & CANFD_MTU.
>>>> Why not using that info. IMHO, ETH_P_CANFD duplicates the info.
>>>
>>>
>>> This has a big advantage when thinking about wireshark & friends.
>>>
>>> You can look into the eth protocol and know the skb data layout.
>>> The length information is some kind of implicit knowledge.
>>
> Oliver,
> 
> Thanks for the explanation.
> It shows how ETH_P_XXX values escape from kernel space.
> It thus promotes to a part of the ABI, which I did not yet realize :-).
> 
> But you did not address my other concern, why you use
> different values for CAN & CANFD.
> 
>>> You can look into the eth protocol and know the skb data layout.
> Again, both layouts are compatible.


Hm - nearly.

As the canfd_frame.flags element is new AND you have a different PDU length.

:-)

> 
>>> The length information is some kind of implicit knowledge.
> Again, there are so many checks to fix the length to one of 2 available
> MTU's that I doubt the implicit character of the lenght information
> for the CAN situation.
> 
> I appreciate your effort in digging up Felix's original posts.
> His opinions on this may be very usefull.


Yes - let's see ...

> 
> Kurt
> 
> [...]
>>
>> For that reason and because of the reference to <linux/if_ether.h> i think
>> it's pretty relevant to wireshark and libpcap - that's why i added Felix in CC.
>>
>> http://sourceforge.net/tracker/?func=detail&aid=2872132&group_id=53067&atid=469579
>>
>> Regards,
>> Oliver
>>



^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-03 13:22       ` Kurt Van Dijck
@ 2012-05-07 11:51         ` Felix Obenhuber
  2012-05-07 18:03           ` Oliver Hartkopp
  0 siblings, 1 reply; 10+ messages in thread
From: Felix Obenhuber @ 2012-05-07 11:51 UTC (permalink / raw)
  To: Oliver Hartkopp, linux-can

On Thu, May 3, 2012 at 3:22 PM, Kurt Van Dijck <kurt.van.dijck@eia.be> wrote:
>> This has a big advantage when thinking about wireshark & friends.
> I must have missed something here. ETH_P_CANFD value somewhere in struct skb
> is something kernel specific. Does wireshark look inside kernel?

No, it doesn't. The detection of a SocketCAN socket/interface is done
very hacky with the interface name in pcap.

pcap-linux.c:432:

#ifdef PCAP_SUPPORT_CAN
	if (strstr(device, "can") || strstr(device, "vcan")) {
		return can_create(device, ebuf);
	}
#endif

Felix

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: Wireshark / libpcap - was Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-04  9:59           ` Oliver Hartkopp
@ 2012-05-07 12:19             ` Felix Obenhuber
  0 siblings, 0 replies; 10+ messages in thread
From: Felix Obenhuber @ 2012-05-07 12:19 UTC (permalink / raw)
  To: linux-can

>>>> You can look into the eth protocol and know the skb data layout.
>> Again, both layouts are compatible.
>
>
> Hm - nearly.
>
> As the canfd_frame.flags element is new AND you have a different PDU length.
>
> :-)
>
>>
>>>> The length information is some kind of implicit knowledge.
>> Again, there are so many checks to fix the length to one of 2 available
>> MTU's that I doubt the implicit character of the lenght information
>> for the CAN situation.

If I got you right, then I think there's no need for ETH_P_CANFD from
the pcap/wireshark point of view:

Both currently don't care about ETH_P_CAN. This could be improved,
cause the detection of a CAN interface is currently done via the
interface name. That works - but can drive people crazy when they
setup a different naming that canXX and vcanXX.

To handle the CANFD information in the Wireshark dissector, the packet
length information is sufficient. I think it's easier to manage the
different formats in wireshark and not down in pcap. Pcap is (almost)
not aware of the the packet content at all. Furthermore you can save a
the FD specific DLT.

Felix

^ permalink raw reply	[flat|nested] 10+ messages in thread

* Re: [RFC] CAN FD support part 2 - ideas and commented source
  2012-05-07 11:51         ` Felix Obenhuber
@ 2012-05-07 18:03           ` Oliver Hartkopp
  0 siblings, 0 replies; 10+ messages in thread
From: Oliver Hartkopp @ 2012-05-07 18:03 UTC (permalink / raw)
  To: Felix Obenhuber; +Cc: linux-can

On 07.05.2012 13:51, Felix Obenhuber wrote:

> On Thu, May 3, 2012 at 3:22 PM, Kurt Van Dijck <kurt.van.dijck@eia.be> wrote:
>>> This has a big advantage when thinking about wireshark & friends.
>> I must have missed something here. ETH_P_CANFD value somewhere in struct skb
>> is something kernel specific. Does wireshark look inside kernel?
> 
> No, it doesn't. The detection of a SocketCAN socket/interface is done
> very hacky with the interface name in pcap.
> 
> pcap-linux.c:432:
> 
> #ifdef PCAP_SUPPORT_CAN
> 	if (strstr(device, "can") || strstr(device, "vcan")) {
> 		return can_create(device, ebuf);
> 	}
> #endif


Oh, oh.

I like to have at least one vcan interface named 'helga' on my system.

Obviously i won't have fun with pcap %-/

Regards,
Oliver

^ permalink raw reply	[flat|nested] 10+ messages in thread

end of thread, other threads:[~2012-05-07 18:03 UTC | newest]

Thread overview: 10+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
     [not found] <4FA242D0.2070604@hartkopp.net>
2012-05-03 11:14 ` [RFC] CAN FD support part 2 - ideas and commented source Oliver Hartkopp
2012-05-03 12:37   ` Kurt Van Dijck
2012-05-03 12:50     ` Oliver Hartkopp
2012-05-03 13:22       ` Kurt Van Dijck
2012-05-07 11:51         ` Felix Obenhuber
2012-05-07 18:03           ` Oliver Hartkopp
2012-05-03 18:01       ` Wireshark / libpcap - was " Oliver Hartkopp
2012-05-04  9:29         ` Kurt Van Dijck
2012-05-04  9:59           ` Oliver Hartkopp
2012-05-07 12:19             ` Felix Obenhuber

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.