2022-09-20 09:57:45 +02:00
|
|
|
#!/usr/bin/env python3
|
2017-11-23 10:44:32 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2022-09-20 10:01:32 +02:00
|
|
|
"""
|
2017-11-23 10:44:32 +01:00
|
|
|
(C) 2012, Mariano Alvira <mar@devl.org>
|
|
|
|
(C) 2014, Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
(C) 2015, Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
(C) 2015, Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
(C) 2015, Cenk Gündoğan <cnkgndgn@gmail.com>
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
3. Neither the name of the Institute nor the names of its contributors
|
|
|
|
may be used to endorse or promote products derived from this software
|
|
|
|
without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
|
|
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
SUCH DAMAGE.
|
2022-09-20 10:01:32 +02:00
|
|
|
"""
|
2017-11-23 10:44:32 +01:00
|
|
|
|
|
|
|
from __future__ import print_function
|
2018-06-19 11:42:18 +02:00
|
|
|
import argparse
|
2017-11-23 10:44:32 +01:00
|
|
|
import sys
|
|
|
|
import re
|
|
|
|
import socket
|
|
|
|
from time import sleep, time
|
|
|
|
from struct import pack
|
|
|
|
from serial import Serial
|
|
|
|
|
|
|
|
# PCAP setup
|
2022-09-20 10:01:32 +02:00
|
|
|
MAGIC = 0xA1B2C3D4
|
2017-11-23 10:44:32 +01:00
|
|
|
MAJOR = 2
|
|
|
|
MINOR = 4
|
|
|
|
ZONE = 0
|
|
|
|
SIG = 0
|
2022-09-20 10:01:32 +02:00
|
|
|
SNAPLEN = 0xFFFF
|
|
|
|
NETWORK = 230 # 802.15.4 no FCS
|
2017-11-23 10:44:32 +01:00
|
|
|
|
2018-09-26 20:30:54 +02:00
|
|
|
DEFAULT_BAUDRATE = 115200
|
|
|
|
|
2017-11-23 10:44:32 +01:00
|
|
|
|
|
|
|
def configure_interface(port, channel):
|
|
|
|
line = ""
|
|
|
|
iface = 0
|
2022-09-20 10:01:32 +02:00
|
|
|
port.write("ifconfig\n".encode())
|
2017-11-23 10:44:32 +01:00
|
|
|
while True:
|
|
|
|
line = port.readline()
|
2022-09-20 10:01:32 +02:00
|
|
|
if line == b"":
|
|
|
|
print("Application has no network interface defined", file=sys.stderr)
|
2017-11-23 10:44:32 +01:00
|
|
|
sys.exit(2)
|
2022-09-20 10:01:32 +02:00
|
|
|
match = re.search(r"^Iface +(\d+)", line.decode(errors="ignore"))
|
2017-11-23 10:44:32 +01:00
|
|
|
if match is not None:
|
|
|
|
iface = int(match.group(1))
|
|
|
|
break
|
|
|
|
|
|
|
|
# set channel, raw mode, and promiscuous mode
|
2022-09-20 10:01:32 +02:00
|
|
|
print("ifconfig %d set chan %d" % (iface, channel), file=sys.stderr)
|
|
|
|
print("ifconfig %d raw" % iface, file=sys.stderr)
|
|
|
|
print("ifconfig %d promisc" % iface, file=sys.stderr)
|
|
|
|
port.write(("ifconfig %d set chan %d\n" % (iface, channel)).encode())
|
|
|
|
port.write(("ifconfig %d raw\n" % iface).encode())
|
|
|
|
port.write(("ifconfig %d promisc\n" % iface).encode())
|
2017-11-23 10:44:32 +01:00
|
|
|
|
|
|
|
|
|
|
|
def generate_pcap(port, out):
|
|
|
|
# count incoming packets
|
|
|
|
count = 0
|
|
|
|
# output overall PCAP header
|
2022-09-20 10:01:32 +02:00
|
|
|
out.write(pack("<LHHLLLL", MAGIC, MAJOR, MINOR, ZONE, SIG, SNAPLEN, NETWORK))
|
2017-11-23 10:44:32 +01:00
|
|
|
sys.stderr.write("RX: %i\r" % count)
|
|
|
|
while True:
|
|
|
|
line = port.readline().rstrip()
|
|
|
|
|
2022-09-20 10:01:32 +02:00
|
|
|
pkt_header = re.match(
|
|
|
|
r">? *rftest-rx --- len (\w+).*", line.decode(errors="ignore")
|
|
|
|
)
|
2017-11-23 10:44:32 +01:00
|
|
|
if pkt_header:
|
|
|
|
now = time()
|
|
|
|
sec = int(now)
|
|
|
|
usec = int((now - sec) * 1000000)
|
|
|
|
length = int(pkt_header.group(1), 16)
|
2022-09-20 10:01:32 +02:00
|
|
|
out.write(pack("<LLLL", sec, usec, length, length))
|
2017-11-23 10:44:32 +01:00
|
|
|
out.flush()
|
|
|
|
count += 1
|
|
|
|
sys.stderr.write("RX: %i\r" % count)
|
|
|
|
continue
|
|
|
|
|
2018-06-19 10:04:02 +02:00
|
|
|
pkt_data = re.match(r"(\w\w )+", line.decode(errors="ignore"))
|
2017-11-23 10:44:32 +01:00
|
|
|
if pkt_data:
|
2022-09-20 10:01:32 +02:00
|
|
|
for part in line.decode(errors="ignore").split(" "):
|
2018-06-19 10:04:02 +02:00
|
|
|
byte = re.match(r"(\w\w)", part)
|
2017-11-23 10:44:32 +01:00
|
|
|
if byte:
|
2022-09-20 10:01:32 +02:00
|
|
|
out.write(pack("<B", int(byte.group(1), 16)))
|
2017-11-23 10:44:32 +01:00
|
|
|
out.flush()
|
|
|
|
|
|
|
|
|
2018-06-19 11:42:18 +02:00
|
|
|
def connect(args):
|
2017-11-23 10:44:32 +01:00
|
|
|
conn = None
|
2018-06-19 13:42:15 +02:00
|
|
|
if args.conn.startswith("/dev/tty") or args.conn.startswith("COM"):
|
2017-11-23 10:44:32 +01:00
|
|
|
# open serial port
|
|
|
|
try:
|
2022-09-20 10:01:32 +02:00
|
|
|
conn = Serial(args.conn, args.baudrate, dsrdtr=0, rtscts=0, timeout=1)
|
2017-11-23 10:44:32 +01:00
|
|
|
except IOError:
|
2018-06-19 13:42:15 +02:00
|
|
|
print("error opening serial port %s" % args.conn, file=sys.stderr)
|
|
|
|
sys.exit(2)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
port = args.conn.split(":")[-1]
|
2022-09-20 10:01:32 +02:00
|
|
|
host = args.conn[: -(len(port) + 1)]
|
2018-06-19 13:42:15 +02:00
|
|
|
port = int(port)
|
|
|
|
except (IndexError, ValueError):
|
|
|
|
print("Can't parse host:port pair %s" % args.conn, file=sys.stderr)
|
2017-11-23 10:44:32 +01:00
|
|
|
sys.exit(2)
|
|
|
|
try:
|
|
|
|
sock = socket.socket()
|
|
|
|
sock.connect((host, port))
|
|
|
|
conn = sock.makefile("r+b", bufsize=0)
|
|
|
|
except IOError:
|
|
|
|
print("error connecting to %s:%s" % (host, port), file=sys.stderr)
|
|
|
|
sys.exit(2)
|
|
|
|
return conn
|
|
|
|
|
2018-06-19 13:42:15 +02:00
|
|
|
|
2018-06-19 11:42:18 +02:00
|
|
|
def main():
|
|
|
|
if sys.version_info > (3,):
|
|
|
|
default_outfile = sys.stdout.buffer
|
|
|
|
else:
|
|
|
|
default_outfile = sys.stdout
|
|
|
|
p = argparse.ArgumentParser()
|
2022-09-20 10:01:32 +02:00
|
|
|
p.add_argument(
|
|
|
|
"-b",
|
|
|
|
"--baudrate",
|
|
|
|
type=int,
|
|
|
|
default=DEFAULT_BAUDRATE,
|
|
|
|
help="Baudrate of the serial port (only evaluated "
|
|
|
|
"for non TCP-terminal, default: %d)" % DEFAULT_BAUDRATE,
|
|
|
|
)
|
|
|
|
p.add_argument(
|
|
|
|
"conn",
|
|
|
|
metavar="tty/host:port",
|
|
|
|
type=str,
|
|
|
|
help="Serial port or TCP (host, port) tuple to "
|
|
|
|
"terminal with sniffer application",
|
|
|
|
)
|
2018-06-19 11:42:18 +02:00
|
|
|
p.add_argument("channel", type=int, help="Channel to sniff on")
|
2022-09-20 10:01:32 +02:00
|
|
|
p.add_argument(
|
|
|
|
"outfile",
|
|
|
|
type=argparse.FileType("w+b"),
|
|
|
|
default=default_outfile,
|
|
|
|
nargs="?",
|
|
|
|
help="PCAP file to output to (default: stdout)",
|
|
|
|
)
|
2018-06-19 11:42:18 +02:00
|
|
|
args = p.parse_args()
|
|
|
|
|
|
|
|
conn = connect(args)
|
2017-11-23 10:44:32 +01:00
|
|
|
|
|
|
|
sleep(1)
|
2018-06-19 11:42:18 +02:00
|
|
|
configure_interface(conn, args.channel)
|
2017-11-23 10:44:32 +01:00
|
|
|
sleep(1)
|
|
|
|
|
|
|
|
try:
|
2018-06-19 11:42:18 +02:00
|
|
|
generate_pcap(conn, args.outfile)
|
2017-11-23 10:44:32 +01:00
|
|
|
except KeyboardInterrupt:
|
|
|
|
conn.close()
|
|
|
|
print()
|
|
|
|
sys.exit(2)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2018-06-19 11:42:18 +02:00
|
|
|
main()
|