btstack/ble/compile-gatt.py

361 lines
9.2 KiB
Python
Raw Normal View History

#!/usr/bin/env python
#
# BLE GATT configuration generator for use with BTstack, v0.1
# Copyright 2011 Matthias Ringwald
#
# Format of input file:
# PRIMARY_SERVICE, SERVICE_UUID
# CHARACTERISTIC, ATTRIBUTE_TYPE_UUID, [READ | WRITE | DYNAMIC], VALUE
import re
header = '''
// {0} generated from {1} for BTstack
// binary representation
// attribute size in bytes (16), flags(16), handle (16), uuid (16/128), value(...)
#include <stdint.h>
const uint8_t profile_data[] =
'''
usage = '''
Usage: ./compile-gatt.py profile.gatt profile.h
'''
import re
import sys
print '''
BLE configuration generator for use with BTstack, v0.1
Copyright 2011 Matthias Ringwald
'''
assigned_uuids = {
'GAP_SERVICE' : 0x1800,
'GATT_SERVICE' : 0x1801,
'GAP_DEVICE_NAME' : 0x2a00,
'GAP_APPEARANCE' : 0x2a01,
'GATT_SERVICE_CHANGED' : 0x2a05,
}
property_flags = {
'BROADCAST' : 0x01,
'READ' : 0x02,
'WRITE_WITHOUT_RESPONSE' : 0x04,
'WRITE' : 0x08,
'NOTIFY': 0x10,
'INDICATE' : 0x20,
'AUTHENTICATED_SIGNED_WRITE' : 0x40,
'EXTENDED_PROPERTIES' : 0x80,
# custom BTstack extension
'DYNAMIC': 0x100,
'LONG_UUID': 0x200,
'AUTHENTICATION_REQUIRED': 0x400,
'AUTHORIZATION_REQUIRED': 0x800,
'ENCRYPTION_KEY_SIZE_7': 0x6000,
'ENCRYPTION_KEY_SIZE_8': 0x7000,
'ENCRYPTION_KEY_SIZE_9': 0x8000,
'ENCRYPTION_KEY_SIZE_10': 0x9000,
'ENCRYPTION_KEY_SIZE_11': 0xa000,
'ENCRYPTION_KEY_SIZE_12': 0xb000,
'ENCRYPTION_KEY_SIZE_13': 0xc000,
'ENCRYPTION_KEY_SIZE_14': 0xd000,
'ENCRYPTION_KEY_SIZE_15': 0xe000,
'ENCRYPTION_KEY_SIZE_16': 0xf000,
# only used by gatt compiler
'CLIENT_CONFIGURATION': 0x10000,
}
2013-12-05 14:26:48 +00:00
services = dict()
current_service_uuid = list()
current_service_start_handle = 0
handle = 1
total_size = 0
2013-12-05 14:26:48 +00:00
def keyForUUID(uuid):
keyUUID = ""
for i in uuid:
keyUUID += "%02x" % i
return keyUUID
def twoByteLEFor(value):
return [ (value & 0xff), (value >> 8)]
def is_128bit_uuid(text):
if re.match("[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{12}", text):
return True
return False
def parseUUID128(uuid):
parts = re.match("([0-9A-Fa-f]{4})([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})-([0-9A-Fa-f]{4})([0-9A-Fa-f]{4})([0-9A-Fa-f]{4})", uuid)
uuid_bytes = []
for i in range(8, 0, -1):
uuid_bytes = uuid_bytes + twoByteLEFor(int(parts.group(i),16))
return uuid_bytes
def parseUUID(uuid):
if uuid in assigned_uuids:
return twoByteLEFor(assigned_uuids[uuid])
if is_128bit_uuid(uuid):
return parseUUID128(uuid)
uuidInt = int(uuid, 16)
return twoByteLEFor(uuidInt)
def parseProperties(properties):
value = 0
parts = properties.split("|")
for property in parts:
property = property.strip()
if property in property_flags:
value |= property_flags[property]
else:
print "WARNING: property %s undefined" % (property)
return value
def write_8(fout, value):
fout.write( "0x%02x, " % (value & 0xff))
def write_16(fout, value):
fout.write('0x%02x, 0x%02x, ' % (value & 0xff, (value >> 8) & 0xff))
def write_uuid(uuid):
for byte in uuid:
fout.write( "0x%02x, " % byte)
def write_string(fout, text):
for l in text.lstrip('"').rstrip('"'):
write_8(fout, ord(l))
def write_sequence(fout, text):
parts = text.split()
for part in parts:
fout.write("0x%s, " % (part.strip()))
def write_indent(fout):
fout.write(" ")
def is_string(text):
if not text.startswith('"'):
return False
if not text.endswith('"'):
return False
return True
2013-12-05 14:26:48 +00:00
def parseService(fout, parts, service_type):
global handle
global total_size
2013-12-05 14:26:48 +00:00
global current_service_uuid
global current_service_start_handle
global services
keyUUID = keyForUUID(current_service_uuid)
if keyUUID:
fout.write("\n")
# print "append service %s = [%d, %d]\n" % (keyUUID, current_service_start_handle, handle-1)
services[keyUUID] = [current_service_start_handle, handle-1]
property = property_flags['READ'];
write_indent(fout)
fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts)))
uuid = parseUUID(parts[1])
uuid_size = len(uuid)
2013-12-05 14:26:48 +00:00
size = 2 + 2 + 2 + uuid_size + 2
2013-12-05 14:26:48 +00:00
if service_type == 0x2802:
size += 4
if uuid_size == 16:
property = property | property_flags['LONG_UUID'];
2013-12-05 14:26:48 +00:00
write_indent(fout)
write_16(fout, size)
write_16(fout, property)
write_16(fout, handle)
write_16(fout, service_type)
write_uuid(uuid)
fout.write("\n")
current_service_uuid = uuid
current_service_start_handle = handle
handle = handle + 1
total_size = total_size + size
def parsePrimaryService(fout, parts):
parseService(fout, parts, 0x2800)
def parseSecondaryService(fout, parts):
parseService(fout, parts, 0x2801)
def parseIncludeService(fout, parts):
global handle
global total_size
property = property_flags['READ'];
2013-12-05 14:26:48 +00:00
write_indent(fout)
fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts)))
uuid = parseUUID(parts[1])
uuid_size = len(uuid)
# print "Include Service ", keyForUUID(uuid)
size = 2 + 2 + 2 + uuid_size + 2 + 4
if uuid_size == 16:
property = property | property_flags['LONG_UUID'];
keyUUID = keyForUUID(uuid)
write_indent(fout)
write_16(fout, size)
write_16(fout, property)
write_16(fout, handle)
2013-12-05 14:26:48 +00:00
write_16(fout, 0x2802)
write_16(fout, services[keyUUID][0])
write_16(fout, services[keyUUID][1])
write_uuid(uuid)
fout.write("\n")
2013-12-05 14:26:48 +00:00
handle = handle + 1
total_size = total_size + size
2013-12-05 14:26:48 +00:00
def parseCharacteristic(fout, parts):
global handle
global total_size
property_read = property_flags['READ'];
uuid = parseUUID(parts[1])
uuid_size = len(uuid)
properties = parseProperties(parts[2])
value = parts[3]
write_indent(fout)
fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts[0:len(parts)-1])))
size = 2 + 2 + 2 + 2 + (1+2+uuid_size)
write_indent(fout)
write_16(fout, size)
write_16(fout, property_read)
write_16(fout, handle)
write_16(fout, 0x2803)
write_8(fout, properties)
write_16(fout, handle+1)
write_uuid(uuid)
fout.write("\n")
handle = handle + 1
total_size = total_size + size
size = 2 + 2 + 2 + uuid_size
if is_string(value):
size = size + len(value) - 2
else:
size = size + len(value.split())
if uuid_size == 16:
properties = properties | property_flags['LONG_UUID'];
write_indent(fout)
fout.write('// 0x%04x VALUE-%s\n' % (handle, '-'.join(parts[1:len(parts)])))
write_indent(fout)
write_16(fout, size)
write_16(fout, properties)
write_16(fout, handle)
write_uuid(uuid)
if is_string(value):
write_string(fout, value)
else:
write_sequence(fout,value)
fout.write("\n")
handle = handle + 1
if (properties & property_flags['CLIENT_CONFIGURATION']) == 0:
return
size = 2 + 2 + 2 + 2 + 2
write_indent(fout)
fout.write('// 0x%04x CLIENT_CHARACTERISTIC_CONFIGURATION\n' % (handle))
write_indent(fout)
write_16(fout, size)
write_16(fout, property_flags['READ'] | property_flags['WRITE'] | property_flags['DYNAMIC'])
write_16(fout, handle)
write_16(fout, 0x2902)
write_16(fout, 0)
fout.write("\n")
handle = handle + 1
2013-12-05 14:26:48 +00:00
def parse(fname_in, fin, fname_out, fout):
global handle
global total_size
fout.write(header.format(fname_out, fname_in))
fout.write('{\n')
for line in fin:
line = line.strip("\n\r ")
if line.startswith("#") or line.startswith("//"):
fout.write("//" + line.lstrip('/#') + '\n')
continue
if len(line) == 0:
continue
parts = line.split(',')
for index, object in enumerate(parts):
parts[index] = object.strip()
if parts[0] == 'PRIMARY_SERVICE':
parsePrimaryService(fout, parts)
continue
2013-12-05 14:26:48 +00:00
if parts[0] == 'SECONDARY_SERVICE':
parseSecondaryService(fout, parts)
continue
if parts[0] == 'INCLUDE_SERVICE':
parseIncludeService(fout, parts)
continue
if parts[0] == 'CHARACTERISTIC':
parseCharacteristic(fout, parts)
continue
print "WARNING: unknown token: %s\n" % (parts[0])
write_indent(fout)
fout.write("// END\n");
write_indent(fout)
write_16(fout,0)
fout.write("\n")
total_size = total_size + 2
fout.write("}; // total size %u bytes \n" % total_size);
fout.close()
print 'Created', fname_out
if (len(sys.argv) < 3):
print usage
sys.exit(1)
try:
fin = open (sys.argv[1], 'r')
fout = open (sys.argv[2], 'w')
parse(sys.argv[1], fin, sys.argv[2], fout)
except IOError as e:
print usage
sys.exit(1)
print 'Compilation successful!\n'