diff options
| author | Stephane Grosjean <s.grosjean@peak-system.com> | 2015-01-23 05:31:25 -0500 |
|---|---|---|
| committer | Marc Kleine-Budde <mkl@pengutronix.de> | 2015-01-28 08:47:06 -0500 |
| commit | 0a25e1f4f18566b750ebd3ae995af64e23111e63 (patch) | |
| tree | a0921092990874e482a05ecd29ce5b66bcd82c3c | |
| parent | faa233d902bb97ac034fa2580e6e0934ccf98cf3 (diff) | |
can: peak_usb: add support for PEAK new CANFD USB adapters
Add support for the following new PEAK-System technik CANFD USB adapters:
PCAN-USB FD single CANFD channel USB adapter
PCAN-USB Pro FD dual CANFD channels USB adapter
Signed-off-by: Stephane Grosjean <s.grosjean@peak-system.com>
Signed-off-by: Oliver Hartkopp <socketcan@hartkopp.net>
Acked-by: Andri Yngvason <andri.yngvason@marel.com>
Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
| -rw-r--r-- | drivers/net/can/usb/Kconfig | 14 | ||||
| -rw-r--r-- | drivers/net/can/usb/peak_usb/Makefile | 2 | ||||
| -rw-r--r-- | drivers/net/can/usb/peak_usb/pcan_ucan.h | 222 | ||||
| -rw-r--r-- | drivers/net/can/usb/peak_usb/pcan_usb_core.c | 4 | ||||
| -rw-r--r-- | drivers/net/can/usb/peak_usb/pcan_usb_core.h | 4 | ||||
| -rw-r--r-- | drivers/net/can/usb/peak_usb/pcan_usb_fd.c | 1095 |
6 files changed, 1337 insertions, 4 deletions
diff --git a/drivers/net/can/usb/Kconfig b/drivers/net/can/usb/Kconfig index f6f55004fe2b..bcb272f6c68a 100644 --- a/drivers/net/can/usb/Kconfig +++ b/drivers/net/can/usb/Kconfig | |||
| @@ -59,10 +59,18 @@ config CAN_KVASER_USB | |||
| 59 | module will be called kvaser_usb. | 59 | module will be called kvaser_usb. |
| 60 | 60 | ||
| 61 | config CAN_PEAK_USB | 61 | config CAN_PEAK_USB |
| 62 | tristate "PEAK PCAN-USB/USB Pro interfaces" | 62 | tristate "PEAK PCAN-USB/USB Pro interfaces for CAN 2.0b/CAN-FD" |
| 63 | ---help--- | 63 | ---help--- |
| 64 | This driver supports the PCAN-USB and PCAN-USB Pro adapters | 64 | This driver supports the PEAK-System Technik USB adapters that enable |
| 65 | from PEAK-System Technik (http://www.peak-system.com). | 65 | access to the CAN bus, with repect to the CAN 2.0b and/or CAN-FD |
| 66 | standards, that is: | ||
| 67 | |||
| 68 | PCAN-USB single CAN 2.0b channel USB adapter | ||
| 69 | PCAN-USB Pro dual CAN 2.0b channels USB adapter | ||
| 70 | PCAN-USB FD single CAN-FD channel USB adapter | ||
| 71 | PCAN-USB Pro FD dual CAN-FD channels USB adapter | ||
| 72 | |||
| 73 | (see also http://www.peak-system.com). | ||
| 66 | 74 | ||
| 67 | config CAN_8DEV_USB | 75 | config CAN_8DEV_USB |
| 68 | tristate "8 devices USB2CAN interface" | 76 | tristate "8 devices USB2CAN interface" |
diff --git a/drivers/net/can/usb/peak_usb/Makefile b/drivers/net/can/usb/peak_usb/Makefile index 1aefbc88d643..1839e9ca62e7 100644 --- a/drivers/net/can/usb/peak_usb/Makefile +++ b/drivers/net/can/usb/peak_usb/Makefile | |||
| @@ -1,2 +1,2 @@ | |||
| 1 | obj-$(CONFIG_CAN_PEAK_USB) += peak_usb.o | 1 | obj-$(CONFIG_CAN_PEAK_USB) += peak_usb.o |
| 2 | peak_usb-y = pcan_usb_core.o pcan_usb.o pcan_usb_pro.o | 2 | peak_usb-y = pcan_usb_core.o pcan_usb.o pcan_usb_pro.o pcan_usb_fd.o |
diff --git a/drivers/net/can/usb/peak_usb/pcan_ucan.h b/drivers/net/can/usb/peak_usb/pcan_ucan.h new file mode 100644 index 000000000000..1ba7c25002e1 --- /dev/null +++ b/drivers/net/can/usb/peak_usb/pcan_ucan.h | |||
| @@ -0,0 +1,222 @@ | |||
| 1 | /* | ||
| 2 | * CAN driver for PEAK System micro-CAN based adapters | ||
| 3 | * | ||
| 4 | * Copyright (C) 2003-2011 PEAK System-Technik GmbH | ||
| 5 | * Copyright (C) 2011-2013 Stephane Grosjean <s.grosjean@peak-system.com> | ||
| 6 | * | ||
| 7 | * This program is free software; you can redistribute it and/or modify it | ||
| 8 | * under the terms of the GNU General Public License as published | ||
| 9 | * by the Free Software Foundation; version 2 of the License. | ||
| 10 | * | ||
| 11 | * This program is distributed in the hope that it will be useful, but | ||
| 12 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 14 | * General Public License for more details. | ||
| 15 | */ | ||
| 16 | #ifndef PUCAN_H | ||
| 17 | #define PUCAN_H | ||
| 18 | |||
| 19 | /* uCAN commands opcodes list (low-order 10 bits) */ | ||
| 20 | #define PUCAN_CMD_NOP 0x000 | ||
| 21 | #define PUCAN_CMD_RESET_MODE 0x001 | ||
| 22 | #define PUCAN_CMD_NORMAL_MODE 0x002 | ||
| 23 | #define PUCAN_CMD_LISTEN_ONLY_MODE 0x003 | ||
| 24 | #define PUCAN_CMD_TIMING_SLOW 0x004 | ||
| 25 | #define PUCAN_CMD_TIMING_FAST 0x005 | ||
| 26 | #define PUCAN_CMD_FILTER_STD 0x008 | ||
| 27 | #define PUCAN_CMD_TX_ABORT 0x009 | ||
| 28 | #define PUCAN_CMD_WR_ERR_CNT 0x00a | ||
| 29 | #define PUCAN_CMD_RX_FRAME_ENABLE 0x00b | ||
| 30 | #define PUCAN_CMD_RX_FRAME_DISABLE 0x00c | ||
| 31 | #define PUCAN_CMD_END_OF_COLLECTION 0x3ff | ||
| 32 | |||
| 33 | /* uCAN received messages list */ | ||
| 34 | #define PUCAN_MSG_CAN_RX 0x0001 | ||
| 35 | #define PUCAN_MSG_ERROR 0x0002 | ||
| 36 | #define PUCAN_MSG_STATUS 0x0003 | ||
| 37 | #define PUCAN_MSG_BUSLOAD 0x0004 | ||
| 38 | #define PUCAN_MSG_CAN_TX 0x1000 | ||
| 39 | |||
| 40 | /* uCAN command common header */ | ||
| 41 | struct __packed pucan_command { | ||
| 42 | __le16 opcode_channel; | ||
| 43 | u16 args[3]; | ||
| 44 | }; | ||
| 45 | |||
| 46 | /* uCAN TIMING_SLOW command fields */ | ||
| 47 | #define PUCAN_TSLOW_SJW_T(s, t) (((s) & 0xf) | ((!!(t)) << 7)) | ||
| 48 | #define PUCAN_TSLOW_TSEG2(t) ((t) & 0xf) | ||
| 49 | #define PUCAN_TSLOW_TSEG1(t) ((t) & 0x3f) | ||
| 50 | #define PUCAN_TSLOW_BRP(b) ((b) & 0x3ff) | ||
| 51 | |||
| 52 | struct __packed pucan_timing_slow { | ||
| 53 | __le16 opcode_channel; | ||
| 54 | |||
| 55 | u8 ewl; /* Error Warning limit */ | ||
| 56 | u8 sjw_t; /* Sync Jump Width + Triple sampling */ | ||
| 57 | u8 tseg2; /* Timing SEGment 2 */ | ||
| 58 | u8 tseg1; /* Timing SEGment 1 */ | ||
| 59 | |||
| 60 | __le16 brp; /* BaudRate Prescaler */ | ||
| 61 | }; | ||
| 62 | |||
| 63 | /* uCAN TIMING_FAST command fields */ | ||
| 64 | #define PUCAN_TFAST_SJW(s) ((s) & 0x3) | ||
| 65 | #define PUCAN_TFAST_TSEG2(t) ((t) & 0x7) | ||
| 66 | #define PUCAN_TFAST_TSEG1(t) ((t) & 0xf) | ||
| 67 | #define PUCAN_TFAST_BRP(b) ((b) & 0x3ff) | ||
| 68 | |||
| 69 | struct __packed pucan_timing_fast { | ||
| 70 | __le16 opcode_channel; | ||
| 71 | |||
| 72 | u8 unused; | ||
| 73 | u8 sjw; /* Sync Jump Width */ | ||
| 74 | u8 tseg2; /* Timing SEGment 2 */ | ||
| 75 | u8 tseg1; /* Timing SEGment 1 */ | ||
| 76 | |||
| 77 | __le16 brp; /* BaudRate Prescaler */ | ||
| 78 | }; | ||
| 79 | |||
| 80 | /* uCAN FILTER_STD command fields */ | ||
| 81 | #define PUCAN_FLTSTD_ROW_IDX_BITS 6 | ||
| 82 | |||
| 83 | struct __packed pucan_filter_std { | ||
| 84 | __le16 opcode_channel; | ||
| 85 | |||
| 86 | __le16 idx; | ||
| 87 | __le32 mask; /* CAN-ID bitmask in idx range */ | ||
| 88 | }; | ||
| 89 | |||
| 90 | /* uCAN WR_ERR_CNT command fields */ | ||
| 91 | #define PUCAN_WRERRCNT_TE 0x4000 /* Tx error cntr write Enable */ | ||
| 92 | #define PUCAN_WRERRCNT_RE 0x8000 /* Rx error cntr write Enable */ | ||
| 93 | |||
| 94 | struct __packed pucan_wr_err_cnt { | ||
| 95 | __le16 opcode_channel; | ||
| 96 | |||
| 97 | __le16 sel_mask; | ||
| 98 | u8 tx_counter; /* Tx error counter new value */ | ||
| 99 | u8 rx_counter; /* Rx error counter new value */ | ||
| 100 | |||
| 101 | u16 unused; | ||
| 102 | }; | ||
| 103 | |||
| 104 | /* uCAN RX_FRAME_ENABLE command fields */ | ||
| 105 | #define PUCAN_FLTEXT_ERROR 0x0001 | ||
| 106 | #define PUCAN_FLTEXT_BUSLOAD 0x0002 | ||
| 107 | |||
| 108 | struct __packed pucan_filter_ext { | ||
| 109 | __le16 opcode_channel; | ||
| 110 | |||
| 111 | __le16 ext_mask; | ||
| 112 | u32 unused; | ||
| 113 | }; | ||
| 114 | |||
| 115 | /* uCAN received messages global format */ | ||
| 116 | struct __packed pucan_msg { | ||
| 117 | __le16 size; | ||
| 118 | __le16 type; | ||
| 119 | __le32 ts_low; | ||
| 120 | __le32 ts_high; | ||
| 121 | }; | ||
| 122 | |||
| 123 | /* uCAN flags for CAN/CANFD messages */ | ||
| 124 | #define PUCAN_MSG_SELF_RECEIVE 0x80 | ||
| 125 | #define PUCAN_MSG_ERROR_STATE_IND 0x40 /* error state indicator */ | ||
| 126 | #define PUCAN_MSG_BITRATE_SWITCH 0x20 /* bitrate switch */ | ||
| 127 | #define PUCAN_MSG_EXT_DATA_LEN 0x10 /* extended data length */ | ||
| 128 | #define PUCAN_MSG_SINGLE_SHOT 0x08 | ||
| 129 | #define PUCAN_MSG_LOOPED_BACK 0x04 | ||
| 130 | #define PUCAN_MSG_EXT_ID 0x02 | ||
| 131 | #define PUCAN_MSG_RTR 0x01 | ||
| 132 | |||
| 133 | struct __packed pucan_rx_msg { | ||
| 134 | __le16 size; | ||
| 135 | __le16 type; | ||
| 136 | __le32 ts_low; | ||
| 137 | __le32 ts_high; | ||
| 138 | __le32 tag_low; | ||
| 139 | __le32 tag_high; | ||
| 140 | u8 channel_dlc; | ||
| 141 | u8 client; | ||
| 142 | __le16 flags; | ||
| 143 | __le32 can_id; | ||
| 144 | u8 d[0]; | ||
| 145 | }; | ||
| 146 | |||
| 147 | /* uCAN error types */ | ||
| 148 | #define PUCAN_ERMSG_BIT_ERROR 0 | ||
| 149 | #define PUCAN_ERMSG_FORM_ERROR 1 | ||
| 150 | #define PUCAN_ERMSG_STUFF_ERROR 2 | ||
| 151 | #define PUCAN_ERMSG_OTHER_ERROR 3 | ||
| 152 | #define PUCAN_ERMSG_ERR_CNT_DEC 4 | ||
| 153 | |||
| 154 | struct __packed pucan_error_msg { | ||
| 155 | __le16 size; | ||
| 156 | __le16 type; | ||
| 157 | __le32 ts_low; | ||
| 158 | __le32 ts_high; | ||
| 159 | u8 channel_type_d; | ||
| 160 | u8 code_g; | ||
| 161 | u8 tx_err_cnt; | ||
| 162 | u8 rx_err_cnt; | ||
| 163 | }; | ||
| 164 | |||
| 165 | #define PUCAN_BUS_PASSIVE 0x20 | ||
| 166 | #define PUCAN_BUS_WARNING 0x40 | ||
| 167 | #define PUCAN_BUS_BUSOFF 0x80 | ||
| 168 | |||
| 169 | struct __packed pucan_status_msg { | ||
| 170 | __le16 size; | ||
| 171 | __le16 type; | ||
| 172 | __le32 ts_low; | ||
| 173 | __le32 ts_high; | ||
| 174 | u8 channel_p_w_b; | ||
| 175 | u8 unused[3]; | ||
| 176 | }; | ||
| 177 | |||
| 178 | /* uCAN transmitted message format */ | ||
| 179 | #define PUCAN_MSG_CHANNEL_DLC(c, d) (((c) & 0xf) | ((d) << 4)) | ||
| 180 | |||
| 181 | struct __packed pucan_tx_msg { | ||
| 182 | __le16 size; | ||
| 183 | __le16 type; | ||
| 184 | __le32 tag_low; | ||
| 185 | __le32 tag_high; | ||
| 186 | u8 channel_dlc; | ||
| 187 | u8 client; | ||
| 188 | __le16 flags; | ||
| 189 | __le32 can_id; | ||
| 190 | u8 d[0]; | ||
| 191 | }; | ||
| 192 | |||
| 193 | /* build the cmd opcode_channel field with respect to the correct endianness */ | ||
| 194 | static inline __le16 pucan_cmd_opcode_channel(struct peak_usb_device *dev, | ||
| 195 | int opcode) | ||
| 196 | { | ||
| 197 | return cpu_to_le16(((dev->ctrl_idx) << 12) | ((opcode) & 0x3ff)); | ||
| 198 | } | ||
| 199 | |||
| 200 | /* return the channel number part from any received message channel_dlc field */ | ||
| 201 | static inline int pucan_msg_get_channel(struct pucan_rx_msg *rm) | ||
| 202 | { | ||
| 203 | return rm->channel_dlc & 0xf; | ||
| 204 | } | ||
| 205 | |||
| 206 | /* return the dlc value from any received message channel_dlc field */ | ||
| 207 | static inline int pucan_msg_get_dlc(struct pucan_rx_msg *rm) | ||
| 208 | { | ||
| 209 | return rm->channel_dlc >> 4; | ||
| 210 | } | ||
| 211 | |||
| 212 | static inline int pucan_ermsg_get_channel(struct pucan_error_msg *em) | ||
| 213 | { | ||
| 214 | return em->channel_type_d & 0x0f; | ||
| 215 | } | ||
| 216 | |||
| 217 | static inline int pucan_stmsg_get_channel(struct pucan_status_msg *sm) | ||
| 218 | { | ||
| 219 | return sm->channel_p_w_b & 0x0f; | ||
| 220 | } | ||
| 221 | |||
| 222 | #endif | ||
diff --git a/drivers/net/can/usb/peak_usb/pcan_usb_core.c b/drivers/net/can/usb/peak_usb/pcan_usb_core.c index 7587337a3b42..7921cff93a63 100644 --- a/drivers/net/can/usb/peak_usb/pcan_usb_core.c +++ b/drivers/net/can/usb/peak_usb/pcan_usb_core.c | |||
| @@ -37,6 +37,8 @@ MODULE_LICENSE("GPL v2"); | |||
| 37 | static struct usb_device_id peak_usb_table[] = { | 37 | static struct usb_device_id peak_usb_table[] = { |
| 38 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USB_PRODUCT_ID)}, | 38 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USB_PRODUCT_ID)}, |
| 39 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USBPRO_PRODUCT_ID)}, | 39 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USBPRO_PRODUCT_ID)}, |
| 40 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USBFD_PRODUCT_ID)}, | ||
| 41 | {USB_DEVICE(PCAN_USB_VENDOR_ID, PCAN_USBPROFD_PRODUCT_ID)}, | ||
| 40 | {} /* Terminating entry */ | 42 | {} /* Terminating entry */ |
| 41 | }; | 43 | }; |
| 42 | 44 | ||
| @@ -46,6 +48,8 @@ MODULE_DEVICE_TABLE(usb, peak_usb_table); | |||
| 46 | static const struct peak_usb_adapter *const peak_usb_adapters_list[] = { | 48 | static const struct peak_usb_adapter *const peak_usb_adapters_list[] = { |
| 47 | &pcan_usb, | 49 | &pcan_usb, |
| 48 | &pcan_usb_pro, | 50 | &pcan_usb_pro, |
| 51 | &pcan_usb_fd, | ||
| 52 | &pcan_usb_pro_fd, | ||
| 49 | }; | 53 | }; |
| 50 | 54 | ||
| 51 | /* | 55 | /* |
diff --git a/drivers/net/can/usb/peak_usb/pcan_usb_core.h b/drivers/net/can/usb/peak_usb/pcan_usb_core.h index 0e76b84fb786..9e624f05ad4d 100644 --- a/drivers/net/can/usb/peak_usb/pcan_usb_core.h +++ b/drivers/net/can/usb/peak_usb/pcan_usb_core.h | |||
| @@ -25,6 +25,8 @@ | |||
| 25 | /* supported device ids. */ | 25 | /* supported device ids. */ |
| 26 | #define PCAN_USB_PRODUCT_ID 0x000c | 26 | #define PCAN_USB_PRODUCT_ID 0x000c |
| 27 | #define PCAN_USBPRO_PRODUCT_ID 0x000d | 27 | #define PCAN_USBPRO_PRODUCT_ID 0x000d |
| 28 | #define PCAN_USBPROFD_PRODUCT_ID 0x0011 | ||
| 29 | #define PCAN_USBFD_PRODUCT_ID 0x0012 | ||
| 28 | 30 | ||
| 29 | #define PCAN_USB_DRIVER_NAME "peak_usb" | 31 | #define PCAN_USB_DRIVER_NAME "peak_usb" |
| 30 | 32 | ||
| @@ -86,6 +88,8 @@ struct peak_usb_adapter { | |||
| 86 | 88 | ||
| 87 | extern const struct peak_usb_adapter pcan_usb; | 89 | extern const struct peak_usb_adapter pcan_usb; |
| 88 | extern const struct peak_usb_adapter pcan_usb_pro; | 90 | extern const struct peak_usb_adapter pcan_usb_pro; |
| 91 | extern const struct peak_usb_adapter pcan_usb_fd; | ||
| 92 | extern const struct peak_usb_adapter pcan_usb_pro_fd; | ||
| 89 | 93 | ||
| 90 | struct peak_time_ref { | 94 | struct peak_time_ref { |
| 91 | struct timeval tv_host_0, tv_host; | 95 | struct timeval tv_host_0, tv_host; |
diff --git a/drivers/net/can/usb/peak_usb/pcan_usb_fd.c b/drivers/net/can/usb/peak_usb/pcan_usb_fd.c new file mode 100644 index 000000000000..962c3f027383 --- /dev/null +++ b/drivers/net/can/usb/peak_usb/pcan_usb_fd.c | |||
| @@ -0,0 +1,1095 @@ | |||
| 1 | /* | ||
| 2 | * CAN driver for PEAK System PCAN-USB FD / PCAN-USB Pro FD adapter | ||
| 3 | * | ||
| 4 | * Copyright (C) 2013-2014 Stephane Grosjean <s.grosjean@peak-system.com> | ||
| 5 | * | ||
| 6 | * This program is free software; you can redistribute it and/or modify it | ||
| 7 | * under the terms of the GNU General Public License as published | ||
| 8 | * by the Free Software Foundation; version 2 of the License. | ||
| 9 | * | ||
| 10 | * This program is distributed in the hope that it will be useful, but | ||
| 11 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 13 | * General Public License for more details. | ||
| 14 | */ | ||
| 15 | #include <linux/netdevice.h> | ||
| 16 | #include <linux/usb.h> | ||
| 17 | #include <linux/module.h> | ||
| 18 | |||
| 19 | #include <linux/can.h> | ||
| 20 | #include <linux/can/dev.h> | ||
| 21 | #include <linux/can/error.h> | ||
| 22 | |||
| 23 | #include "pcan_usb_core.h" | ||
| 24 | #include "pcan_usb_pro.h" | ||
| 25 | #include "pcan_ucan.h" | ||
| 26 | |||
| 27 | MODULE_SUPPORTED_DEVICE("PEAK-System PCAN-USB FD adapter"); | ||
| 28 | MODULE_SUPPORTED_DEVICE("PEAK-System PCAN-USB Pro FD adapter"); | ||
| 29 | |||
| 30 | #define PCAN_USBPROFD_CHANNEL_COUNT 2 | ||
| 31 | #define PCAN_USBFD_CHANNEL_COUNT 1 | ||
| 32 | |||
| 33 | /* PCAN-USB Pro FD adapter internal clock (Hz) */ | ||
| 34 | #define PCAN_UFD_CRYSTAL_HZ 80000000 | ||
| 35 | |||
| 36 | #define PCAN_UFD_CMD_BUFFER_SIZE 512 | ||
| 37 | #define PCAN_UFD_LOSPD_PKT_SIZE 64 | ||
| 38 | |||
| 39 | /* PCAN-USB Pro FD command timeout (ms.) */ | ||
| 40 | #define PCAN_UFD_CMD_TIMEOUT_MS 1000 | ||
| 41 | |||
| 42 | /* PCAN-USB Pro FD rx/tx buffers size */ | ||
| 43 | #define PCAN_UFD_RX_BUFFER_SIZE 2048 | ||
| 44 | #define PCAN_UFD_TX_BUFFER_SIZE 512 | ||
| 45 | |||
| 46 | /* read some versions info from the hw devcie */ | ||
| 47 | struct __packed pcan_ufd_fw_info { | ||
| 48 | __le16 size_of; /* sizeof this */ | ||
| 49 | __le16 type; /* type of this structure */ | ||
| 50 | u8 hw_type; /* Type of hardware (HW_TYPE_xxx) */ | ||
| 51 | u8 bl_version[3]; /* Bootloader version */ | ||
| 52 | u8 hw_version; /* Hardware version (PCB) */ | ||
| 53 | u8 fw_version[3]; /* Firmware version */ | ||
| 54 | __le32 dev_id[2]; /* "device id" per CAN */ | ||
| 55 | __le32 ser_no; /* S/N */ | ||
| 56 | __le32 flags; /* special functions */ | ||
| 57 | }; | ||
| 58 | |||
| 59 | /* handle device specific info used by the netdevices */ | ||
| 60 | struct pcan_usb_fd_if { | ||
| 61 | struct peak_usb_device *dev[PCAN_USB_MAX_CHANNEL]; | ||
| 62 | struct pcan_ufd_fw_info fw_info; | ||
| 63 | struct peak_time_ref time_ref; | ||
| 64 | int cm_ignore_count; | ||
| 65 | int dev_opened_count; | ||
| 66 | }; | ||
| 67 | |||
| 68 | /* device information */ | ||
| 69 | struct pcan_usb_fd_device { | ||
| 70 | struct peak_usb_device dev; | ||
| 71 | struct can_berr_counter bec; | ||
| 72 | struct pcan_usb_fd_if *usb_if; | ||
| 73 | u8 *cmd_buffer_addr; | ||
| 74 | }; | ||
| 75 | |||
| 76 | /* Extended USB commands (non uCAN commands) */ | ||
| 77 | |||
| 78 | /* Clock Modes command */ | ||
| 79 | #define PCAN_UFD_CMD_CLK_SET 0x80 | ||
| 80 | |||
| 81 | #define PCAN_UFD_CLK_80MHZ 0x0 | ||
| 82 | #define PCAN_UFD_CLK_60MHZ 0x1 | ||
| 83 | #define PCAN_UFD_CLK_40MHZ 0x2 | ||
| 84 | #define PCAN_UFD_CLK_30MHZ 0x3 | ||
| 85 | #define PCAN_UFD_CLK_24MHZ 0x4 | ||
| 86 | #define PCAN_UFD_CLK_20MHZ 0x5 | ||
| 87 | #define PCAN_UFD_CLK_DEF PCAN_UFD_CLK_80MHZ | ||
| 88 | |||
| 89 | struct __packed pcan_ufd_clock { | ||
| 90 | __le16 opcode_channel; | ||
| 91 | |||
| 92 | u8 mode; | ||
| 93 | u8 unused[5]; | ||
| 94 | }; | ||
| 95 | |||
| 96 | /* LED control command */ | ||
| 97 | #define PCAN_UFD_CMD_LED_SET 0x86 | ||
| 98 | |||
| 99 | #define PCAN_UFD_LED_DEV 0x00 | ||
| 100 | #define PCAN_UFD_LED_FAST 0x01 | ||
| 101 | #define PCAN_UFD_LED_SLOW 0x02 | ||
| 102 | #define PCAN_UFD_LED_ON 0x03 | ||
| 103 | #define PCAN_UFD_LED_OFF 0x04 | ||
| 104 | #define PCAN_UFD_LED_DEF PCAN_UFD_LED_DEV | ||
| 105 | |||
| 106 | struct __packed pcan_ufd_led { | ||
| 107 | __le16 opcode_channel; | ||
| 108 | |||
| 109 | u8 mode; | ||
| 110 | u8 unused[5]; | ||
| 111 | }; | ||
| 112 | |||
| 113 | /* Extended usage of uCAN commands CMD_RX_FRAME_xxxABLE for PCAN-USB Pro FD */ | ||
| 114 | #define PCAN_UFD_FLTEXT_CALIBRATION 0x8000 | ||
| 115 | |||
| 116 | struct __packed pcan_ufd_filter_ext { | ||
| 117 | __le16 opcode_channel; | ||
| 118 | |||
| 119 | __le16 ext_mask; | ||
| 120 | u16 unused; | ||
| 121 | __le16 usb_mask; | ||
| 122 | }; | ||
| 123 | |||
| 124 | /* Extended usage of uCAN messages for PCAN-USB Pro FD */ | ||
| 125 | #define PCAN_UFD_MSG_CALIBRATION 0x100 | ||
| 126 | |||
| 127 | struct __packed pcan_ufd_ts_msg { | ||
| 128 | __le16 size; | ||
| 129 | __le16 type; | ||
| 130 | __le32 ts_low; | ||
| 131 | __le32 ts_high; | ||
| 132 | __le16 usb_frame_index; | ||
| 133 | u16 unused; | ||
| 134 | }; | ||
| 135 | |||
| 136 | #define PCAN_UFD_MSG_OVERRUN 0x101 | ||
| 137 | |||
| 138 | #define PCAN_UFD_OVMSG_CHANNEL(o) ((o)->channel & 0xf) | ||
| 139 | |||
| 140 | struct __packed pcan_ufd_ovr_msg { | ||
| 141 | __le16 size; | ||
| 142 | __le16 type; | ||
| 143 | __le32 ts_low; | ||
| 144 | __le32 ts_high; | ||
| 145 | u8 channel; | ||
| 146 | u8 unused[3]; | ||
| 147 | }; | ||
| 148 | |||
| 149 | static inline int pufd_omsg_get_channel(struct pcan_ufd_ovr_msg *om) | ||
| 150 | { | ||
| 151 | return om->channel & 0xf; | ||
| 152 | } | ||
| 153 | |||
| 154 | /* Clock mode frequency values */ | ||
| 155 | static const u32 pcan_usb_fd_clk_freq[6] = { | ||
| 156 | [PCAN_UFD_CLK_80MHZ] = 80000000, | ||
| 157 | [PCAN_UFD_CLK_60MHZ] = 60000000, | ||
| 158 | [PCAN_UFD_CLK_40MHZ] = 40000000, | ||
| 159 | [PCAN_UFD_CLK_30MHZ] = 30000000, | ||
| 160 | [PCAN_UFD_CLK_24MHZ] = 24000000, | ||
| 161 | [PCAN_UFD_CLK_20MHZ] = 20000000 | ||
| 162 | }; | ||
| 163 | |||
| 164 | /* return a device USB interface */ | ||
| 165 | static inline | ||
| 166 | struct pcan_usb_fd_if *pcan_usb_fd_dev_if(struct peak_usb_device *dev) | ||
| 167 | { | ||
| 168 | struct pcan_usb_fd_device *pdev = | ||
| 169 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 170 | return pdev->usb_if; | ||
| 171 | } | ||
| 172 | |||
| 173 | /* return a device USB commands buffer */ | ||
| 174 | static inline void *pcan_usb_fd_cmd_buffer(struct peak_usb_device *dev) | ||
| 175 | { | ||
| 176 | struct pcan_usb_fd_device *pdev = | ||
| 177 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 178 | return pdev->cmd_buffer_addr; | ||
| 179 | } | ||
| 180 | |||
| 181 | /* send PCAN-USB Pro FD commands synchronously */ | ||
| 182 | static int pcan_usb_fd_send_cmd(struct peak_usb_device *dev, void *cmd_tail) | ||
| 183 | { | ||
| 184 | void *cmd_head = pcan_usb_fd_cmd_buffer(dev); | ||
| 185 | int err; | ||
| 186 | u8 *packet_ptr; | ||
| 187 | int i, n = 1, packet_len; | ||
| 188 | ptrdiff_t cmd_len; | ||
| 189 | |||
| 190 | /* usb device unregistered? */ | ||
| 191 | if (!(dev->state & PCAN_USB_STATE_CONNECTED)) | ||
| 192 | return 0; | ||
| 193 | |||
| 194 | /* if a packet is not filled completely by commands, the command list | ||
| 195 | * is terminated with an "end of collection" record. | ||
| 196 | */ | ||
| 197 | cmd_len = cmd_tail - cmd_head; | ||
| 198 | if (cmd_len <= (PCAN_UFD_CMD_BUFFER_SIZE - sizeof(u64))) { | ||
| 199 | memset(cmd_tail, 0xff, sizeof(u64)); | ||
| 200 | cmd_len += sizeof(u64); | ||
| 201 | } | ||
| 202 | |||
| 203 | packet_ptr = cmd_head; | ||
| 204 | |||
| 205 | /* firmware is not able to re-assemble 512 bytes buffer in full-speed */ | ||
| 206 | if ((dev->udev->speed != USB_SPEED_HIGH) && | ||
| 207 | (cmd_len > PCAN_UFD_LOSPD_PKT_SIZE)) { | ||
| 208 | packet_len = PCAN_UFD_LOSPD_PKT_SIZE; | ||
| 209 | n += cmd_len / packet_len; | ||
| 210 | } else { | ||
| 211 | packet_len = cmd_len; | ||
| 212 | } | ||
| 213 | |||
| 214 | for (i = 0; i < n; i++) { | ||
| 215 | err = usb_bulk_msg(dev->udev, | ||
| 216 | usb_sndbulkpipe(dev->udev, | ||
| 217 | PCAN_USBPRO_EP_CMDOUT), | ||
| 218 | packet_ptr, packet_len, | ||
| 219 | NULL, PCAN_UFD_CMD_TIMEOUT_MS); | ||
| 220 | if (err) { | ||
| 221 | netdev_err(dev->netdev, | ||
| 222 | "sending command failure: %d\n", err); | ||
| 223 | break; | ||
| 224 | } | ||
| 225 | |||
| 226 | packet_ptr += packet_len; | ||
| 227 | } | ||
| 228 | |||
| 229 | return err; | ||
| 230 | } | ||
| 231 | |||
| 232 | /* build the commands list in the given buffer, to enter operational mode */ | ||
| 233 | static int pcan_usb_fd_build_restart_cmd(struct peak_usb_device *dev, u8 *buf) | ||
| 234 | { | ||
| 235 | struct pucan_wr_err_cnt *prc; | ||
| 236 | struct pucan_command *cmd; | ||
| 237 | u8 *pc = buf; | ||
| 238 | |||
| 239 | /* 1st, reset error counters: */ | ||
| 240 | prc = (struct pucan_wr_err_cnt *)pc; | ||
| 241 | prc->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 242 | PUCAN_CMD_WR_ERR_CNT); | ||
| 243 | |||
| 244 | /* select both counters */ | ||
| 245 | prc->sel_mask = cpu_to_le16(PUCAN_WRERRCNT_TE|PUCAN_WRERRCNT_RE); | ||
| 246 | |||
| 247 | /* and reset their values */ | ||
| 248 | prc->tx_counter = 0; | ||
| 249 | prc->rx_counter = 0; | ||
| 250 | |||
| 251 | /* moves the pointer forward */ | ||
| 252 | pc += sizeof(struct pucan_wr_err_cnt); | ||
| 253 | |||
| 254 | /* next, go back to operational mode */ | ||
| 255 | cmd = (struct pucan_command *)pc; | ||
| 256 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 257 | (dev->can.ctrlmode & CAN_CTRLMODE_LISTENONLY) ? | ||
| 258 | PUCAN_CMD_LISTEN_ONLY_MODE : | ||
| 259 | PUCAN_CMD_NORMAL_MODE); | ||
| 260 | pc += sizeof(struct pucan_command); | ||
| 261 | |||
| 262 | return pc - buf; | ||
| 263 | } | ||
| 264 | |||
| 265 | /* set CAN bus on/off */ | ||
| 266 | static int pcan_usb_fd_set_bus(struct peak_usb_device *dev, u8 onoff) | ||
| 267 | { | ||
| 268 | u8 *pc = pcan_usb_fd_cmd_buffer(dev); | ||
| 269 | int l; | ||
| 270 | |||
| 271 | if (onoff) { | ||
| 272 | /* build the cmds list to enter operational mode */ | ||
| 273 | l = pcan_usb_fd_build_restart_cmd(dev, pc); | ||
| 274 | } else { | ||
| 275 | struct pucan_command *cmd = (struct pucan_command *)pc; | ||
| 276 | |||
| 277 | /* build cmd to go back to reset mode */ | ||
| 278 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 279 | PUCAN_CMD_RESET_MODE); | ||
| 280 | l = sizeof(struct pucan_command); | ||
| 281 | } | ||
| 282 | |||
| 283 | /* send the command */ | ||
| 284 | return pcan_usb_fd_send_cmd(dev, pc + l); | ||
| 285 | } | ||
| 286 | |||
| 287 | /* set filtering masks: | ||
| 288 | * | ||
| 289 | * idx in range [0..63] selects a row #idx, all rows otherwise | ||
| 290 | * mask in range [0..0xffffffff] defines up to 32 CANIDs in the row(s) | ||
| 291 | * | ||
| 292 | * Each bit of this 64 x 32 bits array defines a CANID value: | ||
| 293 | * | ||
| 294 | * bit[i,j] = 1 implies that CANID=(i x 32)+j will be received, while | ||
| 295 | * bit[i,j] = 0 implies that CANID=(i x 32)+j will be discarded. | ||
| 296 | */ | ||
| 297 | static int pcan_usb_fd_set_filter_std(struct peak_usb_device *dev, int idx, | ||
| 298 | u32 mask) | ||
| 299 | { | ||
| 300 | struct pucan_filter_std *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 301 | int i, n; | ||
| 302 | |||
| 303 | /* select all rows when idx is out of range [0..63] */ | ||
| 304 | if ((idx < 0) || (idx >= (1 << PUCAN_FLTSTD_ROW_IDX_BITS))) { | ||
| 305 | n = 1 << PUCAN_FLTSTD_ROW_IDX_BITS; | ||
| 306 | idx = 0; | ||
| 307 | |||
| 308 | /* select the row (and only the row) otherwise */ | ||
| 309 | } else { | ||
| 310 | n = idx + 1; | ||
| 311 | } | ||
| 312 | |||
| 313 | for (i = idx; i < n; i++, cmd++) { | ||
| 314 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 315 | PUCAN_CMD_FILTER_STD); | ||
| 316 | cmd->idx = cpu_to_le16(i); | ||
| 317 | cmd->mask = cpu_to_le32(mask); | ||
| 318 | } | ||
| 319 | |||
| 320 | /* send the command */ | ||
| 321 | return pcan_usb_fd_send_cmd(dev, cmd); | ||
| 322 | } | ||
| 323 | |||
| 324 | /* set/unset notifications filter: | ||
| 325 | * | ||
| 326 | * onoff sets(1)/unset(0) notifications | ||
| 327 | * mask each bit defines a kind of notification to set/unset | ||
| 328 | */ | ||
| 329 | static int pcan_usb_fd_set_filter_ext(struct peak_usb_device *dev, | ||
| 330 | bool onoff, u16 ext_mask, u16 usb_mask) | ||
| 331 | { | ||
| 332 | struct pcan_ufd_filter_ext *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 333 | |||
| 334 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 335 | (onoff) ? PUCAN_CMD_RX_FRAME_ENABLE : | ||
| 336 | PUCAN_CMD_RX_FRAME_DISABLE); | ||
| 337 | |||
| 338 | cmd->ext_mask = cpu_to_le16(ext_mask); | ||
| 339 | cmd->usb_mask = cpu_to_le16(usb_mask); | ||
| 340 | |||
| 341 | /* send the command */ | ||
| 342 | return pcan_usb_fd_send_cmd(dev, ++cmd); | ||
| 343 | } | ||
| 344 | |||
| 345 | /* setup LED control */ | ||
| 346 | static int pcan_usb_fd_set_can_led(struct peak_usb_device *dev, u8 led_mode) | ||
| 347 | { | ||
| 348 | struct pcan_ufd_led *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 349 | |||
| 350 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 351 | PCAN_UFD_CMD_LED_SET); | ||
| 352 | cmd->mode = led_mode; | ||
| 353 | |||
| 354 | /* send the command */ | ||
| 355 | return pcan_usb_fd_send_cmd(dev, ++cmd); | ||
| 356 | } | ||
| 357 | |||
| 358 | /* set CAN clock domain */ | ||
| 359 | static int pcan_usb_fd_set_clock_domain(struct peak_usb_device *dev, | ||
| 360 | u8 clk_mode) | ||
| 361 | { | ||
| 362 | struct pcan_ufd_clock *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 363 | |||
| 364 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 365 | PCAN_UFD_CMD_CLK_SET); | ||
| 366 | cmd->mode = clk_mode; | ||
| 367 | |||
| 368 | /* send the command */ | ||
| 369 | return pcan_usb_fd_send_cmd(dev, ++cmd); | ||
| 370 | } | ||
| 371 | |||
| 372 | /* set bittiming for CAN and CAN-FD header */ | ||
| 373 | static int pcan_usb_fd_set_bittiming_slow(struct peak_usb_device *dev, | ||
| 374 | struct can_bittiming *bt) | ||
| 375 | { | ||
| 376 | struct pucan_timing_slow *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 377 | |||
| 378 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 379 | PUCAN_CMD_TIMING_SLOW); | ||
| 380 | cmd->sjw_t = PUCAN_TSLOW_SJW_T(bt->sjw - 1, | ||
| 381 | dev->can.ctrlmode & CAN_CTRLMODE_3_SAMPLES); | ||
| 382 | |||
| 383 | cmd->tseg2 = PUCAN_TSLOW_TSEG2(bt->phase_seg2 - 1); | ||
| 384 | cmd->tseg1 = PUCAN_TSLOW_TSEG1(bt->prop_seg + bt->phase_seg1 - 1); | ||
| 385 | cmd->brp = cpu_to_le16(PUCAN_TSLOW_BRP(bt->brp - 1)); | ||
| 386 | |||
| 387 | cmd->ewl = 96; /* default */ | ||
| 388 | |||
| 389 | /* send the command */ | ||
| 390 | return pcan_usb_fd_send_cmd(dev, ++cmd); | ||
| 391 | } | ||
| 392 | |||
| 393 | /* set CAN-FD bittiming for data */ | ||
| 394 | static int pcan_usb_fd_set_bittiming_fast(struct peak_usb_device *dev, | ||
| 395 | struct can_bittiming *bt) | ||
| 396 | { | ||
| 397 | struct pucan_timing_fast *cmd = pcan_usb_fd_cmd_buffer(dev); | ||
| 398 | |||
| 399 | cmd->opcode_channel = pucan_cmd_opcode_channel(dev, | ||
| 400 | PUCAN_CMD_TIMING_FAST); | ||
| 401 | cmd->sjw = PUCAN_TFAST_SJW(bt->sjw - 1); | ||
| 402 | cmd->tseg2 = PUCAN_TFAST_TSEG2(bt->phase_seg2 - 1); | ||
| 403 | cmd->tseg1 = PUCAN_TFAST_TSEG1(bt->prop_seg + bt->phase_seg1 - 1); | ||
| 404 | cmd->brp = cpu_to_le16(PUCAN_TFAST_BRP(bt->brp - 1)); | ||
| 405 | |||
| 406 | /* send the command */ | ||
| 407 | return pcan_usb_fd_send_cmd(dev, ++cmd); | ||
| 408 | } | ||
| 409 | |||
| 410 | /* handle restart but in asynchronously way | ||
| 411 | * (uses PCAN-USB Pro code to complete asynchronous request) | ||
| 412 | */ | ||
| 413 | static int pcan_usb_fd_restart_async(struct peak_usb_device *dev, | ||
| 414 | struct urb *urb, u8 *buf) | ||
| 415 | { | ||
| 416 | u8 *pc = buf; | ||
| 417 | |||
| 418 | /* build the entire cmds list in the provided buffer, to go back into | ||
| 419 | * operational mode. | ||
| 420 | */ | ||
| 421 | pc += pcan_usb_fd_build_restart_cmd(dev, pc); | ||
| 422 | |||
| 423 | /* add EOC */ | ||
| 424 | memset(pc, 0xff, sizeof(struct pucan_command)); | ||
| 425 | pc += sizeof(struct pucan_command); | ||
| 426 | |||
| 427 | /* complete the URB */ | ||
| 428 | usb_fill_bulk_urb(urb, dev->udev, | ||
| 429 | usb_sndbulkpipe(dev->udev, PCAN_USBPRO_EP_CMDOUT), | ||
| 430 | buf, pc - buf, | ||
| 431 | pcan_usb_pro_restart_complete, dev); | ||
| 432 | |||
| 433 | /* and submit it. */ | ||
| 434 | return usb_submit_urb(urb, GFP_ATOMIC); | ||
| 435 | } | ||
| 436 | |||
| 437 | static int pcan_usb_fd_drv_loaded(struct peak_usb_device *dev, bool loaded) | ||
| 438 | { | ||
| 439 | struct pcan_usb_fd_device *pdev = | ||
| 440 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 441 | |||
| 442 | pdev->cmd_buffer_addr[0] = 0; | ||
| 443 | pdev->cmd_buffer_addr[1] = !!loaded; | ||
| 444 | |||
| 445 | return pcan_usb_pro_send_req(dev, | ||
| 446 | PCAN_USBPRO_REQ_FCT, | ||
| 447 | PCAN_USBPRO_FCT_DRVLD, | ||
| 448 | pdev->cmd_buffer_addr, | ||
| 449 | PCAN_USBPRO_FCT_DRVLD_REQ_LEN); | ||
| 450 | } | ||
| 451 | |||
| 452 | static int pcan_usb_fd_decode_canmsg(struct pcan_usb_fd_if *usb_if, | ||
| 453 | struct pucan_msg *rx_msg) | ||
| 454 | { | ||
| 455 | struct pucan_rx_msg *rm = (struct pucan_rx_msg *)rx_msg; | ||
| 456 | struct peak_usb_device *dev = usb_if->dev[pucan_msg_get_channel(rm)]; | ||
| 457 | struct net_device *netdev = dev->netdev; | ||
| 458 | struct canfd_frame *cfd; | ||
| 459 | struct sk_buff *skb; | ||
| 460 | const u16 rx_msg_flags = le16_to_cpu(rm->flags); | ||
| 461 | |||
| 462 | if (rx_msg_flags & PUCAN_MSG_EXT_DATA_LEN) { | ||
| 463 | /* CANFD frame case */ | ||
| 464 | skb = alloc_canfd_skb(netdev, &cfd); | ||
| 465 | if (!skb) | ||
| 466 | return -ENOMEM; | ||
| 467 | |||
| 468 | if (rx_msg_flags & PUCAN_MSG_BITRATE_SWITCH) | ||
| 469 | cfd->flags |= CANFD_BRS; | ||
| 470 | |||
| 471 | if (rx_msg_flags & PUCAN_MSG_ERROR_STATE_IND) | ||
| 472 | cfd->flags |= CANFD_ESI; | ||
| 473 | |||
| 474 | cfd->len = can_dlc2len(get_canfd_dlc(pucan_msg_get_dlc(rm))); | ||
| 475 | } else { | ||
| 476 | /* CAN 2.0 frame case */ | ||
| 477 | skb = alloc_can_skb(netdev, (struct can_frame **)&cfd); | ||
| 478 | if (!skb) | ||
| 479 | return -ENOMEM; | ||
| 480 | |||
| 481 | cfd->len = get_can_dlc(pucan_msg_get_dlc(rm)); | ||
| 482 | } | ||
| 483 | |||
| 484 | cfd->can_id = le32_to_cpu(rm->can_id); | ||
| 485 | |||
| 486 | if (rx_msg_flags & PUCAN_MSG_EXT_ID) | ||
| 487 | cfd->can_id |= CAN_EFF_FLAG; | ||
| 488 | |||
| 489 | if (rx_msg_flags & PUCAN_MSG_RTR) | ||
| 490 | cfd->can_id |= CAN_RTR_FLAG; | ||
| 491 | else | ||
| 492 | memcpy(cfd->data, rm->d, cfd->len); | ||
| 493 | |||
| 494 | peak_usb_netif_rx(skb, &usb_if->time_ref, | ||
| 495 | le32_to_cpu(rm->ts_low), le32_to_cpu(rm->ts_high)); | ||
| 496 | |||
| 497 | netdev->stats.rx_packets++; | ||
| 498 | netdev->stats.rx_bytes += cfd->len; | ||
| 499 | |||
| 500 | return 0; | ||
| 501 | } | ||
| 502 | |||
| 503 | /* handle uCAN status message */ | ||
| 504 | static int pcan_usb_fd_decode_status(struct pcan_usb_fd_if *usb_if, | ||
| 505 | struct pucan_msg *rx_msg) | ||
| 506 | { | ||
| 507 | struct pucan_status_msg *sm = (struct pucan_status_msg *)rx_msg; | ||
| 508 | struct peak_usb_device *dev = usb_if->dev[pucan_stmsg_get_channel(sm)]; | ||
| 509 | struct pcan_usb_fd_device *pdev = | ||
| 510 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 511 | enum can_state new_state = CAN_STATE_ERROR_ACTIVE; | ||
| 512 | enum can_state rx_state, tx_state; | ||
| 513 | struct net_device *netdev = dev->netdev; | ||
| 514 | struct can_frame *cf; | ||
| 515 | struct sk_buff *skb; | ||
| 516 | |||
| 517 | /* nothing should be sent while in BUS_OFF state */ | ||
| 518 | if (dev->can.state == CAN_STATE_BUS_OFF) | ||
| 519 | return 0; | ||
| 520 | |||
| 521 | if (sm->channel_p_w_b & PUCAN_BUS_BUSOFF) { | ||
| 522 | new_state = CAN_STATE_BUS_OFF; | ||
| 523 | } else if (sm->channel_p_w_b & PUCAN_BUS_PASSIVE) { | ||
| 524 | new_state = CAN_STATE_ERROR_PASSIVE; | ||
| 525 | } else if (sm->channel_p_w_b & PUCAN_BUS_WARNING) { | ||
| 526 | new_state = CAN_STATE_ERROR_WARNING; | ||
| 527 | } else { | ||
| 528 | /* no error bit (so, no error skb, back to active state) */ | ||
| 529 | dev->can.state = CAN_STATE_ERROR_ACTIVE; | ||
| 530 | pdev->bec.txerr = 0; | ||
| 531 | pdev->bec.rxerr = 0; | ||
| 532 | return 0; | ||
| 533 | } | ||
| 534 | |||
| 535 | /* state hasn't changed */ | ||
| 536 | if (new_state == dev->can.state) | ||
| 537 | return 0; | ||
| 538 | |||
| 539 | /* handle bus state change */ | ||
| 540 | tx_state = (pdev->bec.txerr >= pdev->bec.rxerr) ? new_state : 0; | ||
| 541 | rx_state = (pdev->bec.txerr <= pdev->bec.rxerr) ? new_state : 0; | ||
| 542 | |||
| 543 | /* allocate an skb to store the error frame */ | ||
| 544 | skb = alloc_can_err_skb(netdev, &cf); | ||
| 545 | if (skb) | ||
| 546 | can_change_state(netdev, cf, tx_state, rx_state); | ||
| 547 | |||
| 548 | /* things must be done even in case of OOM */ | ||
| 549 | if (new_state == CAN_STATE_BUS_OFF) | ||
| 550 | can_bus_off(netdev); | ||
| 551 | |||
| 552 | if (!skb) | ||
| 553 | return -ENOMEM; | ||
| 554 | |||
| 555 | peak_usb_netif_rx(skb, &usb_if->time_ref, | ||
| 556 | le32_to_cpu(sm->ts_low), le32_to_cpu(sm->ts_high)); | ||
| 557 | |||
| 558 | netdev->stats.rx_packets++; | ||
| 559 | netdev->stats.rx_bytes += cf->can_dlc; | ||
| 560 | |||
| 561 | return 0; | ||
| 562 | } | ||
| 563 | |||
| 564 | /* handle uCAN error message */ | ||
| 565 | static int pcan_usb_fd_decode_error(struct pcan_usb_fd_if *usb_if, | ||
| 566 | struct pucan_msg *rx_msg) | ||
| 567 | { | ||
| 568 | struct pucan_error_msg *er = (struct pucan_error_msg *)rx_msg; | ||
| 569 | struct peak_usb_device *dev = usb_if->dev[pucan_ermsg_get_channel(er)]; | ||
| 570 | struct pcan_usb_fd_device *pdev = | ||
| 571 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 572 | |||
| 573 | /* keep a trace of tx and rx error counters for later use */ | ||
| 574 | pdev->bec.txerr = er->tx_err_cnt; | ||
| 575 | pdev->bec.rxerr = er->rx_err_cnt; | ||
| 576 | |||
| 577 | return 0; | ||
| 578 | } | ||
| 579 | |||
| 580 | /* handle uCAN overrun message */ | ||
| 581 | static int pcan_usb_fd_decode_overrun(struct pcan_usb_fd_if *usb_if, | ||
| 582 | struct pucan_msg *rx_msg) | ||
| 583 | { | ||
| 584 | struct pcan_ufd_ovr_msg *ov = (struct pcan_ufd_ovr_msg *)rx_msg; | ||
| 585 | struct peak_usb_device *dev = usb_if->dev[pufd_omsg_get_channel(ov)]; | ||
| 586 | struct net_device *netdev = dev->netdev; | ||
| 587 | struct can_frame *cf; | ||
| 588 | struct sk_buff *skb; | ||
| 589 | |||
| 590 | /* allocate an skb to store the error frame */ | ||
| 591 | skb = alloc_can_err_skb(netdev, &cf); | ||
| 592 | if (!skb) | ||
| 593 | return -ENOMEM; | ||
| 594 | |||
| 595 | cf->can_id |= CAN_ERR_CRTL; | ||
| 596 | cf->data[1] |= CAN_ERR_CRTL_RX_OVERFLOW; | ||
| 597 | |||
| 598 | peak_usb_netif_rx(skb, &usb_if->time_ref, | ||
| 599 | le32_to_cpu(ov->ts_low), le32_to_cpu(ov->ts_high)); | ||
| 600 | |||
| 601 | netdev->stats.rx_over_errors++; | ||
| 602 | netdev->stats.rx_errors++; | ||
| 603 | |||
| 604 | return 0; | ||
| 605 | } | ||
| 606 | |||
| 607 | /* handle USB calibration message */ | ||
| 608 | static void pcan_usb_fd_decode_ts(struct pcan_usb_fd_if *usb_if, | ||
| 609 | struct pucan_msg *rx_msg) | ||
| 610 | { | ||
| 611 | struct pcan_ufd_ts_msg *ts = (struct pcan_ufd_ts_msg *)rx_msg; | ||
| 612 | |||
| 613 | /* should wait until clock is stabilized */ | ||
| 614 | if (usb_if->cm_ignore_count > 0) | ||
| 615 | usb_if->cm_ignore_count--; | ||
| 616 | else | ||
| 617 | peak_usb_set_ts_now(&usb_if->time_ref, le32_to_cpu(ts->ts_low)); | ||
| 618 | } | ||
| 619 | |||
| 620 | /* callback for bulk IN urb */ | ||
| 621 | static int pcan_usb_fd_decode_buf(struct peak_usb_device *dev, struct urb *urb) | ||
| 622 | { | ||
| 623 | struct pcan_usb_fd_if *usb_if = pcan_usb_fd_dev_if(dev); | ||
| 624 | struct net_device *netdev = dev->netdev; | ||
| 625 | struct pucan_msg *rx_msg; | ||
| 626 | u8 *msg_ptr, *msg_end; | ||
| 627 | int err = 0; | ||
| 628 | |||
| 629 | /* loop reading all the records from the incoming message */ | ||
| 630 | msg_ptr = urb->transfer_buffer; | ||
| 631 | msg_end = urb->transfer_buffer + urb->actual_length; | ||
| 632 | for (; msg_ptr < msg_end;) { | ||
| 633 | u16 rx_msg_type, rx_msg_size; | ||
| 634 | |||
| 635 | rx_msg = (struct pucan_msg *)msg_ptr; | ||
| 636 | if (!rx_msg->size) { | ||
| 637 | /* null packet found: end of list */ | ||
| 638 | break; | ||
| 639 | } | ||
| 640 | |||
| 641 | rx_msg_size = le16_to_cpu(rx_msg->size); | ||
| 642 | rx_msg_type = le16_to_cpu(rx_msg->type); | ||
| 643 | |||
| 644 | /* check if the record goes out of current packet */ | ||
| 645 | if (msg_ptr + rx_msg_size > msg_end) { | ||
| 646 | netdev_err(netdev, | ||
| 647 | "got frag rec: should inc usb rx buf sze\n"); | ||
| 648 | err = -EBADMSG; | ||
| 649 | break; | ||
| 650 | } | ||
| 651 | |||
| 652 | switch (rx_msg_type) { | ||
| 653 | case PUCAN_MSG_CAN_RX: | ||
| 654 | err = pcan_usb_fd_decode_canmsg(usb_if, rx_msg); | ||
| 655 | if (err < 0) | ||
| 656 | goto fail; | ||
| 657 | break; | ||
| 658 | |||
| 659 | case PCAN_UFD_MSG_CALIBRATION: | ||
| 660 | pcan_usb_fd_decode_ts(usb_if, rx_msg); | ||
| 661 | break; | ||
| 662 | |||
| 663 | case PUCAN_MSG_ERROR: | ||
| 664 | err = pcan_usb_fd_decode_error(usb_if, rx_msg); | ||
| 665 | if (err < 0) | ||
| 666 | goto fail; | ||
| 667 | break; | ||
| 668 | |||
| 669 | case PUCAN_MSG_STATUS: | ||
| 670 | err = pcan_usb_fd_decode_status(usb_if, rx_msg); | ||
| 671 | if (err < 0) | ||
| 672 | goto fail; | ||
| 673 | break; | ||
| 674 | |||
| 675 | case PCAN_UFD_MSG_OVERRUN: | ||
| 676 | err = pcan_usb_fd_decode_overrun(usb_if, rx_msg); | ||
| 677 | if (err < 0) | ||
| 678 | goto fail; | ||
| 679 | break; | ||
| 680 | |||
| 681 | default: | ||
| 682 | netdev_err(netdev, | ||
| 683 | "unhandled msg type 0x%02x (%d): ignored\n", | ||
| 684 | rx_msg_type, rx_msg_type); | ||
| 685 | break; | ||
| 686 | } | ||
| 687 | |||
| 688 | msg_ptr += rx_msg_size; | ||
| 689 | } | ||
| 690 | |||
| 691 | fail: | ||
| 692 | if (err) | ||
| 693 | pcan_dump_mem("received msg", | ||
| 694 | urb->transfer_buffer, urb->actual_length); | ||
| 695 | return err; | ||
| 696 | } | ||
| 697 | |||
| 698 | /* CAN/CANFD frames encoding callback */ | ||
| 699 | static int pcan_usb_fd_encode_msg(struct peak_usb_device *dev, | ||
| 700 | struct sk_buff *skb, u8 *obuf, size_t *size) | ||
| 701 | { | ||
| 702 | struct pucan_tx_msg *tx_msg = (struct pucan_tx_msg *)obuf; | ||
| 703 | struct canfd_frame *cfd = (struct canfd_frame *)skb->data; | ||
| 704 | u16 tx_msg_size, tx_msg_flags; | ||
| 705 | u8 can_dlc; | ||
| 706 | |||
| 707 | tx_msg_size = ALIGN(sizeof(struct pucan_tx_msg) + cfd->len, 4); | ||
| 708 | tx_msg->size = cpu_to_le16(tx_msg_size); | ||
| 709 | tx_msg->type = cpu_to_le16(PUCAN_MSG_CAN_TX); | ||
| 710 | |||
| 711 | tx_msg_flags = 0; | ||
| 712 | if (cfd->can_id & CAN_EFF_FLAG) { | ||
| 713 | tx_msg_flags |= PUCAN_MSG_EXT_ID; | ||
| 714 | tx_msg->can_id = cpu_to_le32(cfd->can_id & CAN_EFF_MASK); | ||
| 715 | } else { | ||
| 716 | tx_msg->can_id = cpu_to_le32(cfd->can_id & CAN_SFF_MASK); | ||
| 717 | } | ||
| 718 | |||
| 719 | if (can_is_canfd_skb(skb)) { | ||
| 720 | /* considering a CANFD frame */ | ||
| 721 | can_dlc = can_len2dlc(cfd->len); | ||
| 722 | |||
| 723 | tx_msg_flags |= PUCAN_MSG_EXT_DATA_LEN; | ||
| 724 | |||
| 725 | if (cfd->flags & CANFD_BRS) | ||
| 726 | tx_msg_flags |= PUCAN_MSG_BITRATE_SWITCH; | ||
| 727 | |||
| 728 | if (cfd->flags & CANFD_ESI) | ||
| 729 | tx_msg_flags |= PUCAN_MSG_ERROR_STATE_IND; | ||
| 730 | } else { | ||
| 731 | /* CAND 2.0 frames */ | ||
| 732 | can_dlc = cfd->len; | ||
| 733 | |||
| 734 | if (cfd->can_id & CAN_RTR_FLAG) | ||
| 735 | tx_msg_flags |= PUCAN_MSG_RTR; | ||
| 736 | } | ||
| 737 | |||
| 738 | tx_msg->flags = cpu_to_le16(tx_msg_flags); | ||
| 739 | tx_msg->channel_dlc = PUCAN_MSG_CHANNEL_DLC(dev->ctrl_idx, can_dlc); | ||
| 740 | memcpy(tx_msg->d, cfd->data, cfd->len); | ||
| 741 | |||
| 742 | /* add null size message to tag the end (messages are 32-bits aligned) | ||
| 743 | */ | ||
| 744 | tx_msg = (struct pucan_tx_msg *)(obuf + tx_msg_size); | ||
| 745 | |||
| 746 | tx_msg->size = 0; | ||
| 747 | |||
| 748 | /* set the whole size of the USB packet to send */ | ||
| 749 | *size = tx_msg_size + sizeof(u32); | ||
| 750 | |||
| 751 | return 0; | ||
| 752 | } | ||
| 753 | |||
| 754 | /* start the interface (last chance before set bus on) */ | ||
| 755 | static int pcan_usb_fd_start(struct peak_usb_device *dev) | ||
| 756 | { | ||
| 757 | struct pcan_usb_fd_device *pdev = | ||
| 758 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 759 | int err; | ||
| 760 | |||
| 761 | /* set filter mode: all acceptance */ | ||
| 762 | err = pcan_usb_fd_set_filter_std(dev, -1, 0xffffffff); | ||
| 763 | if (err) | ||
| 764 | return err; | ||
| 765 | |||
| 766 | /* opening first device: */ | ||
| 767 | if (pdev->usb_if->dev_opened_count == 0) { | ||
| 768 | /* reset time_ref */ | ||
| 769 | peak_usb_init_time_ref(&pdev->usb_if->time_ref, | ||
| 770 | &pcan_usb_pro_fd); | ||
| 771 | |||
| 772 | /* enable USB calibration messages */ | ||
| 773 | err = pcan_usb_fd_set_filter_ext(dev, 1, | ||
| 774 | PUCAN_FLTEXT_ERROR, | ||
| 775 | PCAN_UFD_FLTEXT_CALIBRATION); | ||
| 776 | } | ||
| 777 | |||
| 778 | pdev->usb_if->dev_opened_count++; | ||
| 779 | |||
| 780 | /* reset cached error counters */ | ||
| 781 | pdev->bec.txerr = 0; | ||
| 782 | pdev->bec.rxerr = 0; | ||
| 783 | |||
| 784 | return err; | ||
| 785 | } | ||
| 786 | |||
| 787 | /* socket callback used to copy berr counters values receieved through USB */ | ||
| 788 | static int pcan_usb_fd_get_berr_counter(const struct net_device *netdev, | ||
| 789 | struct can_berr_counter *bec) | ||
| 790 | { | ||
| 791 | struct peak_usb_device *dev = netdev_priv(netdev); | ||
| 792 | struct pcan_usb_fd_device *pdev = | ||
| 793 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 794 | |||
| 795 | *bec = pdev->bec; | ||
| 796 | |||
| 797 | /* must return 0 */ | ||
| 798 | return 0; | ||
| 799 | } | ||
| 800 | |||
| 801 | /* stop interface (last chance before set bus off) */ | ||
| 802 | static int pcan_usb_fd_stop(struct peak_usb_device *dev) | ||
| 803 | { | ||
| 804 | struct pcan_usb_fd_device *pdev = | ||
| 805 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 806 | |||
| 807 | /* turn off special msgs for that interface if no other dev opened */ | ||
| 808 | if (pdev->usb_if->dev_opened_count == 1) | ||
| 809 | pcan_usb_fd_set_filter_ext(dev, 0, | ||
| 810 | PUCAN_FLTEXT_ERROR, | ||
| 811 | PCAN_UFD_FLTEXT_CALIBRATION); | ||
| 812 | pdev->usb_if->dev_opened_count--; | ||
| 813 | |||
| 814 | return 0; | ||
| 815 | } | ||
| 816 | |||
| 817 | /* called when probing, to initialize a device object */ | ||
| 818 | static int pcan_usb_fd_init(struct peak_usb_device *dev) | ||
| 819 | { | ||
| 820 | struct pcan_usb_fd_device *pdev = | ||
| 821 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 822 | int i, err = -ENOMEM; | ||
| 823 | |||
| 824 | /* do this for 1st channel only */ | ||
| 825 | if (!dev->prev_siblings) { | ||
| 826 | /* allocate netdevices common structure attached to first one */ | ||
| 827 | pdev->usb_if = kzalloc(sizeof(*pdev->usb_if), GFP_KERNEL); | ||
| 828 | if (!pdev->usb_if) | ||
| 829 | goto err_out; | ||
| 830 | |||
| 831 | /* allocate command buffer once for all for the interface */ | ||
| 832 | pdev->cmd_buffer_addr = kmalloc(PCAN_UFD_CMD_BUFFER_SIZE, | ||
| 833 | GFP_KERNEL); | ||
| 834 | if (!pdev->cmd_buffer_addr) | ||
| 835 | goto err_out_1; | ||
| 836 | |||
| 837 | /* number of ts msgs to ignore before taking one into account */ | ||
| 838 | pdev->usb_if->cm_ignore_count = 5; | ||
| 839 | |||
| 840 | err = pcan_usb_pro_send_req(dev, PCAN_USBPRO_REQ_INFO, | ||
| 841 | PCAN_USBPRO_INFO_FW, | ||
| 842 | &pdev->usb_if->fw_info, | ||
| 843 | sizeof(pdev->usb_if->fw_info)); | ||
| 844 | if (err) { | ||
| 845 | dev_err(dev->netdev->dev.parent, | ||
| 846 | "unable to read %s firmware info (err %d)\n", | ||
| 847 | dev->adapter->name, err); | ||
| 848 | goto err_out_2; | ||
| 849 | } | ||
| 850 | |||
| 851 | /* explicit use of dev_xxx() instead of netdev_xxx() here: | ||
| 852 | * information displayed are related to the device itself, not | ||
| 853 | * to the canx (channel) device. | ||
| 854 | */ | ||
| 855 | dev_info(dev->netdev->dev.parent, | ||
| 856 | "PEAK-System %s v%u fw v%u.%u.%u (%u channels)\n", | ||
| 857 | dev->adapter->name, pdev->usb_if->fw_info.hw_version, | ||
| 858 | pdev->usb_if->fw_info.fw_version[0], | ||
| 859 | pdev->usb_if->fw_info.fw_version[1], | ||
| 860 | pdev->usb_if->fw_info.fw_version[2], | ||
| 861 | dev->adapter->ctrl_count); | ||
| 862 | |||
| 863 | /* the currently supported hw is non-ISO */ | ||
| 864 | dev->can.ctrlmode = CAN_CTRLMODE_FD_NON_ISO; | ||
| 865 | |||
| 866 | /* tell the hardware the can driver is running */ | ||
| 867 | err = pcan_usb_fd_drv_loaded(dev, 1); | ||
| 868 | if (err) { | ||
| 869 | dev_err(dev->netdev->dev.parent, | ||
| 870 | "unable to tell %s driver is loaded (err %d)\n", | ||
| 871 | dev->adapter->name, err); | ||
| 872 | goto err_out_2; | ||
| 873 | } | ||
| 874 | } else { | ||
| 875 | /* otherwise, simply copy previous sibling's values */ | ||
| 876 | struct pcan_usb_fd_device *ppdev = | ||
| 877 | container_of(dev->prev_siblings, | ||
| 878 | struct pcan_usb_fd_device, dev); | ||
| 879 | |||
| 880 | pdev->usb_if = ppdev->usb_if; | ||
| 881 | pdev->cmd_buffer_addr = ppdev->cmd_buffer_addr; | ||
| 882 | } | ||
| 883 | |||
| 884 | pdev->usb_if->dev[dev->ctrl_idx] = dev; | ||
| 885 | dev->device_number = | ||
| 886 | le32_to_cpu(pdev->usb_if->fw_info.dev_id[dev->ctrl_idx]); | ||
| 887 | |||
| 888 | /* set clock domain */ | ||
| 889 | for (i = 0; i < ARRAY_SIZE(pcan_usb_fd_clk_freq); i++) | ||
| 890 | if (dev->adapter->clock.freq == pcan_usb_fd_clk_freq[i]) | ||
| 891 | break; | ||
| 892 | |||
| 893 | if (i >= ARRAY_SIZE(pcan_usb_fd_clk_freq)) { | ||
| 894 | dev_warn(dev->netdev->dev.parent, | ||
| 895 | "incompatible clock frequencies\n"); | ||
| 896 | err = -EINVAL; | ||
| 897 | goto err_out_2; | ||
| 898 | } | ||
| 899 | |||
| 900 | pcan_usb_fd_set_clock_domain(dev, i); | ||
| 901 | |||
| 902 | /* set LED in default state (end of init phase) */ | ||
| 903 | pcan_usb_fd_set_can_led(dev, PCAN_UFD_LED_DEF); | ||
| 904 | |||
| 905 | return 0; | ||
| 906 | |||
| 907 | err_out_2: | ||
| 908 | kfree(pdev->cmd_buffer_addr); | ||
| 909 | err_out_1: | ||
| 910 | kfree(pdev->usb_if); | ||
| 911 | err_out: | ||
| 912 | return err; | ||
| 913 | } | ||
| 914 | |||
| 915 | /* called when driver module is being unloaded */ | ||
| 916 | static void pcan_usb_fd_exit(struct peak_usb_device *dev) | ||
| 917 | { | ||
| 918 | struct pcan_usb_fd_device *pdev = | ||
| 919 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 920 | |||
| 921 | /* when rmmod called before unplug and if down, should reset things | ||
| 922 | * before leaving | ||
| 923 | */ | ||
| 924 | if (dev->can.state != CAN_STATE_STOPPED) { | ||
| 925 | /* set bus off on the corresponding channel */ | ||
| 926 | pcan_usb_fd_set_bus(dev, 0); | ||
| 927 | } | ||
| 928 | |||
| 929 | /* switch off corresponding CAN LEDs */ | ||
| 930 | pcan_usb_fd_set_can_led(dev, PCAN_UFD_LED_OFF); | ||
| 931 | |||
| 932 | /* if channel #0 (only) */ | ||
| 933 | if (dev->ctrl_idx == 0) { | ||
| 934 | /* turn off calibration message if any device were opened */ | ||
| 935 | if (pdev->usb_if->dev_opened_count > 0) | ||
| 936 | pcan_usb_fd_set_filter_ext(dev, 0, | ||
| 937 | PUCAN_FLTEXT_ERROR, | ||
| 938 | PCAN_UFD_FLTEXT_CALIBRATION); | ||
| 939 | |||
| 940 | /* tell USB adapter that the driver is being unloaded */ | ||
| 941 | pcan_usb_fd_drv_loaded(dev, 0); | ||
| 942 | } | ||
| 943 | } | ||
| 944 | |||
| 945 | /* called when the USB adapter is unplugged */ | ||
| 946 | static void pcan_usb_fd_free(struct peak_usb_device *dev) | ||
| 947 | { | ||
| 948 | /* last device: can free shared objects now */ | ||
| 949 | if (!dev->prev_siblings && !dev->next_siblings) { | ||
| 950 | struct pcan_usb_fd_device *pdev = | ||
| 951 | container_of(dev, struct pcan_usb_fd_device, dev); | ||
| 952 | |||
| 953 | /* free commands buffer */ | ||
| 954 | kfree(pdev->cmd_buffer_addr); | ||
| 955 | |||
| 956 | /* free usb interface object */ | ||
| 957 | kfree(pdev->usb_if); | ||
| 958 | } | ||
| 959 | } | ||
| 960 | |||
| 961 | /* describes the PCAN-USB FD adapter */ | ||
| 962 | const struct peak_usb_adapter pcan_usb_fd = { | ||
| 963 | .name = "PCAN-USB FD", | ||
| 964 | .device_id = PCAN_USBFD_PRODUCT_ID, | ||
| 965 | .ctrl_count = PCAN_USBFD_CHANNEL_COUNT, | ||
| 966 | .ctrlmode_supported = CAN_CTRLMODE_FD | | ||
| 967 | CAN_CTRLMODE_3_SAMPLES | CAN_CTRLMODE_LISTENONLY, | ||
| 968 | .clock = { | ||
| 969 | .freq = PCAN_UFD_CRYSTAL_HZ, | ||
| 970 | }, | ||
| 971 | .bittiming_const = { | ||
| 972 | .name = "pcan_usb_fd", | ||
| 973 | .tseg1_min = 1, | ||
| 974 | .tseg1_max = 64, | ||
| 975 | .tseg2_min = 1, | ||
| 976 | .tseg2_max = 16, | ||
| 977 | .sjw_max = 16, | ||
| 978 | .brp_min = 1, | ||
| 979 | .brp_max = 1024, | ||
| 980 | .brp_inc = 1, | ||
| 981 | }, | ||
| 982 | .data_bittiming_const = { | ||
| 983 | .name = "pcan_usb_fd", | ||
| 984 | .tseg1_min = 1, | ||
| 985 | .tseg1_max = 16, | ||
| 986 | .tseg2_min = 1, | ||
| 987 | .tseg2_max = 8, | ||
| 988 | .sjw_max = 4, | ||
| 989 | .brp_min = 1, | ||
| 990 | .brp_max = 1024, | ||
| 991 | .brp_inc = 1, | ||
| 992 | }, | ||
| 993 | |||
| 994 | /* size of device private data */ | ||
| 995 | .sizeof_dev_private = sizeof(struct pcan_usb_fd_device), | ||
| 996 | |||
| 997 | /* timestamps usage */ | ||
| 998 | .ts_used_bits = 32, | ||
| 999 | .ts_period = 1000000, /* calibration period in ts. */ | ||
| 1000 | .us_per_ts_scale = 1, /* us = (ts * scale) >> shift */ | ||
| 1001 | .us_per_ts_shift = 0, | ||
| 1002 | |||
| 1003 | /* give here messages in/out endpoints */ | ||
| 1004 | .ep_msg_in = PCAN_USBPRO_EP_MSGIN, | ||
| 1005 | .ep_msg_out = {PCAN_USBPRO_EP_MSGOUT_0}, | ||
| 1006 | |||
| 1007 | /* size of rx/tx usb buffers */ | ||
| 1008 | .rx_buffer_size = PCAN_UFD_RX_BUFFER_SIZE, | ||
| 1009 | .tx_buffer_size = PCAN_UFD_TX_BUFFER_SIZE, | ||
| 1010 | |||
| 1011 | /* device callbacks */ | ||
| 1012 | .intf_probe = pcan_usb_pro_probe, /* same as PCAN-USB Pro */ | ||
| 1013 | .dev_init = pcan_usb_fd_init, | ||
| 1014 | |||
| 1015 | .dev_exit = pcan_usb_fd_exit, | ||
| 1016 | .dev_free = pcan_usb_fd_free, | ||
| 1017 | .dev_set_bus = pcan_usb_fd_set_bus, | ||
| 1018 | .dev_set_bittiming = pcan_usb_fd_set_bittiming_slow, | ||
| 1019 | .dev_set_data_bittiming = pcan_usb_fd_set_bittiming_fast, | ||
| 1020 | .dev_decode_buf = pcan_usb_fd_decode_buf, | ||
| 1021 | .dev_start = pcan_usb_fd_start, | ||
| 1022 | .dev_stop = pcan_usb_fd_stop, | ||
| 1023 | .dev_restart_async = pcan_usb_fd_restart_async, | ||
| 1024 | .dev_encode_msg = pcan_usb_fd_encode_msg, | ||
| 1025 | |||
| 1026 | .do_get_berr_counter = pcan_usb_fd_get_berr_counter, | ||
| 1027 | }; | ||
| 1028 | |||
| 1029 | /* describes the PCAN-USB Pro FD adapter */ | ||
| 1030 | const struct peak_usb_adapter pcan_usb_pro_fd = { | ||
| 1031 | .name = "PCAN-USB Pro FD", | ||
| 1032 | .device_id = PCAN_USBPROFD_PRODUCT_ID, | ||
| 1033 | .ctrl_count = PCAN_USBPROFD_CHANNEL_COUNT, | ||
| 1034 | .ctrlmode_supported = CAN_CTRLMODE_FD | | ||
| 1035 | CAN_CTRLMODE_3_SAMPLES | CAN_CTRLMODE_LISTENONLY, | ||
| 1036 | .clock = { | ||
| 1037 | .freq = PCAN_UFD_CRYSTAL_HZ, | ||
| 1038 | }, | ||
| 1039 | .bittiming_const = { | ||
| 1040 | .name = "pcan_usb_pro_fd", | ||
| 1041 | .tseg1_min = 1, | ||
| 1042 | .tseg1_max = 64, | ||
| 1043 | .tseg2_min = 1, | ||
| 1044 | .tseg2_max = 16, | ||
| 1045 | .sjw_max = 16, | ||
| 1046 | .brp_min = 1, | ||
| 1047 | .brp_max = 1024, | ||
| 1048 | .brp_inc = 1, | ||
| 1049 | }, | ||
| 1050 | .data_bittiming_const = { | ||
| 1051 | .name = "pcan_usb_pro_fd", | ||
| 1052 | .tseg1_min = 1, | ||
| 1053 | .tseg1_max = 16, | ||
| 1054 | .tseg2_min = 1, | ||
| 1055 | .tseg2_max = 8, | ||
| 1056 | .sjw_max = 4, | ||
| 1057 | .brp_min = 1, | ||
| 1058 | .brp_max = 1024, | ||
| 1059 | .brp_inc = 1, | ||
| 1060 | }, | ||
| 1061 | |||
| 1062 | /* size of device private data */ | ||
| 1063 | .sizeof_dev_private = sizeof(struct pcan_usb_fd_device), | ||
| 1064 | |||
| 1065 | /* timestamps usage */ | ||
| 1066 | .ts_used_bits = 32, | ||
| 1067 | .ts_period = 1000000, /* calibration period in ts. */ | ||
| 1068 | .us_per_ts_scale = 1, /* us = (ts * scale) >> shift */ | ||
| 1069 | .us_per_ts_shift = 0, | ||
| 1070 | |||
| 1071 | /* give here messages in/out endpoints */ | ||
| 1072 | .ep_msg_in = PCAN_USBPRO_EP_MSGIN, | ||
| 1073 | .ep_msg_out = {PCAN_USBPRO_EP_MSGOUT_0, PCAN_USBPRO_EP_MSGOUT_1}, | ||
| 1074 | |||
| 1075 | /* size of rx/tx usb buffers */ | ||
| 1076 | .rx_buffer_size = PCAN_UFD_RX_BUFFER_SIZE, | ||
| 1077 | .tx_buffer_size = PCAN_UFD_TX_BUFFER_SIZE, | ||
| 1078 | |||
| 1079 | /* device callbacks */ | ||
| 1080 | .intf_probe = pcan_usb_pro_probe, /* same as PCAN-USB Pro */ | ||
| 1081 | .dev_init = pcan_usb_fd_init, | ||
| 1082 | |||
| 1083 | .dev_exit = pcan_usb_fd_exit, | ||
| 1084 | .dev_free = pcan_usb_fd_free, | ||
| 1085 | .dev_set_bus = pcan_usb_fd_set_bus, | ||
| 1086 | .dev_set_bittiming = pcan_usb_fd_set_bittiming_slow, | ||
| 1087 | .dev_set_data_bittiming = pcan_usb_fd_set_bittiming_fast, | ||
| 1088 | .dev_decode_buf = pcan_usb_fd_decode_buf, | ||
| 1089 | .dev_start = pcan_usb_fd_start, | ||
| 1090 | .dev_stop = pcan_usb_fd_stop, | ||
| 1091 | .dev_restart_async = pcan_usb_fd_restart_async, | ||
| 1092 | .dev_encode_msg = pcan_usb_fd_encode_msg, | ||
| 1093 | |||
| 1094 | .do_get_berr_counter = pcan_usb_fd_get_berr_counter, | ||
| 1095 | }; | ||
