aboutsummaryrefslogtreecommitdiffstats
path: root/epan/dissectors/packet-socketcan.c
diff options
context:
space:
mode:
Diffstat (limited to 'epan/dissectors/packet-socketcan.c')
-rw-r--r--epan/dissectors/packet-socketcan.c1877
1 files changed, 947 insertions, 930 deletions
diff --git a/epan/dissectors/packet-socketcan.c b/epan/dissectors/packet-socketcan.c
index cbb83cedbb..e441c44ade 100644
--- a/epan/dissectors/packet-socketcan.c
+++ b/epan/dissectors/packet-socketcan.c
@@ -28,62 +28,71 @@
void proto_register_socketcan(void);
void proto_reg_handoff_socketcan(void);
-static int hf_can_len = -1;
-static int hf_can_infoent_ext = -1;
-static int hf_can_infoent_std = -1;
-static int hf_can_extflag = -1;
-static int hf_can_rtrflag = -1;
-static int hf_can_errflag = -1;
-static int hf_can_reserved = -1;
-static int hf_can_padding = -1;
-
-static int hf_can_err_tx_timeout = -1;
-static int hf_can_err_lostarb = -1;
-static int hf_can_err_ctrl = -1;
-static int hf_can_err_prot = -1;
-static int hf_can_err_trx = -1;
-static int hf_can_err_ack = -1;
-static int hf_can_err_busoff = -1;
-static int hf_can_err_buserror = -1;
-static int hf_can_err_restarted = -1;
-static int hf_can_err_reserved = -1;
-
-static int hf_can_err_lostarb_bit_number = -1;
-
-static int hf_can_err_ctrl_rx_overflow = -1;
-static int hf_can_err_ctrl_tx_overflow = -1;
-static int hf_can_err_ctrl_rx_warning = -1;
-static int hf_can_err_ctrl_tx_warning = -1;
-static int hf_can_err_ctrl_rx_passive = -1;
-static int hf_can_err_ctrl_tx_passive = -1;
-static int hf_can_err_ctrl_active = -1;
-
-static int hf_can_err_prot_error_type_bit = -1;
-static int hf_can_err_prot_error_type_form = -1;
-static int hf_can_err_prot_error_type_stuff = -1;
-static int hf_can_err_prot_error_type_bit0 = -1;
-static int hf_can_err_prot_error_type_bit1 = -1;
-static int hf_can_err_prot_error_type_overload = -1;
-static int hf_can_err_prot_error_type_active = -1;
-static int hf_can_err_prot_error_type_tx = -1;
-
-static int hf_can_err_prot_error_location = -1;
-
-static int hf_can_err_trx_canh = -1;
-static int hf_can_err_trx_canl = -1;
-
-static int hf_can_err_ctrl_specific = -1;
-
-static expert_field ei_can_err_dlc_mismatch = EI_INIT;
-
-static int hf_canfd_brsflag = -1;
-static int hf_canfd_esiflag = -1;
-
-static gint ett_can = -1;
-static gint ett_can_fd = -1;
-
-static int proto_can = -1;
-static int proto_canfd = -1;
+static int hf_can_len;
+static int hf_can_infoent_ext;
+static int hf_can_infoent_std;
+static int hf_can_extflag;
+static int hf_can_rtrflag;
+static int hf_can_errflag;
+static int hf_can_reserved;
+static int hf_can_padding;
+
+static int hf_can_err_tx_timeout;
+static int hf_can_err_lostarb;
+static int hf_can_err_ctrl;
+static int hf_can_err_prot;
+static int hf_can_err_trx;
+static int hf_can_err_ack;
+static int hf_can_err_busoff;
+static int hf_can_err_buserror;
+static int hf_can_err_restarted;
+static int hf_can_err_reserved;
+
+static int hf_can_err_lostarb_bit_number;
+
+static int hf_can_err_ctrl_rx_overflow;
+static int hf_can_err_ctrl_tx_overflow;
+static int hf_can_err_ctrl_rx_warning;
+static int hf_can_err_ctrl_tx_warning;
+static int hf_can_err_ctrl_rx_passive;
+static int hf_can_err_ctrl_tx_passive;
+static int hf_can_err_ctrl_active;
+
+static int hf_can_err_prot_error_type_bit;
+static int hf_can_err_prot_error_type_form;
+static int hf_can_err_prot_error_type_stuff;
+static int hf_can_err_prot_error_type_bit0;
+static int hf_can_err_prot_error_type_bit1;
+static int hf_can_err_prot_error_type_overload;
+static int hf_can_err_prot_error_type_active;
+static int hf_can_err_prot_error_type_tx;
+
+static int hf_can_err_prot_error_location;
+
+static int hf_can_err_trx_canh;
+static int hf_can_err_trx_canl;
+
+static int hf_can_err_ctrl_specific;
+
+static int hf_canxl_priority;
+static int hf_canxl_vcid;
+static int hf_canxl_secflag;
+static int hf_canxl_sdu_type;
+static int hf_canxl_len;
+static int hf_canxl_acceptance_field;
+
+static expert_field ei_can_err_dlc_mismatch;
+
+static int hf_canfd_brsflag;
+static int hf_canfd_esiflag;
+
+static gint ett_can;
+static gint ett_can_fd;
+static gint ett_can_xl;
+
+static int proto_can;
+static int proto_canfd;
+static int proto_canxl;
static gboolean byte_swap = FALSE;
static gboolean heuristic_first = FALSE;
@@ -103,72 +112,89 @@ static heur_dtbl_entry_t *heur_dtbl_entry;
#define CANFD_BRS 0x01 /* bit rate switch (second bitrate for payload data) */
#define CANFD_ESI 0x02 /* error state indicator of the transmitting node */
+#define CANXL_LEN_OFFSET 6
+#define CANXL_DATA_OFFSET 12
+
static dissector_table_t can_id_dissector_table = NULL;
static dissector_table_t can_extended_id_dissector_table = NULL;
static dissector_table_t subdissector_table = NULL;
+static dissector_table_t canxl_sdu_type_dissector_table = NULL;
static dissector_handle_t socketcan_classic_handle;
static dissector_handle_t socketcan_fd_handle;
+static dissector_handle_t socketcan_xl_handle;
+static dissector_handle_t socketcan_bigendian_handle;
+
+
+static const value_string can_err_prot_error_location_vals[] = {
+ { 0x00, "unspecified" },
+ { 0x02, "ID bits 28 - 21 (SFF: 10 - 3)" },
+ { 0x03, "start of frame" },
+ { 0x04, "substitute RTR (SFF: RTR)" },
+ { 0x05, "identifier extension" },
+ { 0x06, "ID bits 20 - 18 (SFF: 2 - 0)" },
+ { 0x07, "ID bits 17-13" },
+ { 0x08, "CRC sequence" },
+ { 0x09, "reserved bit 0" },
+ { 0x0A, "data section" },
+ { 0x0B, "data length code" },
+ { 0x0C, "RTR" },
+ { 0x0D, "reserved bit 1" },
+ { 0x0E, "ID bits 4-0" },
+ { 0x0F, "ID bits 12-5" },
+ { 0x12, "intermission" },
+ { 0x18, "CRC delimiter" },
+ { 0x19, "ACK slot" },
+ { 0x1A, "end of frame" },
+ { 0x1B, "ACK delimiter" },
+ { 0, NULL }
+};
-static const value_string can_err_prot_error_location_vals[] =
-{
- { 0x00, "unspecified" },
- { 0x02, "ID bits 28 - 21 (SFF: 10 - 3)" },
- { 0x03, "start of frame" },
- { 0x04, "substitute RTR (SFF: RTR)" },
- { 0x05, "identifier extension" },
- { 0x06, "ID bits 20 - 18 (SFF: 2 - 0)" },
- { 0x07, "ID bits 17-13" },
- { 0x08, "CRC sequence" },
- { 0x09, "reserved bit 0" },
- { 0x0A, "data section" },
- { 0x0B, "data length code" },
- { 0x0C, "RTR" },
- { 0x0D, "reserved bit 1" },
- { 0x0E, "ID bits 4-0" },
- { 0x0F, "ID bits 12-5" },
- { 0x12, "intermission" },
- { 0x18, "CRC delimiter" },
- { 0x19, "ACK slot" },
- { 0x1A, "end of frame" },
- { 0x1B, "ACK delimiter" },
- { 0, NULL }
+static const value_string can_err_trx_canh_vals[] = {
+ { 0x00, "unspecified" },
+ { 0x04, "no wire" },
+ { 0x05, "short to BAT" },
+ { 0x06, "short to VCC" },
+ { 0x07, "short to GND" },
+ { 0, NULL }
};
-static const value_string can_err_trx_canh_vals[] =
-{
- { 0x00, "unspecified" },
- { 0x04, "no wire" },
- { 0x05, "short to BAT" },
- { 0x06, "short to VCC" },
- { 0x07, "short to GND" },
- { 0, NULL }
+static const value_string can_err_trx_canl_vals[] = {
+ { 0x00, "unspecified" },
+ { 0x04, "no wire" },
+ { 0x05, "short to BAT" },
+ { 0x06, "short to VCC" },
+ { 0x07, "short to GND" },
+ { 0x08, "short to CANH" },
+ { 0, NULL }
};
-static const value_string can_err_trx_canl_vals[] =
-{
- { 0x00, "unspecified" },
- { 0x04, "no wire" },
- { 0x05, "short to BAT" },
- { 0x06, "short to VCC" },
- { 0x07, "short to GND" },
- { 0x08, "short to CANH" },
- { 0, NULL }
+static const value_string canxl_sdu_type_vals[] = {
+ { 0x00, "Reserved" },
+ { CANXL_SDU_TYPE_CONTENT_BASED_ADDRESSING, "Content-based Addressing" },
+ { 0x02, "Reserved for future use" },
+ { CANXL_SDU_TYPE_CLASSICAL_CAN_AND_CAN_FD_MAPPED_TUNNELING, "Classical CAN/CAN FD mapped tunneling" },
+ { CANXL_SDU_TYPE_IEEE_802_3_MAC_FRAME_TUNNELLING, "IEEE 802.3 (MAC frame) tunneling" },
+ { CANXL_SDU_TYPE_IEEE_802_3_MAC_FRAME_MAPPED_TUNNELING, "IEEE 802.3 (MAC frame) mapped tunneling" },
+ { CANXL_SDU_TYPE_CLASSICAL_CAN_MAPPED_TUNNELING, "Classical CAN mapped tunneling" },
+ { CANXL_SDU_TYPE_CAN_FD_MAPPED_TUNNELING, "CAN FD mapped tunneling" },
+ { 0xFF, "Reserved" },
+ { 0, NULL }
};
/********* UATs *********/
/* Interface Config UAT */
typedef struct _interface_config {
- guint interface_id;
- gchar *interface_name;
- guint bus_id;
+ guint interface_id;
+ gchar *interface_name;
+ guint bus_id;
} interface_config_t;
#define DATAFILE_CAN_INTERFACE_MAPPING "CAN_interface_mapping"
static GHashTable *data_can_interfaces_by_id = NULL;
static GHashTable *data_can_interfaces_by_name = NULL;
-static interface_config_t* interface_configs = NULL;
+static interface_config_t *interface_configs = NULL;
static guint interface_config_num = 0;
UAT_HEX_CB_DEF(interface_configs, interface_id, interface_config_t)
@@ -177,116 +203,117 @@ UAT_HEX_CB_DEF(interface_configs, bus_id, interface_config_t)
static void *
copy_interface_config_cb(void *n, const void *o, size_t size _U_) {
- interface_config_t *new_rec = (interface_config_t *)n;
- const interface_config_t *old_rec = (const interface_config_t *)o;
+ interface_config_t *new_rec = (interface_config_t *)n;
+ const interface_config_t *old_rec = (const interface_config_t *)o;
- new_rec->interface_id = old_rec->interface_id;
- new_rec->interface_name = g_strdup(old_rec->interface_name);
- new_rec->bus_id = old_rec->bus_id;
- return new_rec;
+ new_rec->interface_id = old_rec->interface_id;
+ new_rec->interface_name = g_strdup(old_rec->interface_name);
+ new_rec->bus_id = old_rec->bus_id;
+ return new_rec;
}
-static gboolean
+static bool
update_interface_config(void *r, char **err) {
- interface_config_t *rec = (interface_config_t *)r;
+ interface_config_t *rec = (interface_config_t *)r;
- if (rec->interface_id > 0xffffffff) {
- *err = g_strdup_printf("We currently only support 32 bit identifiers (ID: %i Name: %s)",
- rec->interface_id, rec->interface_name);
- return FALSE;
- }
+ if (rec->interface_id > 0xffffffff) {
+ *err = ws_strdup_printf("We currently only support 32 bit identifiers (ID: %i Name: %s)",
+ rec->interface_id, rec->interface_name);
+ return FALSE;
+ }
- if (rec->bus_id > 0xffff) {
- *err = g_strdup_printf("We currently only support 16 bit bus identifiers (ID: %i Name: %s Bus-ID: %i)",
- rec->interface_id, rec->interface_name, rec->bus_id);
- return FALSE;
- }
+ if (rec->bus_id > 0xffff) {
+ *err = ws_strdup_printf("We currently only support 16 bit bus identifiers (ID: %i Name: %s Bus-ID: %i)",
+ rec->interface_id, rec->interface_name, rec->bus_id);
+ return FALSE;
+ }
- return TRUE;
+ return TRUE;
}
static void
free_interface_config_cb(void *r) {
- interface_config_t *rec = (interface_config_t *)r;
- /* freeing result of g_strdup */
- g_free(rec->interface_name);
- rec->interface_name = NULL;
+ interface_config_t *rec = (interface_config_t *)r;
+
+ /* freeing result of g_strdup */
+ g_free(rec->interface_name);
+ rec->interface_name = NULL;
}
static interface_config_t *
ht_lookup_interface_config_by_id(unsigned int identifier) {
- interface_config_t *tmp = NULL;
- unsigned int *id = NULL;
+ interface_config_t *tmp = NULL;
+ unsigned int *id = NULL;
- if (interface_configs == NULL) {
- return NULL;
- }
+ if (interface_configs == NULL) {
+ return NULL;
+ }
- id = wmem_new(wmem_epan_scope(), unsigned int);
- *id = (unsigned int)identifier;
- tmp = (interface_config_t *)g_hash_table_lookup(data_can_interfaces_by_id, id);
- wmem_free(wmem_epan_scope(), id);
+ id = wmem_new(wmem_epan_scope(), unsigned int);
+ *id = (unsigned int)identifier;
+ tmp = (interface_config_t *)g_hash_table_lookup(data_can_interfaces_by_id, id);
+ wmem_free(wmem_epan_scope(), id);
- return tmp;
+ return tmp;
}
static interface_config_t *
ht_lookup_interface_config_by_name(const gchar *name) {
- interface_config_t *tmp = NULL;
- gchar *key = NULL;
+ interface_config_t *tmp = NULL;
+ gchar *key = NULL;
- if (interface_configs == NULL) {
- return NULL;
- }
+ if (interface_configs == NULL) {
+ return NULL;
+ }
- key = wmem_strdup(wmem_epan_scope(), name);
- tmp = (interface_config_t *)g_hash_table_lookup(data_can_interfaces_by_name, key);
- wmem_free(wmem_epan_scope(), key);
+ key = wmem_strdup(wmem_epan_scope(), name);
+ tmp = (interface_config_t *)g_hash_table_lookup(data_can_interfaces_by_name, key);
+ wmem_free(wmem_epan_scope(), key);
- return tmp;
+ return tmp;
}
static void
can_free_key(gpointer key) {
- wmem_free(wmem_epan_scope(), key);
+ wmem_free(wmem_epan_scope(), key);
}
static void
post_update_can_interfaces_cb(void) {
- guint i;
- int *key_id = NULL;
- gchar *key_name = NULL;
-
- /* destroy old hash tables, if they exist */
- if (data_can_interfaces_by_id) {
- g_hash_table_destroy(data_can_interfaces_by_id);
- data_can_interfaces_by_id = NULL;
- }
- if (data_can_interfaces_by_name) {
- g_hash_table_destroy(data_can_interfaces_by_name);
- data_can_interfaces_by_name = NULL;
- }
-
- /* create new hash table */
- data_can_interfaces_by_id = g_hash_table_new_full(g_int_hash, g_int_equal, &can_free_key, NULL);
- data_can_interfaces_by_name = g_hash_table_new_full(g_str_hash, g_str_equal, &can_free_key, NULL);
-
- if (data_can_interfaces_by_id == NULL || data_can_interfaces_by_name == NULL || interface_configs == NULL || interface_config_num == 0) {
- return;
- }
-
- for (i = 0; i < interface_config_num; i++) {
- if (interface_configs[i].interface_id != 0xfffffff) {
- key_id = wmem_new(wmem_epan_scope(), int);
- *key_id = interface_configs[i].interface_id;
- g_hash_table_insert(data_can_interfaces_by_id, key_id, &interface_configs[i]);
- }
-
- if (interface_configs[i].interface_name != NULL && interface_configs[i].interface_name[0] != 0) {
- key_name = wmem_strdup(wmem_epan_scope(), interface_configs[i].interface_name);
- g_hash_table_insert(data_can_interfaces_by_name, key_name, &interface_configs[i]);
- }
- }
+ guint i;
+ int *key_id = NULL;
+ gchar *key_name = NULL;
+
+ /* destroy old hash tables, if they exist */
+ if (data_can_interfaces_by_id) {
+ g_hash_table_destroy(data_can_interfaces_by_id);
+ data_can_interfaces_by_id = NULL;
+ }
+ if (data_can_interfaces_by_name) {
+ g_hash_table_destroy(data_can_interfaces_by_name);
+ data_can_interfaces_by_name = NULL;
+ }
+
+ /* create new hash table */
+ data_can_interfaces_by_id = g_hash_table_new_full(g_int_hash, g_int_equal, &can_free_key, NULL);
+ data_can_interfaces_by_name = g_hash_table_new_full(g_str_hash, g_str_equal, &can_free_key, NULL);
+
+ if (data_can_interfaces_by_id == NULL || data_can_interfaces_by_name == NULL || interface_configs == NULL || interface_config_num == 0) {
+ return;
+ }
+
+ for (i = 0; i < interface_config_num; i++) {
+ if (interface_configs[i].interface_id != 0xfffffff) {
+ key_id = wmem_new(wmem_epan_scope(), int);
+ *key_id = interface_configs[i].interface_id;
+ g_hash_table_insert(data_can_interfaces_by_id, key_id, &interface_configs[i]);
+ }
+
+ if (interface_configs[i].interface_name != NULL && interface_configs[i].interface_name[0] != 0) {
+ key_name = wmem_strdup(wmem_epan_scope(), interface_configs[i].interface_name);
+ g_hash_table_insert(data_can_interfaces_by_name, key_name, &interface_configs[i]);
+ }
+ }
}
/* We match based on the config in the following order:
@@ -294,68 +321,188 @@ post_update_can_interfaces_cb(void) {
* - interface_name matches and interface_id = 0xffffffff
* - interface_name = "" and interface_id matches
*/
-
static guint
get_bus_id(packet_info *pinfo) {
- guint32 interface_id = pinfo->rec->rec_header.packet_header.interface_id;
- const char *interface_name = epan_get_interface_name(pinfo->epan, interface_id);
- interface_config_t *tmp = NULL;
+ if (!(pinfo->rec->presence_flags & WTAP_HAS_INTERFACE_ID)) {
+ return 0;
+ }
- if (!(pinfo->rec->presence_flags & WTAP_HAS_INTERFACE_ID)) {
- return 0;
- }
+ guint32 interface_id = pinfo->rec->rec_header.packet_header.interface_id;
+ unsigned section_number = pinfo->rec->presence_flags & WTAP_HAS_SECTION_NUMBER ? pinfo->rec->section_number : 0;
+ const char *interface_name = epan_get_interface_name(pinfo->epan, interface_id, section_number);
+ interface_config_t *tmp = NULL;
- if (interface_name != NULL && interface_name[0] != 0) {
- tmp = ht_lookup_interface_config_by_name(interface_name);
+ if (interface_name != NULL && interface_name[0] != 0) {
+ tmp = ht_lookup_interface_config_by_name(interface_name);
- if (tmp != NULL && (tmp->interface_id == 0xffffffff || tmp->interface_id == interface_id)) {
- /* name + id match or name match and id = any */
- return tmp->bus_id;
- }
+ if (tmp != NULL && (tmp->interface_id == 0xffffffff || tmp->interface_id == interface_id)) {
+ /* name + id match or name match and id = any */
+ return tmp->bus_id;
+ }
- tmp = ht_lookup_interface_config_by_id(interface_id);
+ tmp = ht_lookup_interface_config_by_id(interface_id);
- if (tmp != NULL && (tmp->interface_name == NULL || tmp->interface_name[0] == 0)) {
- /* id matches and name is any */
- return tmp->bus_id;
- }
- }
+ if (tmp != NULL && (tmp->interface_name == NULL || tmp->interface_name[0] == 0)) {
+ /* id matches and name is any */
+ return tmp->bus_id;
+ }
+ }
- /* we found nothing */
- return 0;
+ /* we found nothing */
+ return 0;
+}
+
+/* Senders and Receivers UAT */
+typedef struct _sender_receiver_config {
+ guint bus_id;
+ guint can_id;
+ gchar *sender_name;
+ gchar *receiver_name;
+} sender_receiver_config_t;
+
+#define DATAFILE_CAN_SENDER_RECEIVER "CAN_senders_receivers"
+
+static GHashTable *data_sender_receiver = NULL;
+static sender_receiver_config_t *sender_receiver_configs = NULL;
+static guint sender_receiver_config_num = 0;
+
+UAT_HEX_CB_DEF(sender_receiver_configs, bus_id, sender_receiver_config_t)
+UAT_HEX_CB_DEF(sender_receiver_configs, can_id, sender_receiver_config_t)
+UAT_CSTRING_CB_DEF(sender_receiver_configs, sender_name, sender_receiver_config_t)
+UAT_CSTRING_CB_DEF(sender_receiver_configs, receiver_name, sender_receiver_config_t)
+
+static void *
+copy_sender_receiver_config_cb(void *n, const void *o, size_t size _U_) {
+ sender_receiver_config_t *new_rec = (sender_receiver_config_t *)n;
+ const sender_receiver_config_t *old_rec = (const sender_receiver_config_t *)o;
+
+ new_rec->bus_id = old_rec->bus_id;
+ new_rec->can_id = old_rec->can_id;
+ new_rec->sender_name = g_strdup(old_rec->sender_name);
+ new_rec->receiver_name = g_strdup(old_rec->receiver_name);
+ return new_rec;
+}
+
+static bool
+update_sender_receiver_config(void *r, char **err) {
+ sender_receiver_config_t *rec = (sender_receiver_config_t *)r;
+
+ if (rec->bus_id > 0xffff) {
+ *err = ws_strdup_printf("We currently only support 16 bit bus identifiers (Bus ID: %i CAN ID: %i)", rec->bus_id, rec->can_id);
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static void
+free_sender_receiver_config_cb(void *r) {
+ sender_receiver_config_t *rec = (sender_receiver_config_t *)r;
+
+ /* freeing result of g_strdup */
+ g_free(rec->sender_name);
+ rec->sender_name = NULL;
+ g_free(rec->receiver_name);
+ rec->receiver_name = NULL;
+}
+
+static guint64
+sender_receiver_key(guint16 bus_id, guint32 can_id) {
+ return ((guint64)bus_id << 32) | can_id;
+}
+
+static sender_receiver_config_t *
+ht_lookup_sender_receiver_config(guint16 bus_id, guint32 can_id) {
+ sender_receiver_config_t *tmp = NULL;
+ guint64 key = 0;
+
+ if (sender_receiver_configs == NULL) {
+ return NULL;
+ }
+
+ key = sender_receiver_key(bus_id, can_id);
+ tmp = (sender_receiver_config_t *)g_hash_table_lookup(data_sender_receiver, &key);
+
+ if (tmp == NULL) {
+ key = sender_receiver_key(0, can_id);
+ tmp = (sender_receiver_config_t *)g_hash_table_lookup(data_sender_receiver, &key);
+ }
+
+ return tmp;
+}
+
+static void
+sender_receiver_free_key(gpointer key) {
+ wmem_free(wmem_epan_scope(), key);
+}
+
+static void
+post_update_sender_receiver_cb(void) {
+ guint i;
+ guint64 *key_id = NULL;
+
+ /* destroy old hash table, if it exist */
+ if (data_sender_receiver) {
+ g_hash_table_destroy(data_sender_receiver);
+ data_sender_receiver = NULL;
+ }
+
+ /* create new hash table */
+ data_sender_receiver = g_hash_table_new_full(g_int64_hash, g_int64_equal, &sender_receiver_free_key, NULL);
+
+ if (data_sender_receiver == NULL || sender_receiver_configs == NULL || sender_receiver_config_num == 0) {
+ return;
+ }
+
+ for (i = 0; i < sender_receiver_config_num; i++) {
+ key_id = wmem_new(wmem_epan_scope(), guint64);
+ *key_id = sender_receiver_key(sender_receiver_configs[i].bus_id, sender_receiver_configs[i].can_id);
+ g_hash_table_insert(data_sender_receiver, key_id, &sender_receiver_configs[i]);
+ }
+}
+
+gboolean
+socketcan_set_source_and_destination_columns(packet_info *pinfo, can_info_t *caninfo) {
+ sender_receiver_config_t *tmp = ht_lookup_sender_receiver_config(caninfo->bus_id, caninfo->id);
+
+ if (tmp != NULL) {
+ /* remove all addresses to support CAN as payload (e.g., TECMP) */
+ clear_address(&pinfo->net_src);
+ clear_address(&pinfo->dl_src);
+ clear_address(&pinfo->src);
+ clear_address(&pinfo->net_dst);
+ clear_address(&pinfo->dl_dst);
+ clear_address(&pinfo->dst);
+
+ col_add_fstr(pinfo->cinfo, COL_DEF_SRC, "%s", tmp->sender_name);
+ col_add_fstr(pinfo->cinfo, COL_DEF_DST, "%s", tmp->receiver_name);
+ return true;
+ }
+ return false;
}
gboolean
-socketcan_call_subdissectors(tvbuff_t* tvb, packet_info* pinfo, proto_tree* tree, struct can_info* can_info, const gboolean use_heuristics_first)
-{
- dissector_table_t effective_can_id_dissector_table = (can_info->id & CAN_EFF_FLAG) ? can_extended_id_dissector_table : can_id_dissector_table;
- guint32 effective_can_id = (can_info->id & CAN_EFF_FLAG) ? can_info->id & CAN_EFF_MASK : can_info->id & CAN_SFF_MASK;
-
- if (!dissector_try_uint_new(effective_can_id_dissector_table, effective_can_id, tvb, pinfo, tree, TRUE, can_info))
- {
- if (!use_heuristics_first)
- {
- if (!dissector_try_payload_new(subdissector_table, tvb, pinfo, tree, TRUE, can_info))
- {
- if (!dissector_try_heuristic(heur_subdissector_list, tvb, pinfo, tree, &heur_dtbl_entry, can_info))
- {
- return FALSE;
- }
- }
- }
- else
- {
- if (!dissector_try_heuristic(heur_subdissector_list, tvb, pinfo, tree, &heur_dtbl_entry, can_info))
- {
- if (!dissector_try_payload_new(subdissector_table, tvb, pinfo, tree, FALSE, can_info))
- {
- return FALSE;
- }
- }
- }
- }
-
- return TRUE;
+socketcan_call_subdissectors(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, struct can_info *can_info, const gboolean use_heuristics_first) {
+ dissector_table_t effective_can_id_dissector_table = (can_info->id & CAN_EFF_FLAG) ? can_extended_id_dissector_table : can_id_dissector_table;
+ guint32 effective_can_id = (can_info->id & CAN_EFF_FLAG) ? can_info->id & CAN_EFF_MASK : can_info->id & CAN_SFF_MASK;
+
+ if (!dissector_try_uint_new(effective_can_id_dissector_table, effective_can_id, tvb, pinfo, tree, TRUE, can_info)) {
+ if (!use_heuristics_first) {
+ if (!dissector_try_payload_new(subdissector_table, tvb, pinfo, tree, TRUE, can_info)) {
+ if (!dissector_try_heuristic(heur_subdissector_list, tvb, pinfo, tree, &heur_dtbl_entry, can_info)) {
+ return FALSE;
+ }
+ }
+ } else {
+ if (!dissector_try_heuristic(heur_subdissector_list, tvb, pinfo, tree, &heur_dtbl_entry, can_info)) {
+ if (!dissector_try_payload_new(subdissector_table, tvb, pinfo, tree, FALSE, can_info)) {
+ return FALSE;
+ }
+ }
+ }
+ }
+
+ return TRUE;
}
/*
@@ -367,738 +514,608 @@ socketcan_call_subdissectors(tvbuff_t* tvb, packet_info* pinfo, proto_tree* tree
* 2) a given SocketCAN frame is known to contain a CAN FD
* packet based on information outside the SocketCAN header;
*
- * 3) we don't know whether the given SocketCAN frame is a
- * classic CAN packet or a CAN FD packet, and will have
- * to check the CANFD_FDF bit in the "FD flags" field of
- * the SocketCAN headder to determine that.
+ * 3) a given SocketCAN frame is known to contain a CAN XL
+ * packet based on information outside the SocketCAN header;
+ *
+ * 4) we don't know whether the given SocketCAN frame is a
+ * classic CAN packet, a CAN FD packet, or a CAN XL packet,
+ * and will have to check the CANXL_XLF bit in the "Frame Length"
+ * field and the CANFD_FDF bit in the "FD flags" field of the
+ * SocketCAN header to determine that.
*/
typedef enum {
- PACKET_TYPE_CAN,
- PACKET_TYPE_CAN_FD,
- PACKET_TYPE_UNKNOWN
+ PACKET_TYPE_CAN,
+ PACKET_TYPE_CAN_FD,
+ PACKET_TYPE_CAN_XL,
+ PACKET_TYPE_UNKNOWN
} can_packet_type_t;
static int
-dissect_socketcan_common(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
- guint encoding, can_packet_type_t can_packet_type)
-{
- proto_tree *can_tree;
- proto_item *ti;
- guint8 frame_type;
- struct can_info can_info;
- int * const *can_flags;
-
- static int * const can_std_flags[] = {
- &hf_can_infoent_std,
- &hf_can_extflag,
- &hf_can_rtrflag,
- &hf_can_errflag,
- NULL,
- };
- static int * const can_ext_flags[] = {
- &hf_can_infoent_ext,
- &hf_can_extflag,
- &hf_can_rtrflag,
- &hf_can_errflag,
- NULL,
- };
- static int * const can_std_flags_fd[] = {
- &hf_can_infoent_std,
- &hf_can_extflag,
- NULL,
- };
- static int * const can_ext_flags_fd[] = {
- &hf_can_infoent_ext,
- &hf_can_extflag,
- NULL,
- };
- static int * const canfd_flag_fields[] = {
- &hf_canfd_brsflag,
- &hf_canfd_esiflag,
- NULL,
- };
- static int * const can_err_flags[] = {
- &hf_can_errflag,
- &hf_can_err_tx_timeout,
- &hf_can_err_lostarb,
- &hf_can_err_ctrl,
- &hf_can_err_prot,
- &hf_can_err_trx,
- &hf_can_err_ack,
- &hf_can_err_busoff,
- &hf_can_err_buserror,
- &hf_can_err_restarted,
- &hf_can_err_reserved,
- NULL,
- };
-
- can_info.id = tvb_get_guint32(tvb, 0, encoding);
- can_info.len = tvb_get_guint8(tvb, CAN_LEN_OFFSET);
+dissect_socketcan_common(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, guint encoding, guint xl_encoding, can_packet_type_t can_packet_type) {
+ proto_tree *can_tree;
+ proto_item *ti;
+ guint8 frame_type;
+ can_info_t can_info;
+ int * const *can_flags_id;
+
+ static int * const can_std_flags_id[] = {
+ &hf_can_infoent_std,
+ &hf_can_extflag,
+ &hf_can_rtrflag,
+ &hf_can_errflag,
+ NULL,
+ };
+ static int * const can_ext_flags_id[] = {
+ &hf_can_infoent_ext,
+ &hf_can_extflag,
+ &hf_can_rtrflag,
+ &hf_can_errflag,
+ NULL,
+ };
+ static int * const canfd_std_flags_id[] = {
+ &hf_can_infoent_std,
+ &hf_can_extflag,
+ NULL,
+ };
+ static int * const canfd_ext_flags_id[] = {
+ &hf_can_infoent_ext,
+ &hf_can_extflag,
+ NULL,
+ };
+ static int * const canfd_flag_fields[] = {
+ &hf_canfd_brsflag,
+ &hf_canfd_esiflag,
+ NULL,
+ };
+ static int * const can_err_flags[] = {
+ &hf_can_errflag,
+ &hf_can_err_tx_timeout,
+ &hf_can_err_lostarb,
+ &hf_can_err_ctrl,
+ &hf_can_err_prot,
+ &hf_can_err_trx,
+ &hf_can_err_ack,
+ &hf_can_err_busoff,
+ &hf_can_err_buserror,
+ &hf_can_err_restarted,
+ &hf_can_err_reserved,
+ NULL,
+ };
+ static int * const canxl_prio_vcid_fields[] = {
+ &hf_canxl_priority,
+ &hf_canxl_vcid,
+ NULL,
+ };
+ static int * const canxl_flag_fields[] = {
+ &hf_canxl_secflag,
+ NULL,
+ };
+
+ /*
+ * If we weren't told the type of this frame, check
+ * whether the CANFD_FDF flag is set in the FD flags
+ * field of the header; if so, it's a CAN FD frame.
+ * otherwise, it's a CAN frame.
+ *
+ * However, trust the CANFD_FDF flag only if the only
+ * bits set in the FD flags field are the known bits,
+ * and the two bytes following that field are both
+ * zero. This is because some older LINKTYPE_CAN_SOCKETCAN
+ * frames had uninitialized junk in the FD flags field,
+ * so we treat a frame with what appears to be uninitialized
+ * junk as being CAN rather than CAN FD, under the assumption
+ * that the CANFD_FDF bit is set because the field is
+ * uninitialized, not because it was explicitly set because
+ * it's a CAN FD frame. At least some newer code that sets
+ * that flag also makes sure that the fields in question are
+ * initialized, so we assume that if they're not initialized
+ * the code is older code that didn't support CAN FD.
+ */
+ if (can_packet_type == PACKET_TYPE_UNKNOWN) {
+ guint8 frame_length;
+ guint8 fd_flags;
+
+ /*
+ * Check whether the frame has the CANXL_XLF flag set in what
+ * is in the location of the frame length field of a CAN classic
+ * or CAN FD frame; if so, then it's a CAN XL frame (and that
+ * field is the flags field of that frame).
+ */
+ frame_length = tvb_get_guint8(tvb, CAN_LEN_OFFSET);
+ if (frame_length & CANXL_XLF) {
+ can_packet_type = PACKET_TYPE_CAN_XL;
+ } else {
+ /*
+ * This is a CAN classic or CAN FD frame.
+ * Check whether the flags field has the CANFD_FDF
+ * flag set, has no unknown flag bits set, and has
+ * no bits set in the two reserved fields. If so,
+ * it's a CAN FD frame; otherwise, it's either a
+ * CAN classic frame, or a frame where the CANFD_FDF
+ * flag is set but where that might just be because
+ * that field contains uninitialized junk rather
+ * than because it's a CAN FD frame, so we treat it
+ * as a CAN classic frame.
+ */
+ fd_flags = tvb_get_guint8(tvb, CANFD_FLAG_OFFSET);
+
+ if ((fd_flags & CANFD_FDF) &&
+ ((fd_flags & ~(CANFD_BRS | CANFD_ESI | CANFD_FDF)) == 0) &&
+ tvb_get_guint8(tvb, CANFD_FLAG_OFFSET + 1) == 0 &&
+ tvb_get_guint8(tvb, CANFD_FLAG_OFFSET + 2) == 0) {
+ can_packet_type = PACKET_TYPE_CAN_FD;
+ } else {
+ if (tvb_reported_length(tvb) == 72)
+ can_packet_type = PACKET_TYPE_CAN_FD;
+ else
+ can_packet_type = PACKET_TYPE_CAN;
+ }
+ }
+ }
+
+ can_info.bus_id = get_bus_id(pinfo);
+
+ if (can_packet_type == PACKET_TYPE_CAN_XL) {
+ can_info.fd = CAN_TYPE_CAN_XL;
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "CANXL");
+ col_clear(pinfo->cinfo, COL_INFO);
+
+ can_info.id = 0; /* XXX - is there an "ID" for XL frames? */
+
+ ti = proto_tree_add_item(tree, proto_can, tvb, 0, -1, ENC_NA);
+ proto_item_set_hidden(ti);
+ ti = proto_tree_add_item(tree, proto_canxl, tvb, 0, -1, ENC_NA);
+ can_tree = proto_item_add_subtree(ti, ett_can_xl);
+
+ guint32 proto_vcid;
+
+ /*
+ * The priority/VCID field is big-endian in LINKTYPE_CAN_SOCKETCAN
+ * captures, for historical reasons. It's host-endian in
+ * Linux cooked captures. This means we use the non-XL encoding.
+ */
+ proto_tree_add_bitmask_list(can_tree, tvb, 0, 4, canxl_prio_vcid_fields, encoding);
+ proto_vcid = tvb_get_guint32(tvb, 0, encoding);
+ col_add_fstr(pinfo->cinfo, COL_INFO, "Priority: %u (0x%03x), VCID: %u (0x%02X)", proto_vcid & 0x7FF, proto_vcid & 0x7FF, (proto_vcid >> 16) & 0xFF, (proto_vcid >> 16) & 0xFF);
+ proto_item_append_text(can_tree, ", Priority: %u (0x%03x), VCID: %u (0x%02X)", proto_vcid & 0x7FF, proto_vcid & 0x7FF, (proto_vcid >> 16) & 0xFF, (proto_vcid >> 16) & 0xFF);
+ proto_tree_add_bitmask_list(can_tree, tvb, 4, 1, canxl_flag_fields, xl_encoding);
+
+ socketcan_set_source_and_destination_columns(pinfo, &can_info);
+
+ guint32 sdu_type;
+
/*
- * If we weren't told the type of this frame, check
- * whether the CANFD_FDF flag is set in the FD flags
- * field of the header; if so, it's a CAN FD frame.
- * otherwise, it's a CAN frame.
- *
- * However, trust the CANFD_FDF flag only if the only
- * bits set in the FD flags field are the known bits,
- * and the two bytes following that field are both
- * zero. This is because some older LINKTYPE_CAN_SOCKETCAN
- * frames had uninitialized junk in the FD flags field,
- * so we treat a frame with what appears to be uninitialized
- * junk as being CAN rather than CAN FD, under the assumption
- * that the CANFD_FDF bit is set because the field is
- * uninitialized, not because it was explicitly set because
- * it's a CAN FD frame. At least some newer code that sets
- * that flag also makes sure that the fields in question are
- * initialized, so we assume that if they're not initialized
- * the code is older code that didn't support CAN FD.
+ * These fields are, if multi-byte, little-endian in
+ * LINKTYPE_CAN_SOCKETCAN captures, so use xl_encoding.
*/
- if (can_packet_type == PACKET_TYPE_UNKNOWN) {
- guint8 fd_flags;
-
- fd_flags = tvb_get_guint8(tvb, CANFD_FLAG_OFFSET);
-
- if ((fd_flags & CANFD_FDF) &&
- ((fd_flags & ~(CANFD_BRS|CANFD_ESI|CANFD_FDF)) == 0) &&
- tvb_get_guint8(tvb, CANFD_FLAG_OFFSET + 1) == 0 &&
- tvb_get_guint8(tvb, CANFD_FLAG_OFFSET + 2) == 0)
- can_packet_type = PACKET_TYPE_CAN_FD;
- else
- can_packet_type = PACKET_TYPE_CAN;
- }
- can_info.fd = (can_packet_type == PACKET_TYPE_CAN_FD);
- can_info.bus_id = get_bus_id(pinfo);
-
- /* Error Message Frames are only encapsulated in Classic CAN frames */
- if (can_packet_type == PACKET_TYPE_CAN && (can_info.id & CAN_ERR_FLAG))
- {
- frame_type = LINUX_CAN_ERR;
- can_flags = can_err_flags;
- }
- else if (can_info.id & CAN_EFF_FLAG)
- {
- frame_type = LINUX_CAN_EXT;
- can_info.id &= (CAN_EFF_MASK | CAN_FLAG_MASK);
- can_flags = (can_packet_type == PACKET_TYPE_CAN_FD) ? can_ext_flags_fd : can_ext_flags;
- }
- else
- {
- frame_type = LINUX_CAN_STD;
- can_info.id &= (CAN_SFF_MASK | CAN_FLAG_MASK);
- can_flags = (can_packet_type == PACKET_TYPE_CAN_FD) ? can_std_flags_fd : can_std_flags;
- }
-
- col_set_str(pinfo->cinfo, COL_PROTOCOL, (can_packet_type == PACKET_TYPE_CAN_FD) ? "CANFD" : "CAN");
- col_clear(pinfo->cinfo, COL_INFO);
-
- guint32 effective_can_id = (can_info.id & CAN_EFF_FLAG) ? can_info.id & CAN_EFF_MASK : can_info.id & CAN_SFF_MASK;
- char* id_name = (can_info.id & CAN_EFF_FLAG) ? "Ext. ID" : "ID";
- col_add_fstr(pinfo->cinfo, COL_INFO, "%s: %d (0x%" G_GINT32_MODIFIER "x), Length: %d", id_name, effective_can_id, effective_can_id, can_info.len);
-
- ti = proto_tree_add_item(tree, (can_packet_type == PACKET_TYPE_CAN_FD) ? proto_canfd : proto_can, tvb, 0, -1, ENC_NA);
- can_tree = proto_item_add_subtree(ti, (can_packet_type == PACKET_TYPE_CAN_FD) ? ett_can_fd : ett_can);
-
- proto_item_append_text(can_tree, ", %s: %d (0x%" G_GINT32_MODIFIER "x), Length: %d", id_name, effective_can_id, effective_can_id, can_info.len);
-
- proto_tree_add_bitmask_list(can_tree, tvb, 0, 4, can_flags, encoding);
- proto_tree_add_item(can_tree, hf_can_len, tvb, CAN_LEN_OFFSET, 1, ENC_NA);
- if (frame_type == LINUX_CAN_ERR && can_info.len != CAN_ERR_DLC)
- {
- proto_tree_add_expert(tree, pinfo, &ei_can_err_dlc_mismatch, tvb, CAN_LEN_OFFSET, 1);
- }
- if (can_packet_type == PACKET_TYPE_CAN_FD) {
- proto_tree_add_bitmask_list(can_tree, tvb, CANFD_FLAG_OFFSET, 1, canfd_flag_fields, ENC_NA);
- proto_tree_add_item(can_tree, hf_can_reserved, tvb, CANFD_FLAG_OFFSET+1, 2, ENC_NA);
- } else
- proto_tree_add_item(can_tree, hf_can_reserved, tvb, CANFD_FLAG_OFFSET, 3, ENC_NA);
-
- if (frame_type == LINUX_CAN_ERR)
- {
- int * const *flag;
- const char *sepa = ": ";
-
- col_set_str(pinfo->cinfo, COL_INFO, "ERR");
-
- for (flag = can_err_flags; *flag; flag++)
- {
- header_field_info *hfi;
-
- hfi = proto_registrar_get_nth(**flag);
- if (!hfi)
- continue;
-
- if ((can_info.id & hfi->bitmask & ~CAN_FLAG_MASK) == 0)
- continue;
-
- col_append_sep_str(pinfo->cinfo, COL_INFO, sepa, hfi->name);
- sepa = ", ";
- }
-
- if (can_info.id & CAN_ERR_LOSTARB)
- proto_tree_add_item(can_tree, hf_can_err_lostarb_bit_number, tvb, CAN_DATA_OFFSET+0, 1, ENC_NA);
- if (can_info.id & CAN_ERR_CTRL)
- {
- static int * const can_err_ctrl_flags[] = {
- &hf_can_err_ctrl_rx_overflow,
- &hf_can_err_ctrl_tx_overflow,
- &hf_can_err_ctrl_rx_warning,
- &hf_can_err_ctrl_tx_warning,
- &hf_can_err_ctrl_rx_passive,
- &hf_can_err_ctrl_tx_passive,
- &hf_can_err_ctrl_active,
- NULL,
- };
-
- proto_tree_add_bitmask_list(can_tree, tvb, CAN_DATA_OFFSET+1, 1, can_err_ctrl_flags, ENC_NA);
- }
- if (can_info.id & CAN_ERR_PROT)
- {
- static int * const can_err_prot_error_type_flags[] = {
- &hf_can_err_prot_error_type_bit,
- &hf_can_err_prot_error_type_form,
- &hf_can_err_prot_error_type_stuff,
- &hf_can_err_prot_error_type_bit0,
- &hf_can_err_prot_error_type_bit1,
- &hf_can_err_prot_error_type_overload,
- &hf_can_err_prot_error_type_active,
- &hf_can_err_prot_error_type_tx,
- NULL
- };
- proto_tree_add_bitmask_list(can_tree, tvb, CAN_DATA_OFFSET+2, 1, can_err_prot_error_type_flags, ENC_NA);
- proto_tree_add_item(can_tree, hf_can_err_prot_error_location, tvb, CAN_DATA_OFFSET+3, 1, ENC_NA);
- }
- if (can_info.id & CAN_ERR_TRX)
- {
- proto_tree_add_item(can_tree, hf_can_err_trx_canh, tvb, CAN_DATA_OFFSET+4, 1, ENC_NA);
- proto_tree_add_item(can_tree, hf_can_err_trx_canl, tvb, CAN_DATA_OFFSET+4, 1, ENC_NA);
- }
- proto_tree_add_item(can_tree, hf_can_err_ctrl_specific, tvb, CAN_DATA_OFFSET+5, 3, ENC_NA);
- }
- else
- {
- tvbuff_t *next_tvb;
-
- if (can_info.id & CAN_RTR_FLAG)
- {
- col_append_str(pinfo->cinfo, COL_INFO, "(Remote Transmission Request)");
- }
-
- next_tvb = tvb_new_subset_length(tvb, CAN_DATA_OFFSET, can_info.len);
-
- if (!socketcan_call_subdissectors(next_tvb, pinfo, tree, &can_info, heuristic_first))
- {
- call_data_dissector(next_tvb, pinfo, tree);
- }
- }
-
- if (tvb_captured_length_remaining(tvb, CAN_DATA_OFFSET+can_info.len) > 0)
- {
- proto_tree_add_item(can_tree, hf_can_padding, tvb, CAN_DATA_OFFSET+can_info.len, -1, ENC_NA);
- }
-
- return tvb_captured_length(tvb);
+ proto_tree_add_item_ret_uint(can_tree, hf_canxl_sdu_type, tvb, 5, 1, ENC_NA, &sdu_type);
+ proto_tree_add_item_ret_uint(can_tree, hf_canxl_len, tvb, CANXL_LEN_OFFSET, 2, xl_encoding, &can_info.len);
+ col_append_fstr(pinfo->cinfo, COL_INFO, ", Length: %u", can_info.len);
+ proto_item_append_text(can_tree, ", Length: %u", can_info.len);
+ proto_tree_add_item(can_tree, hf_canxl_acceptance_field, tvb, CANXL_LEN_OFFSET+2, 4, xl_encoding);
+
+ tvbuff_t *next_tvb;
+
+ next_tvb = tvb_new_subset_length(tvb, CANXL_DATA_OFFSET, can_info.len);
+
+ if (!dissector_try_uint_new(canxl_sdu_type_dissector_table, sdu_type, next_tvb, pinfo, tree, TRUE, &can_info)) {
+ call_data_dissector(next_tvb, pinfo, tree);
+ }
+
+ if (tvb_captured_length_remaining(tvb, CANXL_DATA_OFFSET+can_info.len) > 0) {
+ proto_tree_add_item(can_tree, hf_can_padding, tvb, CANXL_DATA_OFFSET+can_info.len, -1, ENC_NA);
+ }
+ } else {
+ if (can_packet_type == PACKET_TYPE_CAN_FD) {
+ can_info.fd = CAN_TYPE_CAN_FD;
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "CANFD");
+ } else {
+ can_info.fd = CAN_TYPE_CAN_CLASSIC;
+ col_set_str(pinfo->cinfo, COL_PROTOCOL, "CAN");
+ }
+ col_clear(pinfo->cinfo, COL_INFO);
+
+ ti = proto_tree_add_item(tree, proto_can, tvb, 0, -1, ENC_NA);
+ if (can_packet_type == PACKET_TYPE_CAN_FD) {
+ proto_item_set_hidden(ti);
+ ti = proto_tree_add_item(tree, proto_canfd, tvb, 0, -1, ENC_NA);
+ }
+ can_tree = proto_item_add_subtree(ti, (can_packet_type == PACKET_TYPE_CAN_FD) ? ett_can_fd : ett_can);
+
+ /* Get the ID and flags field */
+ can_info.id = tvb_get_guint32(tvb, 0, encoding);
+
+ /* Error Message Frames are only encapsulated in Classic CAN frames */
+ if (can_packet_type == PACKET_TYPE_CAN && (can_info.id & CAN_ERR_FLAG)) {
+ frame_type = LINUX_CAN_ERR;
+ can_flags_id = can_err_flags;
+ } else if (can_info.id & CAN_EFF_FLAG) {
+ frame_type = LINUX_CAN_EXT;
+ can_info.id &= (CAN_EFF_MASK | CAN_FLAG_MASK);
+ can_flags_id = (can_packet_type == PACKET_TYPE_CAN_FD) ? canfd_ext_flags_id : can_ext_flags_id;
+ } else {
+ frame_type = LINUX_CAN_STD;
+ can_info.id &= (CAN_SFF_MASK | CAN_FLAG_MASK);
+ can_flags_id = (can_packet_type == PACKET_TYPE_CAN_FD) ? canfd_std_flags_id : can_std_flags_id;
+ }
+
+ socketcan_set_source_and_destination_columns(pinfo, &can_info);
+
+ proto_tree_add_bitmask_list(can_tree, tvb, 0, 4, can_flags_id, encoding);
+ if (can_info.id & CAN_EFF_FLAG) {
+ col_add_fstr(pinfo->cinfo, COL_INFO, "Ext. ID: %u (0x%08x)", can_info.id & CAN_EFF_MASK, can_info.id & CAN_EFF_MASK);
+ proto_item_append_text(can_tree, ", Ext. ID: %u (0x%08x)", can_info.id & CAN_EFF_MASK, can_info.id & CAN_EFF_MASK);
+ } else {
+ col_add_fstr(pinfo->cinfo, COL_INFO, "ID: %u (0x%03x)", can_info.id & CAN_SFF_MASK, can_info.id & CAN_SFF_MASK);
+ proto_item_append_text(can_tree, ", ID: %u (0x%03x)", can_info.id & CAN_SFF_MASK, can_info.id & CAN_SFF_MASK);
+ }
+ proto_tree_add_item_ret_uint(can_tree, hf_can_len, tvb, CAN_LEN_OFFSET, 1, ENC_NA, &can_info.len);
+ col_append_fstr(pinfo->cinfo, COL_INFO, ", Length: %u", can_info.len);
+ proto_item_append_text(can_tree, ", Length: %u", can_info.len);
+
+ if (frame_type == LINUX_CAN_ERR && can_info.len != CAN_ERR_DLC) {
+ proto_tree_add_expert(tree, pinfo, &ei_can_err_dlc_mismatch, tvb, CAN_LEN_OFFSET, 1);
+ }
+
+ if (can_packet_type == PACKET_TYPE_CAN_FD) {
+ proto_tree_add_bitmask_list(can_tree, tvb, CANFD_FLAG_OFFSET, 1, canfd_flag_fields, ENC_NA);
+ proto_tree_add_item(can_tree, hf_can_reserved, tvb, CANFD_FLAG_OFFSET+1, 2, ENC_NA);
+ } else {
+ proto_tree_add_item(can_tree, hf_can_reserved, tvb, CANFD_FLAG_OFFSET, 3, ENC_NA);
+ }
+
+ if (frame_type == LINUX_CAN_ERR) {
+ int * const *flag;
+ const char *sepa = ": ";
+
+ col_set_str(pinfo->cinfo, COL_INFO, "ERR");
+
+ for (flag = can_err_flags; *flag; flag++) {
+ header_field_info *hfi;
+
+ hfi = proto_registrar_get_nth(**flag);
+ if (!hfi)
+ continue;
+
+ if ((can_info.id & hfi->bitmask & ~CAN_FLAG_MASK) == 0)
+ continue;
+
+ col_append_sep_str(pinfo->cinfo, COL_INFO, sepa, hfi->name);
+ sepa = ", ";
+ }
+
+ if (can_info.id & CAN_ERR_LOSTARB) {
+ proto_tree_add_item(can_tree, hf_can_err_lostarb_bit_number, tvb, CAN_DATA_OFFSET + 0, 1, ENC_NA);
+ }
+
+ if (can_info.id & CAN_ERR_CTRL) {
+ static int * const can_err_ctrl_flags[] = {
+ &hf_can_err_ctrl_rx_overflow,
+ &hf_can_err_ctrl_tx_overflow,
+ &hf_can_err_ctrl_rx_warning,
+ &hf_can_err_ctrl_tx_warning,
+ &hf_can_err_ctrl_rx_passive,
+ &hf_can_err_ctrl_tx_passive,
+ &hf_can_err_ctrl_active,
+ NULL,
+ };
+
+ proto_tree_add_bitmask_list(can_tree, tvb, CAN_DATA_OFFSET+1, 1, can_err_ctrl_flags, ENC_NA);
+ }
+
+ if (can_info.id & CAN_ERR_PROT) {
+ static int * const can_err_prot_error_type_flags[] = {
+ &hf_can_err_prot_error_type_bit,
+ &hf_can_err_prot_error_type_form,
+ &hf_can_err_prot_error_type_stuff,
+ &hf_can_err_prot_error_type_bit0,
+ &hf_can_err_prot_error_type_bit1,
+ &hf_can_err_prot_error_type_overload,
+ &hf_can_err_prot_error_type_active,
+ &hf_can_err_prot_error_type_tx,
+ NULL
+ };
+ proto_tree_add_bitmask_list(can_tree, tvb, CAN_DATA_OFFSET+2, 1, can_err_prot_error_type_flags, ENC_NA);
+ proto_tree_add_item(can_tree, hf_can_err_prot_error_location, tvb, CAN_DATA_OFFSET+3, 1, ENC_NA);
+ }
+
+ if (can_info.id & CAN_ERR_TRX) {
+ proto_tree_add_item(can_tree, hf_can_err_trx_canh, tvb, CAN_DATA_OFFSET+4, 1, ENC_NA);
+ proto_tree_add_item(can_tree, hf_can_err_trx_canl, tvb, CAN_DATA_OFFSET+4, 1, ENC_NA);
+ }
+
+ proto_tree_add_item(can_tree, hf_can_err_ctrl_specific, tvb, CAN_DATA_OFFSET+5, 3, ENC_NA);
+ } else {
+ tvbuff_t *next_tvb;
+
+ if (can_info.id & CAN_RTR_FLAG) {
+ col_append_str(pinfo->cinfo, COL_INFO, "(Remote Transmission Request)");
+ }
+
+ next_tvb = tvb_new_subset_length(tvb, CAN_DATA_OFFSET, can_info.len);
+
+ if (!socketcan_call_subdissectors(next_tvb, pinfo, tree, &can_info, heuristic_first)) {
+ call_data_dissector(next_tvb, pinfo, tree);
+ }
+ }
+
+ if (tvb_captured_length_remaining(tvb, CAN_DATA_OFFSET+can_info.len) > 0) {
+ proto_tree_add_item(can_tree, hf_can_padding, tvb, CAN_DATA_OFFSET+can_info.len, -1, ENC_NA);
+ }
+ }
+
+ return tvb_captured_length(tvb);
}
static int
-dissect_socketcan_bigendian(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
- void* data _U_)
-{
- return dissect_socketcan_common(tvb, pinfo, tree,
- byte_swap ? ENC_LITTLE_ENDIAN : ENC_BIG_ENDIAN, PACKET_TYPE_UNKNOWN);
+dissect_socketcan_bigendian(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) {
+ return dissect_socketcan_common(tvb, pinfo, tree,
+ byte_swap ? ENC_LITTLE_ENDIAN : ENC_BIG_ENDIAN,
+ ENC_LITTLE_ENDIAN,
+ PACKET_TYPE_UNKNOWN);
}
static int
-dissect_socketcan_classic(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
- void* data _U_)
-{
- return dissect_socketcan_common(tvb, pinfo, tree,
- byte_swap ? ENC_ANTI_HOST_ENDIAN : ENC_HOST_ENDIAN, PACKET_TYPE_CAN);
+dissect_socketcan_classic(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) {
+ return dissect_socketcan_common(tvb, pinfo, tree,
+ byte_swap ? ENC_ANTI_HOST_ENDIAN : ENC_HOST_ENDIAN,
+ ENC_HOST_ENDIAN, /* Not used, as this is CAN classic, not CAN XL */
+ PACKET_TYPE_CAN);
}
static int
-dissect_socketcan_fd(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree,
- void* data _U_)
-{
- return dissect_socketcan_common(tvb, pinfo, tree,
- byte_swap ? ENC_ANTI_HOST_ENDIAN : ENC_HOST_ENDIAN, PACKET_TYPE_CAN_FD);
+dissect_socketcan_fd(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) {
+ return dissect_socketcan_common(tvb, pinfo, tree,
+ byte_swap ? ENC_ANTI_HOST_ENDIAN : ENC_HOST_ENDIAN,
+ ENC_HOST_ENDIAN, /* Not used, as this is CAN FD, not CAN XL */
+ PACKET_TYPE_CAN_FD);
}
-void
-proto_register_socketcan(void)
-{
- static hf_register_info hf[] = {
- {
- &hf_can_infoent_ext,
- {
- "ID", "can.id",
- FT_UINT32, BASE_DEC_HEX,
- NULL, CAN_EFF_MASK,
- NULL, HFILL
- }
- },
- {
- &hf_can_infoent_std,
- {
- "ID", "can.id",
- FT_UINT32, BASE_DEC_HEX,
- NULL, CAN_SFF_MASK,
- NULL, HFILL
- }
- },
- {
- &hf_can_extflag,
- {
- "Extended Flag", "can.flags.xtd",
- FT_BOOLEAN, 32,
- NULL, CAN_EFF_FLAG,
- NULL, HFILL
- }
- },
- {
- &hf_can_rtrflag,
- {
- "Remote Transmission Request Flag", "can.flags.rtr",
- FT_BOOLEAN, 32,
- NULL, CAN_RTR_FLAG,
- NULL, HFILL
- }
- },
- {
- &hf_can_errflag,
- {
- "Error Message Flag", "can.flags.err",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_FLAG,
- NULL, HFILL
- }
- },
- {
- &hf_can_len,
- {
- "Frame-Length", "can.len",
- FT_UINT8, BASE_DEC,
- NULL, 0x0,
- NULL, HFILL
- }
- },
- {
- &hf_can_reserved,
- {
- "Reserved", "can.reserved",
- FT_BYTES, BASE_NONE,
- NULL, 0x0,
- NULL, HFILL
- }
- },
- {
- &hf_can_padding,
- {
- "Padding", "can.padding",
- FT_BYTES, BASE_NONE,
- NULL, 0x0,
- NULL, HFILL
- }
- },
- {
- &hf_canfd_brsflag,
- {
- "Bit Rate Setting", "canfd.flags.brs",
- FT_BOOLEAN, 8,
- NULL, CANFD_BRS,
- NULL, HFILL
- }
- },
- {
- &hf_canfd_esiflag,
- {
- "Error State Indicator", "canfd.flags.esi",
- FT_BOOLEAN, 8,
- NULL, CANFD_ESI,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_tx_timeout,
- {
- "Transmit timeout", "can.err.tx_timeout",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_TX_TIMEOUT,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_lostarb,
- {
- "Lost arbitration", "can.err.lostarb",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_LOSTARB,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl,
- {
- "Controller problems", "can.err.ctrl",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_CTRL,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot,
- {
- "Protocol violation", "can.err.prot",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_PROT,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_trx,
- {
- "Transceiver status", "can.err.trx",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_TRX,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ack,
- {
- "No acknowledgement", "can.err.ack",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_ACK,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_busoff,
- {
- "Bus off", "can.err.busoff",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_BUSOFF,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_buserror,
- {
- "Bus error", "can.err.buserror",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_BUSERROR,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_restarted,
- {
- "Controller restarted", "can.err.restarted",
- FT_BOOLEAN, 32,
- NULL, CAN_ERR_RESTARTED,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_reserved,
- {
- "Reserved", "can.err.reserved",
- FT_UINT32, BASE_HEX,
- NULL, CAN_ERR_RESERVED,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_lostarb_bit_number,
- {
- "Lost arbitration in bit number", "can.err.lostarb.bitnum",
- FT_UINT8, BASE_DEC,
- NULL, 0,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_rx_overflow,
- {
- "RX buffer overflow", "can.err.ctrl.rx_overflow",
- FT_BOOLEAN, 8,
- NULL, 0x01,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_tx_overflow,
- {
- "TX buffer overflow", "can.err.ctrl.tx_overflow",
- FT_BOOLEAN, 8,
- NULL, 0x02,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_rx_warning,
- {
- "Reached warning level for RX errors", "can.err.ctrl.rx_warning",
- FT_BOOLEAN, 8,
- NULL, 0x04,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_tx_warning,
- {
- "Reached warning level for TX errors", "can.err.ctrl.tx_warning",
- FT_BOOLEAN, 8,
- NULL, 0x08,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_rx_passive,
- {
- "Reached error passive status RX", "can.err.ctrl.rx_passive",
- FT_BOOLEAN, 8,
- NULL, 0x10,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_tx_passive,
- {
- "Reached error passive status TX", "can.err.ctrl.tx_passive",
- FT_BOOLEAN, 8,
- NULL, 0x20,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_active,
- {
- "Recovered to error active state", "can.err.ctrl.active",
- FT_BOOLEAN, 8,
- NULL, 0x40,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_bit,
- {
- "Single bit error", "can.err.prot.type.bit",
- FT_BOOLEAN, 8,
- NULL, 0x01,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_form,
- {
- "Frame format error", "can.err.prot.type.form",
- FT_BOOLEAN, 8,
- NULL, 0x02,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_stuff,
- {
- "Bit stuffing error", "can.err.prot.type.stuff",
- FT_BOOLEAN, 8,
- NULL, 0x04,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_bit0,
- {
- "Unable to send dominant bit", "can.err.prot.type.bit0",
- FT_BOOLEAN, 8,
- NULL, 0x08,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_bit1,
- {
- "Unable to send recessive bit", "can.err.prot.type.bit1",
- FT_BOOLEAN, 8,
- NULL, 0x10,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_overload,
- {
- "Bus overload", "can.err.prot.type.overload",
- FT_BOOLEAN, 8,
- NULL, 0x20,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_active,
- {
- "Active error announcement", "can.err.prot.type.active",
- FT_BOOLEAN, 8,
- NULL, 0x40,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_type_tx,
- {
- "Error occurred on transmission", "can.err.prot.type.tx",
- FT_BOOLEAN, 8,
- NULL, 0x80,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_prot_error_location,
- {
- "Protocol error location", "can.err.prot.location",
- FT_UINT8, BASE_DEC,
- VALS(can_err_prot_error_location_vals), 0,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_trx_canh,
- {
- "Transceiver CANH status", "can.err.trx.canh",
- FT_UINT8, BASE_DEC,
- VALS(can_err_trx_canh_vals), 0x0F,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_trx_canl,
- {
- "Transceiver CANL status", "can.err.trx.canl",
- FT_UINT8, BASE_DEC,
- VALS(can_err_trx_canl_vals), 0xF0,
- NULL, HFILL
- }
- },
- {
- &hf_can_err_ctrl_specific,
- {
- "Controller specific data", "can.err.ctrl_specific",
- FT_BYTES, SEP_SPACE,
- NULL, 0,
- NULL, HFILL
- }
- },
- };
- uat_t *can_interface_uat = NULL;
-
- /* Setup protocol subtree array */
- static gint *ett[] =
- {
- &ett_can,
- &ett_can_fd
- };
-
- static ei_register_info ei[] = {
- {
- &ei_can_err_dlc_mismatch,
- {
- "can.err.dlc_mismatch", PI_MALFORMED, PI_ERROR,
- "ERROR: DLC mismatch", EXPFILL
- }
- }
- };
-
- module_t *can_module;
-
- proto_can = proto_register_protocol("Controller Area Network", "CAN", "can");
- /*
- * "can-hostendian" is a legacy name (there never was, in any libpcap
- * release, a SocketCAN LINKTYPE_ value for a host-endian CAN ID
- * and flags field); we need to keep it around in case some candump
- * or Busmaster capture that was saved as a pcap or pcapng file,
- * as those use a linktype of LINKTYPE_WIRESHARK_UPPER_PDU with
- * "can-hostendian" as the dissector name.
- */
- socketcan_classic_handle = register_dissector("can-hostendian", dissect_socketcan_classic, proto_can);
-
- proto_canfd = proto_register_protocol("Controller Area Network FD", "CANFD", "canfd");
- socketcan_fd_handle = register_dissector("canfd", dissect_socketcan_fd, proto_canfd);
-
- proto_register_field_array(proto_can, hf, array_length(hf));
- proto_register_subtree_array(ett, array_length(ett));
-
- expert_register_field_array(expert_register_protocol(proto_can), ei, array_length(ei));
-
- can_module = prefs_register_protocol(proto_can, NULL);
-
- prefs_register_obsolete_preference(can_module, "protocol");
- prefs_register_bool_preference(can_module, "byte_swap",
- "Byte-swap the CAN ID/flags field",
- "Whether the CAN ID/flags field should be byte-swapped",
- &byte_swap);
-
- prefs_register_bool_preference(can_module, "try_heuristic_first",
- "Try heuristic sub-dissectors first",
- "Try to decode a packet using an heuristic sub-dissector"
- " before using a sub-dissector registered to \"decode as\"",
- &heuristic_first);
-
- can_id_dissector_table = register_dissector_table("can.id", "CAN ID", proto_can, FT_UINT32, BASE_DEC);
-
- can_extended_id_dissector_table = register_dissector_table("can.extended_id", "CAN Extended ID", proto_can, FT_UINT32, BASE_DEC);
-
- subdissector_table = register_decode_as_next_proto(proto_can, "can.subdissector", "CAN next level dissector", NULL);
-
- heur_subdissector_list = register_heur_dissector_list("can", proto_can);
-
- static uat_field_t can_interface_mapping_uat_fields[] = {
- UAT_FLD_HEX(interface_configs, interface_id, "Interface ID", "ID of the Interface with 0xffffffff = any (hex uint32 without leading 0x)"),
- UAT_FLD_CSTRING(interface_configs, interface_name, "Interface Name", "Name of the Interface, empty = any (string)"),
- UAT_FLD_HEX(interface_configs, bus_id, "Bus ID", "Bus ID of the Interface (hex uint16 without leading 0x)"),
- UAT_END_FIELDS
- };
-
- can_interface_uat = uat_new("CAN Interface Mapping",
- sizeof(interface_config_t), /* record size */
- DATAFILE_CAN_INTERFACE_MAPPING, /* filename */
- TRUE, /* from profile */
- (void**)&interface_configs, /* data_ptr */
- &interface_config_num, /* numitems_ptr */
- UAT_AFFECTS_DISSECTION, /* but not fields */
- NULL, /* help */
- copy_interface_config_cb, /* copy callback */
- update_interface_config, /* update callback */
- free_interface_config_cb, /* free callback */
- post_update_can_interfaces_cb, /* post update callback */
- NULL, /* reset callback */
- can_interface_mapping_uat_fields /* UAT field definitions */
- );
-
- prefs_register_uat_preference(can_module, "_can_interface_mapping", "Interface Mapping",
- "A table to define the mapping between interface and Bus ID.", can_interface_uat);
+static int
+dissect_socketcan_xl(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void *data _U_) {
+ return dissect_socketcan_common(tvb, pinfo, tree,
+ byte_swap ? ENC_ANTI_HOST_ENDIAN : ENC_HOST_ENDIAN,
+ ENC_HOST_ENDIAN,
+ PACKET_TYPE_CAN_XL);
}
void
-proto_reg_handoff_socketcan(void)
-{
- dissector_handle_t socketcan_bigendian_handle;
+proto_register_socketcan(void) {
+ static hf_register_info hf[] = {
+ { &hf_can_infoent_ext, {
+ "ID", "can.id", FT_UINT32, BASE_DEC_HEX, NULL, CAN_EFF_MASK, NULL, HFILL } },
+ { &hf_can_infoent_std, {
+ "ID", "can.id", FT_UINT32, BASE_DEC_HEX, NULL, CAN_SFF_MASK, NULL, HFILL } },
+ { &hf_can_extflag, {
+ "Extended Flag", "can.flags.xtd", FT_BOOLEAN, 32, NULL, CAN_EFF_FLAG, NULL, HFILL } },
+ { &hf_can_rtrflag, {
+ "Remote Transmission Request Flag", "can.flags.rtr", FT_BOOLEAN, 32, NULL, CAN_RTR_FLAG, NULL, HFILL } },
+ { &hf_can_errflag, {
+ "Error Message Flag", "can.flags.err", FT_BOOLEAN, 32, NULL, CAN_ERR_FLAG, NULL, HFILL } },
+ { &hf_can_len, {
+ "Frame-Length", "can.len", FT_UINT8, BASE_DEC, NULL, 0x0, NULL, HFILL } },
+ { &hf_can_reserved, {
+ "Reserved", "can.reserved", FT_BYTES, BASE_NONE, NULL, 0x0, NULL, HFILL } },
+ { &hf_can_padding, {
+ "Padding", "can.padding", FT_BYTES, BASE_NONE, NULL, 0x0, NULL, HFILL } },
+ { &hf_canfd_brsflag, {
+ "Bit Rate Setting", "canfd.flags.brs", FT_BOOLEAN, 8, NULL, CANFD_BRS, NULL, HFILL } },
+ { &hf_canfd_esiflag, {
+ "Error State Indicator", "canfd.flags.esi", FT_BOOLEAN, 8, NULL, CANFD_ESI, NULL, HFILL } },
+ { &hf_can_err_tx_timeout, {
+ "Transmit timeout", "can.err.tx_timeout", FT_BOOLEAN, 32, NULL, CAN_ERR_TX_TIMEOUT, NULL, HFILL } },
+ { &hf_can_err_lostarb, {
+ "Lost arbitration", "can.err.lostarb", FT_BOOLEAN, 32, NULL, CAN_ERR_LOSTARB, NULL, HFILL } },
+ { &hf_can_err_ctrl, {
+ "Controller problems", "can.err.ctrl", FT_BOOLEAN, 32, NULL, CAN_ERR_CTRL, NULL, HFILL } },
+ { &hf_can_err_prot, {
+ "Protocol violation", "can.err.prot", FT_BOOLEAN, 32, NULL, CAN_ERR_PROT, NULL, HFILL } },
+ { &hf_can_err_trx, {
+ "Transceiver status", "can.err.trx", FT_BOOLEAN, 32, NULL, CAN_ERR_TRX, NULL, HFILL } },
+ { &hf_can_err_ack, {
+ "No acknowledgment", "can.err.ack", FT_BOOLEAN, 32, NULL, CAN_ERR_ACK, NULL, HFILL } },
+ { &hf_can_err_busoff, {
+ "Bus off", "can.err.busoff", FT_BOOLEAN, 32, NULL, CAN_ERR_BUSOFF, NULL, HFILL } },
+ { &hf_can_err_buserror, {
+ "Bus error", "can.err.buserror", FT_BOOLEAN, 32, NULL, CAN_ERR_BUSERROR, NULL, HFILL } },
+ { &hf_can_err_restarted, {
+ "Controller restarted", "can.err.restarted", FT_BOOLEAN, 32, NULL, CAN_ERR_RESTARTED, NULL, HFILL } },
+ { &hf_can_err_reserved, {
+ "Reserved", "can.err.reserved", FT_UINT32, BASE_HEX, NULL, CAN_ERR_RESERVED, NULL, HFILL } },
+ { &hf_can_err_lostarb_bit_number, {
+ "Lost arbitration in bit number", "can.err.lostarb.bitnum", FT_UINT8, BASE_DEC, NULL, 0, NULL, HFILL } },
+ { &hf_can_err_ctrl_rx_overflow, {
+ "RX buffer overflow", "can.err.ctrl.rx_overflow", FT_BOOLEAN, 8, NULL, 0x01, NULL, HFILL } },
+ { &hf_can_err_ctrl_tx_overflow, {
+ "TX buffer overflow", "can.err.ctrl.tx_overflow", FT_BOOLEAN, 8, NULL, 0x02, NULL, HFILL } },
+ { &hf_can_err_ctrl_rx_warning, {
+ "Reached warning level for RX errors", "can.err.ctrl.rx_warning", FT_BOOLEAN, 8, NULL, 0x04, NULL, HFILL } },
+ { &hf_can_err_ctrl_tx_warning, {
+ "Reached warning level for TX errors", "can.err.ctrl.tx_warning", FT_BOOLEAN, 8, NULL, 0x08, NULL, HFILL } },
+ { &hf_can_err_ctrl_rx_passive, {
+ "Reached error passive status RX", "can.err.ctrl.rx_passive", FT_BOOLEAN, 8, NULL, 0x10, NULL, HFILL } },
+ { &hf_can_err_ctrl_tx_passive, {
+ "Reached error passive status TX", "can.err.ctrl.tx_passive", FT_BOOLEAN, 8, NULL, 0x20, NULL, HFILL } },
+ { &hf_can_err_ctrl_active, {
+ "Recovered to error active state", "can.err.ctrl.active", FT_BOOLEAN, 8, NULL, 0x40, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_bit, {
+ "Single bit error", "can.err.prot.type.bit", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_BIT, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_form, {
+ "Frame format error", "can.err.prot.type.form", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_FORM, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_stuff, {
+ "Bit stuffing error", "can.err.prot.type.stuff", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_STUFF, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_bit0, {
+ "Unable to send dominant bit", "can.err.prot.type.bit0", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_BIT0, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_bit1, {
+ "Unable to send recessive bit", "can.err.prot.type.bit1", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_BIT1, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_overload, {
+ "Bus overload", "can.err.prot.type.overload", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_OVERLOAD, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_active, {
+ "Active error announcement", "can.err.prot.type.active", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_ACTIVE, NULL, HFILL } },
+ { &hf_can_err_prot_error_type_tx, {
+ "Error occurred on transmission", "can.err.prot.type.tx", FT_BOOLEAN, 8, NULL, CAN_ERR_PROT_TX, NULL, HFILL } },
+ { &hf_can_err_prot_error_location, {
+ "Protocol error location", "can.err.prot.location", FT_UINT8, BASE_DEC, VALS(can_err_prot_error_location_vals), 0, NULL, HFILL } },
+ { &hf_can_err_trx_canh, {
+ "Transceiver CANH status", "can.err.trx.canh", FT_UINT8, BASE_DEC, VALS(can_err_trx_canh_vals), 0x0F, NULL, HFILL } },
+ { &hf_can_err_trx_canl, {
+ "Transceiver CANL status", "can.err.trx.canl", FT_UINT8, BASE_DEC, VALS(can_err_trx_canl_vals), 0xF0, NULL, HFILL } },
+ { &hf_can_err_ctrl_specific, {
+ "Controller specific data", "can.err.ctrl_specific", FT_BYTES, SEP_SPACE, NULL, 0, NULL, HFILL } },
+ { &hf_canxl_priority, {
+ "Priority", "canxl.priority", FT_UINT32, BASE_DEC, NULL, 0x0000FFFF, NULL, HFILL } },
+ { &hf_canxl_vcid, {
+ "VCID", "canxl.vcid", FT_UINT32, BASE_DEC, NULL, 0x00FF0000, NULL, HFILL } },
+ { &hf_canxl_secflag, {
+ "Simple Extended Context", "canxl.flags.sec", FT_BOOLEAN, 8, NULL, CANXL_SEC, NULL, HFILL } },
+ { &hf_canxl_sdu_type, {
+ "SDU type", "canxl.sdu_type", FT_UINT8, BASE_HEX, VALS(canxl_sdu_type_vals), 0, NULL, HFILL } },
+ { &hf_canxl_len, {
+ "Frame-Length", "canxl.len", FT_UINT16, BASE_DEC, NULL, 0x0, NULL, HFILL } },
+ { &hf_canxl_acceptance_field, {
+ "Acceptance field", "canxl.acceptance_field", FT_UINT32, BASE_DEC_HEX, NULL, 0, NULL, HFILL } },
+ };
+
+ uat_t *can_interface_uat = NULL;
+ uat_t *sender_receiver_uat = NULL;
+
+ /* Setup protocol subtree array */
+ static gint *ett[] = {
+ &ett_can,
+ &ett_can_fd,
+ &ett_can_xl
+ };
+
+ static ei_register_info ei[] = {
+ { &ei_can_err_dlc_mismatch, {
+ "can.err.dlc_mismatch", PI_MALFORMED, PI_ERROR, "ERROR: DLC mismatch", EXPFILL } }
+ };
+
+ module_t *can_module;
+
+ proto_can = proto_register_protocol("Controller Area Network", "CAN", "can");
+
+ /*
+ * "can-hostendian" is a legacy name (there never was, in any libpcap
+ * release, a SocketCAN LINKTYPE_ value for a host-endian CAN ID
+ * and flags field); we need to keep it around in case some candump
+ * or Busmaster capture that was saved as a pcap or pcapng file,
+ * as those use a linktype of LINKTYPE_WIRESHARK_UPPER_PDU with
+ * "can-hostendian" as the dissector name.
+ *
+ * "can-bigendian" is also a legacy name (fpr CAN XL frames, the
+ * fields in the header are in *little-endian* order); we keep it
+ * around for the same reason. It's used for the dissector for
+ * LINKTYPE_CAN_SOCKETCAN.
+ */
+ socketcan_classic_handle = register_dissector("can-hostendian", dissect_socketcan_classic, proto_can);
+ socketcan_bigendian_handle = register_dissector("can-bigendian", dissect_socketcan_bigendian, proto_can);
+
+ proto_canfd = proto_register_protocol("Controller Area Network FD", "CANFD", "canfd");
+ socketcan_fd_handle = register_dissector("canfd", dissect_socketcan_fd, proto_canfd);
+
+ proto_canxl = proto_register_protocol("Controller Area Network XL", "CANXL", "canxl");
+ socketcan_xl_handle = register_dissector("canxl", dissect_socketcan_xl, proto_canxl);
+
+ proto_register_field_array(proto_can, hf, array_length(hf));
+ proto_register_subtree_array(ett, array_length(ett));
+
+ expert_register_field_array(expert_register_protocol(proto_can), ei, array_length(ei));
+
+ can_module = prefs_register_protocol(proto_can, NULL);
+
+ prefs_register_obsolete_preference(can_module, "protocol");
+
+ prefs_register_bool_preference(can_module, "byte_swap", "Byte-swap the CAN ID/flags field",
+ "Whether the CAN ID/flags field should be byte-swapped in CAN classic and CAN FD packets",
+ &byte_swap);
+
+ prefs_register_bool_preference(can_module, "try_heuristic_first", "Try heuristic sub-dissectors first",
+ "Try to decode a packet using an heuristic sub-dissector before using a sub-dissector registered to \"decode as\"",
+ &heuristic_first);
+
+ can_id_dissector_table = register_dissector_table("can.id", "CAN ID", proto_can, FT_UINT32, BASE_DEC);
+
+ can_extended_id_dissector_table = register_dissector_table("can.extended_id", "CAN Extended ID", proto_can, FT_UINT32, BASE_DEC);
+
+ subdissector_table = register_decode_as_next_proto(proto_can, "can.subdissector", "CAN next level dissector", NULL);
+
+ canxl_sdu_type_dissector_table = register_dissector_table("canxl.sdu_type", "CAN XL SDU type", proto_canxl, FT_UINT8, BASE_HEX);
+
+ heur_subdissector_list = register_heur_dissector_list_with_description("can", "CAN heuristic", proto_can);
+
+ static uat_field_t can_interface_mapping_uat_fields[] = {
+ UAT_FLD_HEX(interface_configs, interface_id, "Interface ID", "ID of the Interface with 0xffffffff = any (hex uint32 without leading 0x)"),
+ UAT_FLD_CSTRING(interface_configs, interface_name, "Interface Name", "Name of the Interface, empty = any (string)"),
+ UAT_FLD_HEX(interface_configs, bus_id, "Bus ID", "Bus ID of the Interface (hex uint16 without leading 0x)"),
+ UAT_END_FIELDS
+ };
+
+ can_interface_uat = uat_new("CAN Interface Mapping",
+ sizeof(interface_config_t), /* record size */
+ DATAFILE_CAN_INTERFACE_MAPPING, /* filename */
+ TRUE, /* from profile */
+ (void**)&interface_configs, /* data_ptr */
+ &interface_config_num, /* numitems_ptr */
+ UAT_AFFECTS_DISSECTION, /* but not fields */
+ NULL, /* help */
+ copy_interface_config_cb, /* copy callback */
+ update_interface_config, /* update callback */
+ free_interface_config_cb, /* free callback */
+ post_update_can_interfaces_cb, /* post update callback */
+ NULL, /* reset callback */
+ can_interface_mapping_uat_fields /* UAT field definitions */
+ );
+
+ prefs_register_uat_preference(can_module, "_can_interface_mapping", "Interface Mapping",
+ "A table to define the mapping between interface and Bus ID.", can_interface_uat);
+
+ static uat_field_t sender_receiver_mapping_uat_fields[] = {
+ UAT_FLD_HEX(sender_receiver_configs, bus_id, "Bus ID", "Bus ID of the Interface with 0 meaning any (hex uint16 without leading 0x)."),
+ UAT_FLD_HEX(sender_receiver_configs, can_id, "CAN ID", "ID of the CAN Message (hex uint32 without leading 0x)"),
+ UAT_FLD_CSTRING(sender_receiver_configs, sender_name, "Sender Name", "Name of Sender(s)"),
+ UAT_FLD_CSTRING(sender_receiver_configs, receiver_name, "Receiver Name", "Name of Receiver(s)"),
+ UAT_END_FIELDS
+ };
+
+ sender_receiver_uat = uat_new("Sender Receiver Config",
+ sizeof(sender_receiver_config_t), /* record size */
+ DATAFILE_CAN_SENDER_RECEIVER, /* filename */
+ TRUE, /* from profile */
+ (void**)&sender_receiver_configs, /* data_ptr */
+ &sender_receiver_config_num, /* numitems_ptr */
+ UAT_AFFECTS_DISSECTION, /* but not fields */
+ NULL, /* help */
+ copy_sender_receiver_config_cb, /* copy callback */
+ update_sender_receiver_config, /* update callback */
+ free_sender_receiver_config_cb, /* free callback */
+ post_update_sender_receiver_cb, /* post update callback */
+ NULL, /* reset callback */
+ sender_receiver_mapping_uat_fields /* UAT field definitions */
+ );
+
+ prefs_register_uat_preference(can_module, "_sender_receiver_config", "Sender Receiver Config",
+ "A table to define the mapping between Bus ID and CAN ID to Sender and Receiver.", sender_receiver_uat);
+}
- socketcan_bigendian_handle = create_dissector_handle(dissect_socketcan_bigendian, proto_can);
- dissector_add_uint("wtap_encap", WTAP_ENCAP_SOCKETCAN, socketcan_bigendian_handle);
+void
+proto_reg_handoff_socketcan(void) {
+ dissector_add_uint("wtap_encap", WTAP_ENCAP_SOCKETCAN, socketcan_bigendian_handle);
- dissector_add_uint("sll.ltype", LINUX_SLL_P_CAN, socketcan_classic_handle);
- dissector_add_uint("sll.ltype", LINUX_SLL_P_CANFD, socketcan_fd_handle);
+ dissector_add_uint("sll.ltype", LINUX_SLL_P_CAN, socketcan_classic_handle);
+ dissector_add_uint("sll.ltype", LINUX_SLL_P_CANFD, socketcan_fd_handle);
+ dissector_add_uint("sll.ltype", LINUX_SLL_P_CANXL, socketcan_xl_handle);
}
/*
* Editor modelines - https://www.wireshark.org/tools/modelines.html
*
* Local variables:
- * c-basic-offset: 8
+ * c-basic-offset: 4
* tab-width: 8
- * indent-tabs-mode: t
+ * indent-tabs-mode: nil
* End:
*
- * vi: set shiftwidth=8 tabstop=8 noexpandtab:
- * :indentSize=8:tabSize=8:noTabs=false:
+ * vi: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
*/