| /* |
| * Copyright 2007 Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com> |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public |
| * License along with this library; if not, write to the |
| * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| /** |
| * SECTION:element-pcapparse |
| * |
| * Extracts payloads from Ethernet-encapsulated IP packets. |
| * Use #GstPcapParse:src-ip, #GstPcapParse:dst-ip, |
| * #GstPcapParse:src-port and #GstPcapParse:dst-port to restrict which packets |
| * should be included. |
| * |
| * <refsect2> |
| * <title>Example pipelines</title> |
| * |[ |
| * gst-launch-0.10 filesrc location=h264crasher.pcap ! pcapparse ! rtph264depay |
| * ! ffdec_h264 ! fakesink |
| * ]| Read from a pcap dump file using filesrc, extract the raw UDP packets, |
| * depayload and decode them. |
| * </refsect2> |
| */ |
| |
| /* TODO: |
| * - React on state-change and update state accordingly. |
| * - Implement support for timestamping the buffers. |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include "gstpcapparse.h" |
| |
| #include <string.h> |
| |
| #ifndef G_OS_WIN32 |
| #include <arpa/inet.h> |
| #include <netinet/in.h> |
| #include <string.h> |
| #else |
| #include <winsock2.h> |
| #endif |
| |
| enum |
| { |
| PROP_0, |
| PROP_SRC_IP, |
| PROP_DST_IP, |
| PROP_SRC_PORT, |
| PROP_DST_PORT, |
| PROP_CAPS, |
| PROP_TS_OFFSET |
| }; |
| |
| GST_DEBUG_CATEGORY_STATIC (gst_pcap_parse_debug); |
| #define GST_CAT_DEFAULT gst_pcap_parse_debug |
| |
| static GstStaticPadTemplate sink_template = GST_STATIC_PAD_TEMPLATE ("sink", |
| GST_PAD_SINK, |
| GST_PAD_ALWAYS, |
| GST_STATIC_CAPS ("raw/x-pcap")); |
| |
| static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src", |
| GST_PAD_SRC, |
| GST_PAD_ALWAYS, |
| GST_STATIC_CAPS_ANY); |
| |
| static void gst_pcap_parse_finalize (GObject * object); |
| static void gst_pcap_parse_get_property (GObject * object, guint prop_id, |
| GValue * value, GParamSpec * pspec); |
| static void gst_pcap_parse_set_property (GObject * object, guint prop_id, |
| const GValue * value, GParamSpec * pspec); |
| |
| static void gst_pcap_parse_reset (GstPcapParse * self); |
| |
| static GstFlowReturn gst_pcap_parse_chain (GstPad * pad, |
| GstObject * parent, GstBuffer * buffer); |
| static gboolean gst_pcap_sink_event (GstPad * pad, |
| GstObject * parent, GstEvent * event); |
| |
| #define parent_class gst_pcap_parse_parent_class |
| G_DEFINE_TYPE (GstPcapParse, gst_pcap_parse, GST_TYPE_ELEMENT); |
| |
| static void |
| gst_pcap_parse_class_init (GstPcapParseClass * klass) |
| { |
| GObjectClass *gobject_class = G_OBJECT_CLASS (klass); |
| GstElementClass *element_class = GST_ELEMENT_CLASS (klass); |
| |
| gobject_class->finalize = gst_pcap_parse_finalize; |
| gobject_class->get_property = gst_pcap_parse_get_property; |
| gobject_class->set_property = gst_pcap_parse_set_property; |
| |
| g_object_class_install_property (gobject_class, |
| PROP_SRC_IP, g_param_spec_string ("src-ip", "Source IP", |
| "Source IP to restrict to", "", |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, |
| PROP_DST_IP, g_param_spec_string ("dst-ip", "Destination IP", |
| "Destination IP to restrict to", "", |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, |
| PROP_SRC_PORT, g_param_spec_int ("src-port", "Source port", |
| "Source port to restrict to", -1, G_MAXUINT16, -1, |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, |
| PROP_DST_PORT, g_param_spec_int ("dst-port", "Destination port", |
| "Destination port to restrict to", -1, G_MAXUINT16, -1, |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, PROP_CAPS, |
| g_param_spec_boxed ("caps", "Caps", |
| "The caps of the source pad", GST_TYPE_CAPS, |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, PROP_TS_OFFSET, |
| g_param_spec_int64 ("ts-offset", "Timestamp Offset", |
| "Relative timestamp offset (ns) to apply (-1 = use absolute packet time)", |
| -1, G_MAXINT64, -1, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| gst_element_class_add_pad_template (element_class, |
| gst_static_pad_template_get (&sink_template)); |
| gst_element_class_add_pad_template (element_class, |
| gst_static_pad_template_get (&src_template)); |
| |
| gst_element_class_set_static_metadata (element_class, "PCapParse", |
| "Raw/Parser", |
| "Parses a raw pcap stream", |
| "Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com>"); |
| |
| GST_DEBUG_CATEGORY_INIT (gst_pcap_parse_debug, "pcapparse", 0, "pcap parser"); |
| } |
| |
| static void |
| gst_pcap_parse_init (GstPcapParse * self) |
| { |
| self->sink_pad = gst_pad_new_from_static_template (&sink_template, "sink"); |
| gst_pad_set_chain_function (self->sink_pad, |
| GST_DEBUG_FUNCPTR (gst_pcap_parse_chain)); |
| gst_pad_use_fixed_caps (self->sink_pad); |
| gst_pad_set_event_function (self->sink_pad, |
| GST_DEBUG_FUNCPTR (gst_pcap_sink_event)); |
| gst_element_add_pad (GST_ELEMENT (self), self->sink_pad); |
| |
| self->src_pad = gst_pad_new_from_static_template (&src_template, "src"); |
| gst_pad_use_fixed_caps (self->src_pad); |
| gst_element_add_pad (GST_ELEMENT (self), self->src_pad); |
| |
| self->src_ip = -1; |
| self->dst_ip = -1; |
| self->src_port = -1; |
| self->dst_port = -1; |
| self->offset = -1; |
| |
| self->adapter = gst_adapter_new (); |
| |
| gst_pcap_parse_reset (self); |
| } |
| |
| static void |
| gst_pcap_parse_finalize (GObject * object) |
| { |
| GstPcapParse *self = GST_PCAP_PARSE (object); |
| |
| g_object_unref (self->adapter); |
| if (self->caps) |
| gst_caps_unref (self->caps); |
| |
| G_OBJECT_CLASS (parent_class)->finalize (object); |
| } |
| |
| static const gchar * |
| get_ip_address_as_string (gint64 ip_addr) |
| { |
| if (ip_addr >= 0) { |
| struct in_addr addr; |
| addr.s_addr = ip_addr; |
| return inet_ntoa (addr); |
| } else { |
| return ""; |
| } |
| } |
| |
| static void |
| set_ip_address_from_string (gint64 * ip_addr, const gchar * ip_str) |
| { |
| if (ip_str[0] != '\0') { |
| gulong addr = inet_addr (ip_str); |
| if (addr != INADDR_NONE) |
| *ip_addr = addr; |
| } else { |
| *ip_addr = -1; |
| } |
| } |
| |
| static void |
| gst_pcap_parse_get_property (GObject * object, guint prop_id, |
| GValue * value, GParamSpec * pspec) |
| { |
| GstPcapParse *self = GST_PCAP_PARSE (object); |
| |
| switch (prop_id) { |
| case PROP_SRC_IP: |
| g_value_set_string (value, get_ip_address_as_string (self->src_ip)); |
| break; |
| |
| case PROP_DST_IP: |
| g_value_set_string (value, get_ip_address_as_string (self->dst_ip)); |
| break; |
| |
| case PROP_SRC_PORT: |
| g_value_set_int (value, self->src_port); |
| break; |
| |
| case PROP_DST_PORT: |
| g_value_set_int (value, self->dst_port); |
| break; |
| |
| case PROP_CAPS: |
| gst_value_set_caps (value, self->caps); |
| break; |
| |
| case PROP_TS_OFFSET: |
| g_value_set_int64 (value, self->offset); |
| break; |
| |
| default: |
| G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); |
| break; |
| } |
| } |
| |
| static void |
| gst_pcap_parse_set_property (GObject * object, guint prop_id, |
| const GValue * value, GParamSpec * pspec) |
| { |
| GstPcapParse *self = GST_PCAP_PARSE (object); |
| |
| switch (prop_id) { |
| case PROP_SRC_IP: |
| set_ip_address_from_string (&self->src_ip, g_value_get_string (value)); |
| break; |
| |
| case PROP_DST_IP: |
| set_ip_address_from_string (&self->dst_ip, g_value_get_string (value)); |
| break; |
| |
| case PROP_SRC_PORT: |
| self->src_port = g_value_get_int (value); |
| break; |
| |
| case PROP_DST_PORT: |
| self->dst_port = g_value_get_int (value); |
| break; |
| |
| case PROP_CAPS: |
| { |
| const GstCaps *new_caps_val; |
| GstCaps *new_caps, *old_caps; |
| |
| new_caps_val = gst_value_get_caps (value); |
| if (new_caps_val == NULL) { |
| new_caps = gst_caps_new_any (); |
| } else { |
| new_caps = gst_caps_copy (new_caps_val); |
| } |
| |
| old_caps = self->caps; |
| self->caps = new_caps; |
| if (old_caps) |
| gst_caps_unref (old_caps); |
| |
| gst_pad_set_caps (self->src_pad, new_caps); |
| break; |
| } |
| |
| case PROP_TS_OFFSET: |
| self->offset = g_value_get_int64 (value); |
| break; |
| |
| default: |
| G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); |
| break; |
| } |
| } |
| |
| static void |
| gst_pcap_parse_reset (GstPcapParse * self) |
| { |
| self->initialized = FALSE; |
| self->swap_endian = FALSE; |
| self->cur_packet_size = -1; |
| self->cur_ts = GST_CLOCK_TIME_NONE; |
| self->base_ts = GST_CLOCK_TIME_NONE; |
| self->newsegment_sent = FALSE; |
| |
| gst_adapter_clear (self->adapter); |
| } |
| |
| static guint32 |
| gst_pcap_parse_read_uint32 (GstPcapParse * self, const guint8 * p) |
| { |
| guint32 val = *((guint32 *) p); |
| |
| if (self->swap_endian) { |
| #if G_BYTE_ORDER == G_LITTLE_ENDIAN |
| return GUINT32_FROM_BE (val); |
| #else |
| return GUINT32_FROM_LE (val); |
| #endif |
| } else { |
| return val; |
| } |
| } |
| |
| #define ETH_HEADER_LEN 14 |
| #define SLL_HEADER_LEN 16 |
| #define IP_HEADER_MIN_LEN 20 |
| #define UDP_HEADER_LEN 8 |
| |
| #define IP_PROTO_UDP 17 |
| #define IP_PROTO_TCP 6 |
| |
| |
| static gboolean |
| gst_pcap_parse_scan_frame (GstPcapParse * self, |
| const guint8 * buf, |
| gint buf_size, const guint8 ** payload, gint * payload_size) |
| { |
| const guint8 *buf_ip = 0; |
| const guint8 *buf_proto; |
| guint16 eth_type; |
| guint8 b; |
| guint8 ip_header_size; |
| guint8 ip_protocol; |
| guint32 ip_src_addr; |
| guint32 ip_dst_addr; |
| guint16 src_port; |
| guint16 dst_port; |
| guint16 len; |
| |
| switch (self->linktype) { |
| case LINKTYPE_ETHER: |
| if (buf_size < ETH_HEADER_LEN + IP_HEADER_MIN_LEN + UDP_HEADER_LEN) |
| return FALSE; |
| |
| eth_type = GUINT16_FROM_BE (*((guint16 *) (buf + 12))); |
| buf_ip = buf + ETH_HEADER_LEN; |
| break; |
| case LINKTYPE_SLL: |
| if (buf_size < SLL_HEADER_LEN + IP_HEADER_MIN_LEN + UDP_HEADER_LEN) |
| return FALSE; |
| |
| eth_type = GUINT16_FROM_BE (*((guint16 *) (buf + 14))); |
| buf_ip = buf + SLL_HEADER_LEN; |
| break; |
| case LINKTYPE_RAW: |
| if (buf_size < IP_HEADER_MIN_LEN + UDP_HEADER_LEN) |
| return FALSE; |
| |
| eth_type = 0x800; /* This is fine since IPv4/IPv6 is parse elsewhere */ |
| buf_ip = buf; |
| break; |
| |
| default: |
| return FALSE; |
| } |
| |
| if (eth_type != 0x800) |
| return FALSE; |
| |
| b = *buf_ip; |
| if (((b >> 4) & 0x0f) != 4) |
| return FALSE; |
| |
| ip_header_size = (b & 0x0f) * 4; |
| if (buf_ip + ip_header_size > buf + buf_size) |
| return FALSE; |
| |
| ip_protocol = *(buf_ip + 9); |
| GST_LOG_OBJECT (self, "ip proto %d", (gint) ip_protocol); |
| |
| if (ip_protocol != IP_PROTO_UDP && ip_protocol != IP_PROTO_TCP) |
| return FALSE; |
| |
| /* ip info */ |
| ip_src_addr = *((guint32 *) (buf_ip + 12)); |
| ip_dst_addr = *((guint32 *) (buf_ip + 16)); |
| buf_proto = buf_ip + ip_header_size; |
| |
| /* ok for tcp and udp */ |
| src_port = GUINT16_FROM_BE (*((guint16 *) (buf_proto + 0))); |
| dst_port = GUINT16_FROM_BE (*((guint16 *) (buf_proto + 2))); |
| |
| /* extract some params and data according to protocol */ |
| if (ip_protocol == IP_PROTO_UDP) { |
| len = GUINT16_FROM_BE (*((guint16 *) (buf_proto + 4))); |
| if (len < UDP_HEADER_LEN || buf_proto + len > buf + buf_size) |
| return FALSE; |
| |
| *payload = buf_proto + UDP_HEADER_LEN; |
| *payload_size = len - UDP_HEADER_LEN; |
| } else { |
| if (buf_proto + 12 >= buf + buf_size) |
| return FALSE; |
| len = (buf_proto[12] >> 4) * 4; |
| if (buf_proto + len > buf + buf_size) |
| return FALSE; |
| |
| /* all remaining data following tcp header is payload */ |
| *payload = buf_proto + len; |
| *payload_size = self->cur_packet_size - (buf_proto - buf) - len; |
| } |
| |
| /* but still filter as configured */ |
| if (self->src_ip >= 0 && ip_src_addr != self->src_ip) |
| return FALSE; |
| |
| if (self->dst_ip >= 0 && ip_dst_addr != self->dst_ip) |
| return FALSE; |
| |
| if (self->src_port >= 0 && src_port != self->src_port) |
| return FALSE; |
| |
| if (self->dst_port >= 0 && dst_port != self->dst_port) |
| return FALSE; |
| |
| return TRUE; |
| } |
| |
| static GstFlowReturn |
| gst_pcap_parse_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer) |
| { |
| GstPcapParse *self = GST_PCAP_PARSE (parent); |
| GstFlowReturn ret = GST_FLOW_OK; |
| GstBufferList *list = NULL; |
| |
| gst_adapter_push (self->adapter, buffer); |
| |
| while (ret == GST_FLOW_OK) { |
| gint avail; |
| const guint8 *data; |
| |
| avail = gst_adapter_available (self->adapter); |
| |
| if (self->initialized) { |
| if (self->cur_packet_size >= 0) { |
| if (avail < self->cur_packet_size) |
| break; |
| |
| if (self->cur_packet_size > 0) { |
| const guint8 *payload_data; |
| gint payload_size; |
| |
| data = gst_adapter_map (self->adapter, self->cur_packet_size); |
| |
| GST_LOG_OBJECT (self, "examining packet size %" G_GINT64_FORMAT, |
| self->cur_packet_size); |
| |
| if (gst_pcap_parse_scan_frame (self, data, self->cur_packet_size, |
| &payload_data, &payload_size)) { |
| GstBuffer *out_buf; |
| guintptr offset = payload_data - data; |
| |
| self->cur_packet_size -= offset; |
| self->cur_packet_size -= payload_size; |
| |
| gst_adapter_unmap (self->adapter); |
| gst_adapter_flush (self->adapter, offset); |
| out_buf = gst_adapter_take_buffer_fast (self->adapter, |
| payload_size); |
| |
| if (GST_CLOCK_TIME_IS_VALID (self->cur_ts)) { |
| if (!GST_CLOCK_TIME_IS_VALID (self->base_ts)) |
| self->base_ts = self->cur_ts; |
| if (self->offset >= 0) { |
| self->cur_ts -= self->base_ts; |
| self->cur_ts += self->offset; |
| } |
| } |
| GST_BUFFER_TIMESTAMP (out_buf) = self->cur_ts; |
| |
| |
| if (list == NULL) |
| list = gst_buffer_list_new (); |
| gst_buffer_list_add (list, out_buf); |
| } else { |
| gst_adapter_unmap (self->adapter); |
| gst_adapter_flush (self->adapter, self->cur_packet_size); |
| } |
| } |
| |
| self->cur_packet_size = -1; |
| } else { |
| guint32 ts_sec; |
| guint32 ts_usec; |
| guint32 incl_len; |
| |
| if (avail < 16) |
| break; |
| |
| data = gst_adapter_map (self->adapter, 16); |
| |
| ts_sec = gst_pcap_parse_read_uint32 (self, data + 0); |
| ts_usec = gst_pcap_parse_read_uint32 (self, data + 4); |
| incl_len = gst_pcap_parse_read_uint32 (self, data + 8); |
| /* orig_len = gst_pcap_parse_read_uint32 (self, data + 12); */ |
| |
| gst_adapter_unmap (self->adapter); |
| gst_adapter_flush (self->adapter, 16); |
| |
| self->cur_ts = ts_sec * GST_SECOND + ts_usec * GST_USECOND; |
| self->cur_packet_size = incl_len; |
| } |
| } else { |
| guint32 magic; |
| guint32 linktype; |
| guint16 major_version; |
| |
| if (avail < 24) |
| break; |
| |
| data = gst_adapter_map (self->adapter, 24); |
| |
| magic = *((guint32 *) data); |
| major_version = *((guint16 *) (data + 4)); |
| linktype = gst_pcap_parse_read_uint32 (self, data + 20); |
| gst_adapter_unmap (self->adapter); |
| |
| if (magic == 0xa1b2c3d4) { |
| self->swap_endian = FALSE; |
| } else if (magic == 0xd4c3b2a1) { |
| self->swap_endian = TRUE; |
| major_version = major_version << 8 | major_version >> 8; |
| } else { |
| GST_ELEMENT_ERROR (self, STREAM, WRONG_TYPE, (NULL), |
| ("File is not a libpcap file, magic is %X", magic)); |
| ret = GST_FLOW_ERROR; |
| goto out; |
| } |
| |
| if (major_version != 2) { |
| GST_ELEMENT_ERROR (self, STREAM, WRONG_TYPE, (NULL), |
| ("File is not a libpcap major version 2, but %u", major_version)); |
| ret = GST_FLOW_ERROR; |
| goto out; |
| } |
| |
| if (linktype != LINKTYPE_ETHER && linktype != LINKTYPE_SLL && |
| linktype != LINKTYPE_RAW) { |
| GST_ELEMENT_ERROR (self, STREAM, WRONG_TYPE, (NULL), |
| ("Only dumps of type Ethernet, raw IP or Linux Cooked (SLL) " |
| "understood; type %d unknown", linktype)); |
| ret = GST_FLOW_ERROR; |
| goto out; |
| } |
| |
| GST_DEBUG_OBJECT (self, "linktype %u", linktype); |
| self->linktype = linktype; |
| |
| gst_adapter_flush (self->adapter, 24); |
| self->initialized = TRUE; |
| } |
| } |
| |
| if (list) { |
| if (!self->newsegment_sent && GST_CLOCK_TIME_IS_VALID (self->cur_ts)) { |
| GstSegment segment; |
| |
| if (self->caps) |
| gst_pad_set_caps (self->src_pad, self->caps); |
| gst_segment_init (&segment, GST_FORMAT_TIME); |
| segment.start = self->cur_ts; |
| gst_pad_push_event (self->src_pad, gst_event_new_segment (&segment)); |
| self->newsegment_sent = TRUE; |
| } |
| |
| ret = gst_pad_push_list (self->src_pad, list); |
| list = NULL; |
| } |
| |
| out: |
| |
| if (list) |
| gst_buffer_list_unref (list); |
| |
| if (ret != GST_FLOW_OK) |
| gst_pcap_parse_reset (self); |
| |
| return ret; |
| } |
| |
| static gboolean |
| gst_pcap_sink_event (GstPad * pad, GstObject * parent, GstEvent * event) |
| { |
| gboolean ret = TRUE; |
| GstPcapParse *self = GST_PCAP_PARSE (parent); |
| |
| switch (GST_EVENT_TYPE (event)) { |
| case GST_EVENT_SEGMENT: |
| /* Drop it, we'll replace it with our own */ |
| gst_event_unref (event); |
| break; |
| default: |
| ret = gst_pad_push_event (self->src_pad, event); |
| break; |
| } |
| |
| return ret; |
| } |