/**************************************************************************** * net/utils/net_snoop.c * * SPDX-License-Identifier: Apache-2.0 * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define SNOOP_VERSION_1 1 #define SNOOP_VERSION_2 2 /* microseconds since midnight, January 1st, 0 AD nominal Gregorian. */ #define SNOOP_EPOCH_USEC(tv) (((tv).tv_sec - 0x386d4380ll) * 1000000ll \ + (tv).tv_usec + 0x00e03ab44a676000ll) /**************************************************************************** * Private Type Definitions ****************************************************************************/ /* The availability of tools to capture, display and interpret packets * traversing a network has proven extremely useful in debugging * networking problems. The ability to capture packets and store them * for later analysis allows one to de-couple the tasks of collecting * information about a network problem and analysing that information. * * More info about snoop datalink type, please refer to * https://www.rfc-editor.org/rfc/rfc1761.txt and * https://fte.com/webhelpii/hsu/Content/Technical_Information/ * BT_Snoop_File_Format.htm */ /* The snoop packet capture file is an array of octets structured as * follows: * * +------------------------+ * | | * | File Header | * | | * +------------------------+ * | | * | Packet Record | * ~ Number 1 ~ * | | * +------------------------+ * . . * . . * . . * +------------------------+ * | | * | Packet Record | * ~ Number N ~ * | | * +------------------------+ */ /* snoop_file_header_s * * The structure of the File Header is as follows: * * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | | * + Identification Pattern + * | | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Version Number | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Datalink Type | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ */ begin_packed_struct struct snoop_file_header_s { uint8_t magic[8]; /* Identification Pattern */ uint32_t version; /* Version Number */ uint32_t datalink; /* Datalink Type */ } end_packed_struct; /* snoop_packet_header_s * * The structure of the packet record is as follows: * * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Original Length | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Included Length | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Packet Record Length | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Cumulative Drops | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Timestamp Seconds | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | Timestamp Microseconds | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ * | | * . . * . Packet Data . * . . * + +- - - - - - - -+ * | | Pad | * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ */ begin_packed_struct struct snoop_packet_header_s { uint32_t orig_len; /* actual length of packet */ uint32_t incl_len; /* number of octets captured in file */ union { uint32_t flags; /* Packet Flags: 1 hci cmd , eg: btsnoop */ uint32_t rec_len; /* length of record */ }; uint32_t cum_drops; /* cumulative number of dropped packets */ union { uint64_t ts_usec; /* timestamp microseconds, eg: btsnoop */ struct { uint32_t ts_sec; /* timestamp seconds */ uint32_t ts_usec; /* timestamp microseconds */ } ts; }; } end_packed_struct; /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: snoop_fill_packet_header * * Description: * This function fill snoop packet header info. * ****************************************************************************/ static void snoop_fill_packet_header(FAR struct snoop_s *snoop, uint32_t bytes, uint32_t drops, uint32_t flags, FAR struct snoop_packet_header_s *header) { struct timeval tv; switch (snoop->datalink) { case SNOOP_DATALINK_HCI_UNENCAP: case SNOOP_DATALINK_HCI_UART: case SNOOP_DATALINK_HCI_BSCP: case SNOOP_DATALINK_HCI_SERIAL: gettimeofday(&tv, NULL); header->ts_usec = htobe64(SNOOP_EPOCH_USEC(tv)); header->flags = htobe32(flags); break; case SNOOP_DATALINK_TYPE_TOKENBUS: case SNOOP_DATALINK_TYPE_TOKERING: case SNOOP_DATALINK_TYPE_METRONET: case SNOOP_DATALINK_TYPE_ETHERNET: case SNOOP_DATALINK_TYPE_HDLC: case SNOOP_DATALINK_TYPE_CHARSYNC: case SNOOP_DATALINK_TYPE_IBMC2C: case SNOOP_DATALINK_TYPE_FDDI: case SNOOP_DATALINK_TYPE_OTHER: gettimeofday(&tv, NULL); header->ts.ts_sec = htobe32(tv.tv_sec); header->ts.ts_usec = htobe32(tv.tv_usec); header->rec_len = htobe32(flags); break; default: DEBUGASSERT(false); } header->orig_len = htobe32(bytes); header->incl_len = htobe32(bytes); header->cum_drops = htobe32(drops); } /**************************************************************************** * Name: snoop_flush * * Description: * This function could flush snoop buf into file. * ****************************************************************************/ static int snoop_flush(FAR struct snoop_s *snoop) { ssize_t ret; if (snoop->next == 0) { return 0; } do { ret = file_write(&snoop->filep, snoop->buf, snoop->next); if (ret < 0) { break; } snoop->next -= ret; memmove(snoop->buf, snoop->buf + ret, snoop->next); } while (snoop->next > 0); #ifndef CONFIG_DISABLE_MOUNTPOINT if (snoop->autosync) { ret = file_fsync(&snoop->filep); } #endif return ret; } /**************************************************************************** * Name: snoop_flush_lock * * Description: * Snoop flush atomic * ****************************************************************************/ static int snoop_flush_lock(FAR struct snoop_s *snoop) { irqstate_t flags; int ret; flags = enter_critical_section(); nxmutex_lock(&snoop->mutex); ret = snoop_flush(snoop); nxmutex_unlock(&snoop->mutex); leave_critical_section(flags); return ret; } /**************************************************************************** * Name: snoop_flush_work * * Description: * Do snoop flush work. * ****************************************************************************/ static void snoop_flush_work(FAR void *arg) { snoop_flush_lock((FAR struct snoop_s *)arg); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: snoop_open * * Description: * This function open snoop file by datalink. * * Input Parameters: * snoop The snoop driver struct * filename Snoop file name * datalink Snoop datalink type, such as SNOOP_DATALINK_TYPE_XX * autosync whether do file_sync when snoop_dump * * Returned Value: * OK on success; Negated errno on failure. * * Assumptions: * ****************************************************************************/ int snoop_open(FAR struct snoop_s *snoop, FAR const char *filename, uint32_t datalink, bool autosync) { struct snoop_file_header_s header; int ret; if (!snoop) { return -EINVAL; } switch (datalink) { case SNOOP_DATALINK_TYPE_TOKENBUS: case SNOOP_DATALINK_TYPE_TOKERING: case SNOOP_DATALINK_TYPE_METRONET: case SNOOP_DATALINK_TYPE_ETHERNET: case SNOOP_DATALINK_TYPE_HDLC: case SNOOP_DATALINK_TYPE_CHARSYNC: case SNOOP_DATALINK_TYPE_IBMC2C: case SNOOP_DATALINK_TYPE_FDDI: case SNOOP_DATALINK_TYPE_OTHER: { static const uint8_t snoop_magic[] = { 's', 'n', 'o', 'o', 'p', '\0', '\0', '\0' }; memcpy(header.magic, snoop_magic, nitems(snoop_magic)); header.version = htobe32(SNOOP_VERSION_2); break; }; case SNOOP_DATALINK_HCI_UNENCAP: case SNOOP_DATALINK_HCI_UART: case SNOOP_DATALINK_HCI_BSCP: case SNOOP_DATALINK_HCI_SERIAL: { static const uint8_t btsnoop_magic[] = { 'b', 't', 's', 'n', 'o', 'o', 'p', '\0' }; memcpy(header.magic, btsnoop_magic, nitems(btsnoop_magic)); header.version = htobe32(SNOOP_VERSION_1); break; } default: { return -EINVAL; } } ret = file_open(&snoop->filep, filename, O_RDWR | O_CREAT | O_CLOEXEC, 0666); if (ret < 0) { return ret; } snoop->datalink = datalink; snoop->autosync = autosync; snoop->next = 0; header.datalink = htobe32(datalink); ret = file_write(&snoop->filep, &header, sizeof(header)); if (ret != sizeof(header)) { ret = ret < 0 ? ret : -EINVAL; goto error; } nxmutex_init(&snoop->mutex); return OK; error: snoop_close(snoop); return ret; } /**************************************************************************** * Name: snoop_dump * * Description: * This function dump nbytes buf data into snoop file. * * Input Parameters: * snoop The snoop driver struct * buf Snoop buffer * nbytes Snoop buffer size * drops cumulative number of dropped packets * flags Packet Flags: 1 hci cmd , eg: btsnoop * * Returned Value: * OK on success; Negated errno on failure. * * Assumptions: * ****************************************************************************/ int snoop_dump(FAR struct snoop_s *snoop, FAR const void *buf, uint32_t nbytes, uint32_t drops, uint32_t flags) { struct snoop_packet_header_s header; irqstate_t irqflags; int ret = 0; if (!snoop) { return -EINVAL; } snoop_fill_packet_header(snoop, nbytes, drops, flags, &header); irqflags = enter_critical_section(); if (up_interrupt_context()) { if (sizeof(snoop->buf) - snoop->next < nbytes + sizeof(struct snoop_packet_header_s)) { ret = -ENOMEM; goto out_leave; } memcpy(snoop->buf + snoop->next, &header, sizeof(header)); snoop->next += sizeof(header); memcpy(snoop->buf + snoop->next, buf, nbytes); snoop->next += nbytes; if (work_available(&snoop->work)) { work_queue(HPWORK, &snoop->work, snoop_flush_work, snoop, 0); } goto out_leave; } else { nxmutex_lock(&snoop->mutex); ret = snoop_flush(snoop); if (ret < 0) { goto out_unlock; } ret = file_write(&snoop->filep, &header, sizeof(header)); if (ret < 0) { goto out_unlock; } else if (ret != sizeof(header)) { ret = -EINVAL; goto out_unlock; } ret = file_write(&snoop->filep, buf, nbytes); if (ret < 0) { goto out_unlock; } else if (ret != nbytes) { ret = -EINVAL; goto out_unlock; } } out_unlock: nxmutex_unlock(&snoop->mutex); out_leave: leave_critical_section(irqflags); return ret; } /**************************************************************************** * Name: snoop_sync * * Description: * This function sync snoop buffer. * * Input Parameters: * snoop The snoop driver struct * * Returned Value: * OK on success; Negated errno on failure. * * Assumptions: * ****************************************************************************/ int snoop_sync(FAR struct snoop_s *snoop) { if (!snoop) { return -EINVAL; } return snoop_flush_lock(snoop); } /**************************************************************************** * Name: snoop_close * * Description: * This function close snoop file. * * Input Parameters: * snoop The snoop driver struct * * Returned Value: * OK on success; Negated errno on failure. * * Assumptions: * ****************************************************************************/ int snoop_close(FAR struct snoop_s *snoop) { if (!snoop) { return -EINVAL; } nxmutex_destroy(&snoop->mutex); return file_close(&snoop->filep); }