2014-11-06 11:35:59 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# BlueKitchen GmbH (c) 2014
|
|
|
|
|
|
|
|
# convert log output to PacketLogger format
|
|
|
|
# can be viewed with Wireshark
|
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
# APPLE PacketLogger
|
|
|
|
# typedef struct {
|
|
|
|
# uint32_t len;
|
|
|
|
# uint32_t ts_sec;
|
|
|
|
# uint32_t ts_usec;
|
|
|
|
# uint8_t type; // 0xfc for note
|
|
|
|
# }
|
|
|
|
|
2014-11-06 11:35:59 +00:00
|
|
|
import re
|
|
|
|
import sys
|
2014-11-06 14:12:51 +00:00
|
|
|
import time
|
2016-01-06 11:36:16 +01:00
|
|
|
import os
|
2014-11-06 14:12:51 +00:00
|
|
|
|
2016-10-03 10:58:38 +02:00
|
|
|
default_date="2001-01-01"
|
|
|
|
default_hours = 12
|
2014-11-06 14:12:51 +00:00
|
|
|
packet_counter = 0
|
2016-10-03 10:58:38 +02:00
|
|
|
last_time = default_date + " " + str(default_hours) + ":00:00.000"
|
2014-11-06 11:35:59 +00:00
|
|
|
|
|
|
|
def chop(line, prefix):
|
|
|
|
if line.startswith(prefix):
|
|
|
|
return line[len(prefix):]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def str2hex(value):
|
|
|
|
if value:
|
|
|
|
return int(value, 16)
|
|
|
|
return None
|
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
def arrayForNet32(value):
|
|
|
|
return bytearray([value >> 24, (value >> 16) & 0xff, (value >> 8) & 0xff, value & 0xff])
|
|
|
|
|
|
|
|
def generateTimestamp(t):
|
|
|
|
global last_time
|
|
|
|
global packet_counter
|
|
|
|
|
|
|
|
# use last_time if time missing for this entry
|
|
|
|
if not t:
|
|
|
|
t = last_time
|
|
|
|
if t:
|
|
|
|
last_time = t
|
2016-10-03 10:58:38 +02:00
|
|
|
|
|
|
|
# check for date
|
|
|
|
parts = t.split(' ')
|
|
|
|
have_date = True
|
|
|
|
if len(parts) == 1:
|
|
|
|
# only time, prepend fixed date
|
|
|
|
have_date = False
|
|
|
|
t = "2000-01-01 " + t;
|
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
# handle ms
|
2014-11-17 12:00:52 +00:00
|
|
|
try:
|
|
|
|
(t1, t2) = t.split('.')
|
|
|
|
if t1 and t2:
|
2016-10-03 10:58:38 +02:00
|
|
|
t_obj = time.strptime(t1, "%Y-%m-%d %H:%M:%S")
|
2014-11-17 12:00:52 +00:00
|
|
|
tv_sec = int(time.mktime(t_obj))
|
2016-10-03 10:58:38 +02:00
|
|
|
if not have_date:
|
|
|
|
# start at 12:00
|
|
|
|
tv_sec += 12*60*60
|
2014-11-17 12:00:52 +00:00
|
|
|
tv_usec = int(t2) * 1000
|
|
|
|
return (tv_sec, tv_usec)
|
|
|
|
except ValueError:
|
|
|
|
# print 'Cannot parse time', t
|
|
|
|
pass
|
2016-10-03 10:58:38 +02:00
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
packet_counter += 1
|
|
|
|
return (packet_counter, 0)
|
|
|
|
|
|
|
|
def dumpPacket(fout, timestamp, type, data):
|
|
|
|
length = 9 + len(data)
|
|
|
|
(tv_sec, tv_usec) = generateTimestamp(timestamp)
|
|
|
|
fout.write(arrayForNet32(length))
|
|
|
|
fout.write(arrayForNet32(tv_sec))
|
|
|
|
fout.write(arrayForNet32(tv_usec))
|
|
|
|
fout.write(bytearray([type]))
|
|
|
|
fout.write(data)
|
2014-11-06 11:35:59 +00:00
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
def handleHexPacket(fout, timestamp, type, text):
|
2014-11-17 12:00:52 +00:00
|
|
|
try:
|
2015-12-18 21:38:26 +01:00
|
|
|
data = bytearray(list(map(str2hex, text.strip().split())))
|
2014-11-17 12:00:52 +00:00
|
|
|
dumpPacket(fout, timestamp, type, data)
|
|
|
|
except TypeError:
|
2015-12-18 21:38:26 +01:00
|
|
|
print('Cannot parse hexdump', text.strip())
|
2014-11-06 11:35:59 +00:00
|
|
|
|
2014-11-06 14:12:51 +00:00
|
|
|
if len(sys.argv) == 1:
|
2015-12-18 21:38:26 +01:00
|
|
|
print('BTstack Console to PacketLogger converter')
|
|
|
|
print('Copyright 2014, BlueKitchen GmbH')
|
|
|
|
print('')
|
|
|
|
print('Usage: ', sys.argv[0], 'asci-log-file.txt [hci_dump.pkgl]')
|
|
|
|
print('Converted hci_dump.pklg can be viewed with Wireshark and OS X PacketLogger')
|
2014-11-06 14:12:51 +00:00
|
|
|
exit(0)
|
2014-11-06 11:35:59 +00:00
|
|
|
|
2014-11-07 15:24:24 +00:00
|
|
|
infile = sys.argv[1]
|
2016-01-06 11:36:16 +01:00
|
|
|
outfile = os.path.splitext(infile)[0] + ".pklg"
|
2014-11-06 11:35:59 +00:00
|
|
|
if len(sys.argv) > 2:
|
|
|
|
outfile = sys.argv[2]
|
|
|
|
|
|
|
|
# with open(outfile, 'w') as fout:
|
2014-11-06 14:12:51 +00:00
|
|
|
with open (outfile, 'wb') as fout:
|
2015-12-18 21:38:26 +01:00
|
|
|
with open (infile, 'rt') as fin:
|
2014-11-06 14:12:51 +00:00
|
|
|
packet_counter = 0
|
|
|
|
for line in fin:
|
|
|
|
timestamp = None
|
2017-04-03 19:12:02 +02:00
|
|
|
# strip newlines
|
|
|
|
line = line.strip("\n\r")
|
|
|
|
# skip empyt lines
|
|
|
|
if len(line) == 0:
|
|
|
|
continue
|
2016-10-03 10:58:38 +02:00
|
|
|
parts = re.match('\[(.*)\] (.*)', line)
|
2014-11-06 14:12:51 +00:00
|
|
|
if parts and len(parts.groups()) == 2:
|
|
|
|
(timestamp, line) = parts.groups()
|
|
|
|
rest = chop(line,'CMD => ')
|
|
|
|
if rest:
|
|
|
|
handleHexPacket(fout, timestamp, 0, rest)
|
|
|
|
continue
|
|
|
|
rest = chop(line,'EVT <= ')
|
|
|
|
if rest:
|
|
|
|
handleHexPacket(fout, timestamp, 1, rest)
|
|
|
|
continue
|
|
|
|
rest = chop(line,'ACL => ')
|
|
|
|
if rest:
|
|
|
|
handleHexPacket(fout, timestamp, 2, rest)
|
|
|
|
continue
|
|
|
|
rest = chop(line,'ACL <= ')
|
|
|
|
if rest:
|
|
|
|
handleHexPacket(fout, timestamp, 3, rest)
|
|
|
|
continue
|
|
|
|
rest = chop(line,'LOG -- ')
|
|
|
|
if rest:
|
|
|
|
line = rest
|
2015-12-18 21:38:26 +01:00
|
|
|
dumpPacket(fout, timestamp, 0xfc, line.encode('ascii'))
|