xref: /btstack/tool/compile_gatt.py (revision e22a261227a2522f26982870e4eee0b42606d652)
1b3fcedb9SMatthias Ringwald#!/usr/bin/env python
2b3fcedb9SMatthias Ringwald#
3b3fcedb9SMatthias Ringwald# BLE GATT configuration generator for use with BTstack, v0.1
4b3fcedb9SMatthias Ringwald# Copyright 2011 Matthias Ringwald
5b3fcedb9SMatthias Ringwald#
6b3fcedb9SMatthias Ringwald# Format of input file:
7b3fcedb9SMatthias Ringwald# PRIMARY_SERVICE, SERVICE_UUID
8b3fcedb9SMatthias Ringwald# CHARACTERISTIC, ATTRIBUTE_TYPE_UUID, [READ | WRITE | DYNAMIC], VALUE
9b3fcedb9SMatthias Ringwald
10b3fcedb9SMatthias Ringwaldimport codecs
11b165f97bSMatthias Ringwaldimport csv
12b165f97bSMatthias Ringwaldimport io
13b165f97bSMatthias Ringwaldimport os
14b165f97bSMatthias Ringwaldimport re
15b165f97bSMatthias Ringwaldimport string
1660b51a4cSMatthias Ringwaldimport sys
17b3fcedb9SMatthias Ringwald
18b3fcedb9SMatthias Ringwaldheader = '''
19b3fcedb9SMatthias Ringwald// {0} generated from {1} for BTstack
20b3fcedb9SMatthias Ringwald
21b3fcedb9SMatthias Ringwald// binary representation
22b3fcedb9SMatthias Ringwald// attribute size in bytes (16), flags(16), handle (16), uuid (16/128), value(...)
23b3fcedb9SMatthias Ringwald
24b3fcedb9SMatthias Ringwald#include <stdint.h>
25b3fcedb9SMatthias Ringwald
26b3fcedb9SMatthias Ringwaldconst uint8_t profile_data[] =
27b3fcedb9SMatthias Ringwald'''
28b3fcedb9SMatthias Ringwald
29b3fcedb9SMatthias Ringwaldusage = '''
30b3fcedb9SMatthias RingwaldUsage: ./compile_gatt.py profile.gatt profile.h
31b3fcedb9SMatthias Ringwald'''
32b3fcedb9SMatthias Ringwald
33b3fcedb9SMatthias Ringwald
34b3fcedb9SMatthias Ringwaldprint('''
35b3fcedb9SMatthias RingwaldBLE configuration generator for use with BTstack, v0.1
36b3fcedb9SMatthias RingwaldCopyright 2011 Matthias Ringwald
37b3fcedb9SMatthias Ringwald''')
38b3fcedb9SMatthias Ringwald
39b3fcedb9SMatthias Ringwaldassigned_uuids = {
40b3fcedb9SMatthias Ringwald    'GAP_SERVICE'          : 0x1800,
41b3fcedb9SMatthias Ringwald    'GATT_SERVICE'         : 0x1801,
42b3fcedb9SMatthias Ringwald    'GAP_DEVICE_NAME'      : 0x2a00,
43b3fcedb9SMatthias Ringwald    'GAP_APPEARANCE'       : 0x2a01,
44b3fcedb9SMatthias Ringwald    'GAP_PERIPHERAL_PRIVACY_FLAG' : 0x2A02,
45b3fcedb9SMatthias Ringwald    'GAP_RECONNECTION_ADDRESS'    : 0x2A03,
46b3fcedb9SMatthias Ringwald    'GAP_PERIPHERAL_PREFERRED_CONNECTION_PARAMETERS' : 0x2A04,
47b3fcedb9SMatthias Ringwald    'GATT_SERVICE_CHANGED' : 0x2a05,
48b3fcedb9SMatthias Ringwald}
49b3fcedb9SMatthias Ringwald
50b3fcedb9SMatthias Ringwaldproperty_flags = {
51eb6072adSMatthias Ringwald    # GATT Characteristic Properties
52b3fcedb9SMatthias Ringwald    'BROADCAST' :                   0x01,
53b3fcedb9SMatthias Ringwald    'READ' :                        0x02,
54b3fcedb9SMatthias Ringwald    'WRITE_WITHOUT_RESPONSE' :      0x04,
55b3fcedb9SMatthias Ringwald    'WRITE' :                       0x08,
56b3fcedb9SMatthias Ringwald    'NOTIFY':                       0x10,
57b3fcedb9SMatthias Ringwald    'INDICATE' :                    0x20,
58b3fcedb9SMatthias Ringwald    'AUTHENTICATED_SIGNED_WRITE' :  0x40,
59b3fcedb9SMatthias Ringwald    'EXTENDED_PROPERTIES' :         0x80,
60b3fcedb9SMatthias Ringwald    # custom BTstack extension
61b3fcedb9SMatthias Ringwald    'DYNAMIC':                      0x100,
62b3fcedb9SMatthias Ringwald    'LONG_UUID':                    0x200,
63*e22a2612SMatthias Ringwald
64*e22a2612SMatthias Ringwald    # read permissions
65*e22a2612SMatthias Ringwald    'READ_PERMISSION_BIT_0':        0x400,
66*e22a2612SMatthias Ringwald    'READ_PERMISSION_BIT_1':        0x800,
67*e22a2612SMatthias Ringwald
68*e22a2612SMatthias Ringwald    #
69b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_7':       0x6000,
70b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_8':       0x7000,
71b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_9':       0x8000,
72b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_10':      0x9000,
73b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_11':      0xa000,
74b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_12':      0xb000,
75b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_13':      0xc000,
76b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_14':      0xd000,
77b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_15':      0xe000,
78b3fcedb9SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_16':      0xf000,
79*e22a2612SMatthias Ringwald    'ENCRYPTION_KEY_SIZE_MASK':    0xf000,
80eb6072adSMatthias Ringwald
81b3fcedb9SMatthias Ringwald    # only used by gatt compiler >= 0xffff
82b3fcedb9SMatthias Ringwald    # Extended Properties
83*e22a2612SMatthias Ringwald    'RELIABLE_WRITE':              0x0010000,
84*e22a2612SMatthias Ringwald    'AUTHENTICATION_REQUIRED':     0x0020000,
85*e22a2612SMatthias Ringwald    'AUTHORIZATION_REQUIRED':      0x0040000,
86*e22a2612SMatthias Ringwald    'READ_ANYBODY':                0x0080000,
87*e22a2612SMatthias Ringwald    'READ_ENCRYPTED':              0x0100000,
88*e22a2612SMatthias Ringwald    'READ_AUTHENTICATED':          0x0200000,
89*e22a2612SMatthias Ringwald    'READ_AUTHORIZED':             0x0400000,
90*e22a2612SMatthias Ringwald    'WRITE_ANYBODY':               0x0800000,
91*e22a2612SMatthias Ringwald    'WRITE_ENCRYPTED':             0x1000000,
92*e22a2612SMatthias Ringwald    'WRITE_AUTHENTICATED':         0x2000000,
93*e22a2612SMatthias Ringwald    'WRITE_AUTHORIZED':            0x4000000,
94eb6072adSMatthias Ringwald
95eb6072adSMatthias Ringwald    # Broadcast, Notify, Indicate, Extended Properties are only used to describe a GATT Characteristic, but are free to use with att_db
96*e22a2612SMatthias Ringwald
97*e22a2612SMatthias Ringwald    # write permissions
98*e22a2612SMatthias Ringwald    'WRITE_PERMISSION_BIT_0':      0x01,
99*e22a2612SMatthias Ringwald    'WRITE_PERMISSION_BIT_1':      0x10,
100eb6072adSMatthias Ringwald    # 0x20
101eb6072adSMatthias Ringwald    # 0x80
102b3fcedb9SMatthias Ringwald}
103b3fcedb9SMatthias Ringwald
10460b51a4cSMatthias Ringwaldbtstack_root = ''
105b3fcedb9SMatthias Ringwaldservices = dict()
106b3fcedb9SMatthias Ringwaldcharacteristic_indices = dict()
107b3fcedb9SMatthias Ringwaldpresentation_formats = dict()
108b3fcedb9SMatthias Ringwaldcurrent_service_uuid_string = ""
109b3fcedb9SMatthias Ringwaldcurrent_service_start_handle = 0
110b3fcedb9SMatthias Ringwaldcurrent_characteristic_uuid_string = ""
111729074c4SMatthias Ringwalddefines_for_characteristics = []
112729074c4SMatthias Ringwalddefines_for_services = []
113b3fcedb9SMatthias Ringwald
114b3fcedb9SMatthias Ringwaldhandle = 1
115b3fcedb9SMatthias Ringwaldtotal_size = 0
116b3fcedb9SMatthias Ringwald
117b165f97bSMatthias Ringwalddef read_defines(infile):
118b165f97bSMatthias Ringwald    defines = dict()
119b165f97bSMatthias Ringwald    with open (infile, 'rt') as fin:
120b165f97bSMatthias Ringwald        for line in fin:
121b165f97bSMatthias Ringwald            parts = re.match('#define\s+(\w+)\s+(\w+)',line)
122b165f97bSMatthias Ringwald            if parts and len(parts.groups()) == 2:
123b165f97bSMatthias Ringwald                (key, value) = parts.groups()
124b165f97bSMatthias Ringwald                defines[key] = int(value, 16)
125b165f97bSMatthias Ringwald    return defines
126b165f97bSMatthias Ringwald
127b3fcedb9SMatthias Ringwalddef keyForUUID(uuid):
128b3fcedb9SMatthias Ringwald    keyUUID = ""
129b3fcedb9SMatthias Ringwald    for i in uuid:
130b3fcedb9SMatthias Ringwald        keyUUID += "%02x" % i
131b3fcedb9SMatthias Ringwald    return keyUUID
132b3fcedb9SMatthias Ringwald
133b3fcedb9SMatthias Ringwalddef c_string_for_uuid(uuid):
134b3fcedb9SMatthias Ringwald    return uuid.replace('-', '_')
135b3fcedb9SMatthias Ringwald
136b3fcedb9SMatthias Ringwalddef twoByteLEFor(value):
137b3fcedb9SMatthias Ringwald    return [ (value & 0xff), (value >> 8)]
138b3fcedb9SMatthias Ringwald
139b3fcedb9SMatthias Ringwalddef is_128bit_uuid(text):
140b3fcedb9SMatthias Ringwald    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):
141b3fcedb9SMatthias Ringwald        return True
142b3fcedb9SMatthias Ringwald    return False
143b3fcedb9SMatthias Ringwald
144b3fcedb9SMatthias Ringwalddef parseUUID128(uuid):
145b3fcedb9SMatthias Ringwald    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)
146b3fcedb9SMatthias Ringwald    uuid_bytes = []
147b3fcedb9SMatthias Ringwald    for i in range(8, 0, -1):
148b3fcedb9SMatthias Ringwald        uuid_bytes = uuid_bytes + twoByteLEFor(int(parts.group(i),16))
149b3fcedb9SMatthias Ringwald    return uuid_bytes
150b3fcedb9SMatthias Ringwald
151b3fcedb9SMatthias Ringwalddef parseUUID(uuid):
152b3fcedb9SMatthias Ringwald    if uuid in assigned_uuids:
153b3fcedb9SMatthias Ringwald        return twoByteLEFor(assigned_uuids[uuid])
154b165f97bSMatthias Ringwald    uuid_upper = uuid.upper().replace('.','_')
155b165f97bSMatthias Ringwald    if uuid_upper in bluetooth_gatt:
156b165f97bSMatthias Ringwald        return twoByteLEFor(bluetooth_gatt[uuid_upper])
157b3fcedb9SMatthias Ringwald    if is_128bit_uuid(uuid):
158b3fcedb9SMatthias Ringwald        return parseUUID128(uuid)
159b3fcedb9SMatthias Ringwald    uuidInt = int(uuid, 16)
160b3fcedb9SMatthias Ringwald    return twoByteLEFor(uuidInt)
161b3fcedb9SMatthias Ringwald
162b3fcedb9SMatthias Ringwalddef parseProperties(properties):
163b3fcedb9SMatthias Ringwald    value = 0
164b3fcedb9SMatthias Ringwald    parts = properties.split("|")
165b3fcedb9SMatthias Ringwald    for property in parts:
166b3fcedb9SMatthias Ringwald        property = property.strip()
167b3fcedb9SMatthias Ringwald        if property in property_flags:
168b3fcedb9SMatthias Ringwald            value |= property_flags[property]
169b3fcedb9SMatthias Ringwald        else:
170b3fcedb9SMatthias Ringwald            print("WARNING: property %s undefined" % (property))
171*e22a2612SMatthias Ringwald
172*e22a2612SMatthias Ringwald    return value;
173*e22a2612SMatthias Ringwald
174*e22a2612SMatthias Ringwalddef gatt_characteristic_properties(properties):
175*e22a2612SMatthias Ringwald    return properties & 0xff
176*e22a2612SMatthias Ringwald
177*e22a2612SMatthias Ringwalddef att_flags(properties):
178*e22a2612SMatthias Ringwald    print("in %x" % properties)
179*e22a2612SMatthias Ringwald    # drop Broadcast (0x01), Notify (0x10), Indicate (0x20)- not used for flags
180*e22a2612SMatthias Ringwald    properties &= 0x1ffce
181*e22a2612SMatthias Ringwald
182*e22a2612SMatthias Ringwald    # rw permissions distinct
183*e22a2612SMatthias Ringwald    distinct_permissions_used = properties & (
184*e22a2612SMatthias Ringwald        property_flags['READ_AUTHORIZED'] |
185*e22a2612SMatthias Ringwald        property_flags['READ_AUTHENTICATED'] |
186*e22a2612SMatthias Ringwald        property_flags['READ_ENCRYPTED'] |
187*e22a2612SMatthias Ringwald        property_flags['READ_ANYBODY'] |
188*e22a2612SMatthias Ringwald        property_flags['WRITE_AUTHORIZED'] |
189*e22a2612SMatthias Ringwald        property_flags['WRITE_AUTHENTICATED'] |
190*e22a2612SMatthias Ringwald        property_flags['WRITE_ENCRYPTED'] |
191*e22a2612SMatthias Ringwald        property_flags['WRITE_ANYBODY']
192*e22a2612SMatthias Ringwald    ) != 0
193*e22a2612SMatthias Ringwald
194*e22a2612SMatthias Ringwald    # post process properties
195*e22a2612SMatthias Ringwald    encryption_key_size_specified = (properties & property_flags['ENCRYPTION_KEY_SIZE_MASK']) != 0
196*e22a2612SMatthias Ringwald
197*e22a2612SMatthias Ringwald    # set encrypted for both, if distinct permissions not used
198*e22a2612SMatthias Ringwald    if encryption_key_size_specified and not distinct_permissions_used:
199*e22a2612SMatthias Ringwald        properties |= property_flags['READ_ENCRYPTED'] | property_flags['WRITE_ENCRYPTED']
200*e22a2612SMatthias Ringwald
201*e22a2612SMatthias Ringwald    # convert user permissions to att db flags
202*e22a2612SMatthias Ringwald    if properties & property_flags['READ_AUTHORIZED']:
203*e22a2612SMatthias Ringwald        properties |= property_flags['READ_PERMISSION_BIT_0'] | property_flags['READ_PERMISSION_BIT_1']
204*e22a2612SMatthias Ringwald    elif properties & property_flags['READ_AUTHENTICATED']:
205*e22a2612SMatthias Ringwald        properties |= property_flags['READ_PERMISSION_BIT_1']
206*e22a2612SMatthias Ringwald    elif properties & property_flags['READ_ENCRYPTED']:
207*e22a2612SMatthias Ringwald        properties |= property_flags['READ_PERMISSION_BIT_0']
208*e22a2612SMatthias Ringwald    if properties & property_flags['WRITE_AUTHORIZED']:
209*e22a2612SMatthias Ringwald        properties |= property_flags['WRITE_PERMISSION_BIT_0'] | property_flags['WRITE_PERMISSION_BIT_1']
210*e22a2612SMatthias Ringwald    elif properties & property_flags['WRITE_AUTHENTICATED']:
211*e22a2612SMatthias Ringwald        properties |= property_flags['WRITE_PERMISSION_BIT_1']
212*e22a2612SMatthias Ringwald    elif properties & property_flags['WRITE_ENCRYPTED']:
213*e22a2612SMatthias Ringwald        properties |= property_flags['WRITE_PERMISSION_BIT_0']
214*e22a2612SMatthias Ringwald    print("out %x" % properties)
215*e22a2612SMatthias Ringwald
216*e22a2612SMatthias Ringwald    return properties
217*e22a2612SMatthias Ringwald
218*e22a2612SMatthias Ringwalddef write_permissions_and_key_size(properties):
219*e22a2612SMatthias Ringwald    return att_flags(properties) & (property_flags['ENCRYPTION_KEY_SIZE_MASK'] | property_flags['WRITE_PERMISSION_BIT_0'] | property_flags['WRITE_PERMISSION_BIT_1'])
220b3fcedb9SMatthias Ringwald
221b3fcedb9SMatthias Ringwalddef write_8(fout, value):
222b3fcedb9SMatthias Ringwald    fout.write( "0x%02x, " % (value & 0xff))
223b3fcedb9SMatthias Ringwald
224b3fcedb9SMatthias Ringwalddef write_16(fout, value):
225b3fcedb9SMatthias Ringwald    fout.write('0x%02x, 0x%02x, ' % (value & 0xff, (value >> 8) & 0xff))
226b3fcedb9SMatthias Ringwald
227b3fcedb9SMatthias Ringwalddef write_uuid(uuid):
228b3fcedb9SMatthias Ringwald    for byte in uuid:
229b3fcedb9SMatthias Ringwald        fout.write( "0x%02x, " % byte)
230b3fcedb9SMatthias Ringwald
231b3fcedb9SMatthias Ringwalddef write_string(fout, text):
232b3fcedb9SMatthias Ringwald    for l in text.lstrip('"').rstrip('"'):
233b3fcedb9SMatthias Ringwald        write_8(fout, ord(l))
234b3fcedb9SMatthias Ringwald
235b3fcedb9SMatthias Ringwalddef write_sequence(fout, text):
236b3fcedb9SMatthias Ringwald    parts = text.split()
237b3fcedb9SMatthias Ringwald    for part in parts:
238b3fcedb9SMatthias Ringwald        fout.write("0x%s, " % (part.strip()))
239b3fcedb9SMatthias Ringwald
240b3fcedb9SMatthias Ringwalddef write_indent(fout):
241b3fcedb9SMatthias Ringwald    fout.write("    ")
242b3fcedb9SMatthias Ringwald
243b3fcedb9SMatthias Ringwalddef is_string(text):
244b3fcedb9SMatthias Ringwald    for item in text.split(" "):
245b3fcedb9SMatthias Ringwald        if not all(c in string.hexdigits for c in item):
246b3fcedb9SMatthias Ringwald            return True
247b3fcedb9SMatthias Ringwald    return False
248b3fcedb9SMatthias Ringwald
249b3fcedb9SMatthias Ringwalddef add_client_characteristic_configuration(properties):
250b3fcedb9SMatthias Ringwald    return properties & (property_flags['NOTIFY'] | property_flags['INDICATE'])
251b3fcedb9SMatthias Ringwald
252729074c4SMatthias Ringwalddef serviceDefinitionComplete(fout):
253729074c4SMatthias Ringwald    global services
254729074c4SMatthias Ringwald    if current_service_uuid_string:
255729074c4SMatthias Ringwald        fout.write("\n")
256729074c4SMatthias Ringwald        # print("append service %s = [%d, %d]" % (current_characteristic_uuid_string, current_service_start_handle, handle-1))
257729074c4SMatthias Ringwald        defines_for_services.append('#define ATT_SERVICE_%s_START_HANDLE 0x%04x' % (current_service_uuid_string, current_service_start_handle))
258729074c4SMatthias Ringwald        defines_for_services.append('#define ATT_SERVICE_%s_END_HANDLE 0x%04x' % (current_service_uuid_string, handle-1))
259729074c4SMatthias Ringwald        services[current_service_uuid_string] = [current_service_start_handle, handle-1]
260729074c4SMatthias Ringwald
261b3fcedb9SMatthias Ringwalddef parseService(fout, parts, service_type):
262b3fcedb9SMatthias Ringwald    global handle
263b3fcedb9SMatthias Ringwald    global total_size
264b3fcedb9SMatthias Ringwald    global current_service_uuid_string
265b3fcedb9SMatthias Ringwald    global current_service_start_handle
266b3fcedb9SMatthias Ringwald
267729074c4SMatthias Ringwald    serviceDefinitionComplete(fout)
268b3fcedb9SMatthias Ringwald
269b3fcedb9SMatthias Ringwald    property = property_flags['READ'];
270b3fcedb9SMatthias Ringwald
271b3fcedb9SMatthias Ringwald    write_indent(fout)
272b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts)))
273b3fcedb9SMatthias Ringwald
274b3fcedb9SMatthias Ringwald    uuid = parseUUID(parts[1])
275b3fcedb9SMatthias Ringwald    uuid_size = len(uuid)
276b3fcedb9SMatthias Ringwald
277b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + uuid_size + 2
278b3fcedb9SMatthias Ringwald
279b3fcedb9SMatthias Ringwald    if service_type == 0x2802:
280b3fcedb9SMatthias Ringwald        size += 4
281b3fcedb9SMatthias Ringwald
282b3fcedb9SMatthias Ringwald    write_indent(fout)
283b3fcedb9SMatthias Ringwald    write_16(fout, size)
284b3fcedb9SMatthias Ringwald    write_16(fout, property)
285b3fcedb9SMatthias Ringwald    write_16(fout, handle)
286b3fcedb9SMatthias Ringwald    write_16(fout, service_type)
287b3fcedb9SMatthias Ringwald    write_uuid(uuid)
288b3fcedb9SMatthias Ringwald    fout.write("\n")
289b3fcedb9SMatthias Ringwald
290729074c4SMatthias Ringwald    current_service_uuid_string = c_string_for_uuid(parts[1])
291b3fcedb9SMatthias Ringwald    current_service_start_handle = handle
292b3fcedb9SMatthias Ringwald    handle = handle + 1
293b3fcedb9SMatthias Ringwald    total_size = total_size + size
294b3fcedb9SMatthias Ringwald
295b3fcedb9SMatthias Ringwalddef parsePrimaryService(fout, parts):
296b3fcedb9SMatthias Ringwald    parseService(fout, parts, 0x2800)
297b3fcedb9SMatthias Ringwald
298b3fcedb9SMatthias Ringwalddef parseSecondaryService(fout, parts):
299b3fcedb9SMatthias Ringwald    parseService(fout, parts, 0x2801)
300b3fcedb9SMatthias Ringwald
301b3fcedb9SMatthias Ringwalddef parseIncludeService(fout, parts):
302b3fcedb9SMatthias Ringwald    global handle
303b3fcedb9SMatthias Ringwald    global total_size
304b3fcedb9SMatthias Ringwald
305*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
306b3fcedb9SMatthias Ringwald
307b3fcedb9SMatthias Ringwald    write_indent(fout)
308b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts)))
309b3fcedb9SMatthias Ringwald
310b3fcedb9SMatthias Ringwald    uuid = parseUUID(parts[1])
311b3fcedb9SMatthias Ringwald    uuid_size = len(uuid)
312b3fcedb9SMatthias Ringwald    if uuid_size > 2:
313b3fcedb9SMatthias Ringwald        uuid_size = 0
314729074c4SMatthias Ringwald    # print("Include Service ", c_string_for_uuid(uuid))
315b3fcedb9SMatthias Ringwald
316b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + 4 + uuid_size
317b3fcedb9SMatthias Ringwald
318729074c4SMatthias Ringwald    keyUUID = c_string_for_uuid(parts[1])
319b3fcedb9SMatthias Ringwald
320b3fcedb9SMatthias Ringwald    write_indent(fout)
321b3fcedb9SMatthias Ringwald    write_16(fout, size)
322*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
323b3fcedb9SMatthias Ringwald    write_16(fout, handle)
324b3fcedb9SMatthias Ringwald    write_16(fout, 0x2802)
325b3fcedb9SMatthias Ringwald    write_16(fout, services[keyUUID][0])
326b3fcedb9SMatthias Ringwald    write_16(fout, services[keyUUID][1])
327b3fcedb9SMatthias Ringwald    if uuid_size > 0:
328b3fcedb9SMatthias Ringwald        write_uuid(uuid)
329b3fcedb9SMatthias Ringwald    fout.write("\n")
330b3fcedb9SMatthias Ringwald
331b3fcedb9SMatthias Ringwald    handle = handle + 1
332b3fcedb9SMatthias Ringwald    total_size = total_size + size
333b3fcedb9SMatthias Ringwald
334b3fcedb9SMatthias Ringwald
335b3fcedb9SMatthias Ringwalddef parseCharacteristic(fout, parts):
336b3fcedb9SMatthias Ringwald    global handle
337b3fcedb9SMatthias Ringwald    global total_size
338b3fcedb9SMatthias Ringwald    global current_characteristic_uuid_string
339b3fcedb9SMatthias Ringwald    global characteristic_indices
340b3fcedb9SMatthias Ringwald
341*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
342b3fcedb9SMatthias Ringwald
343b3fcedb9SMatthias Ringwald    # enumerate characteristics with same UUID, using optional name tag if available
344b3fcedb9SMatthias Ringwald    current_characteristic_uuid_string = c_string_for_uuid(parts[1]);
345b3fcedb9SMatthias Ringwald    index = 1
346b3fcedb9SMatthias Ringwald    if current_characteristic_uuid_string in characteristic_indices:
347b3fcedb9SMatthias Ringwald        index = characteristic_indices[current_characteristic_uuid_string] + 1
348b3fcedb9SMatthias Ringwald    characteristic_indices[current_characteristic_uuid_string] = index
349b3fcedb9SMatthias Ringwald    if len(parts) > 4:
350b3fcedb9SMatthias Ringwald        current_characteristic_uuid_string += '_' + parts[4].upper().replace(' ','_')
351b3fcedb9SMatthias Ringwald    else:
352b3fcedb9SMatthias Ringwald        current_characteristic_uuid_string += ('_%02x' % index)
353b3fcedb9SMatthias Ringwald
354b3fcedb9SMatthias Ringwald    uuid       = parseUUID(parts[1])
355b3fcedb9SMatthias Ringwald    uuid_size  = len(uuid)
356b3fcedb9SMatthias Ringwald    properties = parseProperties(parts[2])
357b3fcedb9SMatthias Ringwald    value = ', '.join([str(x) for x in parts[3:]])
358b3fcedb9SMatthias Ringwald
359b3fcedb9SMatthias Ringwald    # reliable writes is defined in an extended properties
360b3fcedb9SMatthias Ringwald    if (properties & property_flags['RELIABLE_WRITE']):
361b3fcedb9SMatthias Ringwald        properties = properties | property_flags['EXTENDED_PROPERTIES']
362b3fcedb9SMatthias Ringwald
363b3fcedb9SMatthias Ringwald    write_indent(fout)
364b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x %s\n' % (handle, '-'.join(parts[0:3])))
365b3fcedb9SMatthias Ringwald
366*e22a2612SMatthias Ringwald
367*e22a2612SMatthias Ringwald    characteristic_properties = gatt_characteristic_properties(properties)
368b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + (1+2+uuid_size)
369b3fcedb9SMatthias Ringwald    write_indent(fout)
370b3fcedb9SMatthias Ringwald    write_16(fout, size)
371*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
372b3fcedb9SMatthias Ringwald    write_16(fout, handle)
373b3fcedb9SMatthias Ringwald    write_16(fout, 0x2803)
374*e22a2612SMatthias Ringwald    write_8(fout, characteristic_properties)
375b3fcedb9SMatthias Ringwald    write_16(fout, handle+1)
376b3fcedb9SMatthias Ringwald    write_uuid(uuid)
377b3fcedb9SMatthias Ringwald    fout.write("\n")
378b3fcedb9SMatthias Ringwald    handle = handle + 1
379b3fcedb9SMatthias Ringwald    total_size = total_size + size
380b3fcedb9SMatthias Ringwald
381b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + uuid_size
382b3fcedb9SMatthias Ringwald    if is_string(value):
383b3fcedb9SMatthias Ringwald        size = size + len(value)
384b3fcedb9SMatthias Ringwald    else:
385b3fcedb9SMatthias Ringwald        size = size + len(value.split())
386b3fcedb9SMatthias Ringwald
387*e22a2612SMatthias Ringwald    value_flags = att_flags(properties)
3888ea3236cSMatthias Ringwald
3898ea3236cSMatthias Ringwald    # add UUID128 flag for value handle
390b3fcedb9SMatthias Ringwald    if uuid_size == 16:
391*e22a2612SMatthias Ringwald        value_flags = value_flags | property_flags['LONG_UUID'];
392b3fcedb9SMatthias Ringwald
393b3fcedb9SMatthias Ringwald    write_indent(fout)
394b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x VALUE-%s-'"'%s'"'\n' % (handle, '-'.join(parts[1:3]),value))
395b3fcedb9SMatthias Ringwald    write_indent(fout)
396b3fcedb9SMatthias Ringwald    write_16(fout, size)
397*e22a2612SMatthias Ringwald    write_16(fout, value_flags)
398b3fcedb9SMatthias Ringwald    write_16(fout, handle)
399b3fcedb9SMatthias Ringwald    write_uuid(uuid)
400b3fcedb9SMatthias Ringwald    if is_string(value):
401b3fcedb9SMatthias Ringwald        write_string(fout, value)
402b3fcedb9SMatthias Ringwald    else:
403b3fcedb9SMatthias Ringwald        write_sequence(fout,value)
404b3fcedb9SMatthias Ringwald
405b3fcedb9SMatthias Ringwald    fout.write("\n")
406729074c4SMatthias Ringwald    defines_for_characteristics.append('#define ATT_CHARACTERISTIC_%s_VALUE_HANDLE 0x%04x' % (current_characteristic_uuid_string, handle))
407b3fcedb9SMatthias Ringwald    handle = handle + 1
408b3fcedb9SMatthias Ringwald
409b3fcedb9SMatthias Ringwald    if add_client_characteristic_configuration(properties):
410*e22a2612SMatthias Ringwald        # use write permissions and encryption key size from attribute value and set READ_ANYBODY | READ | WRITE | DYNAMIC
411*e22a2612SMatthias Ringwald        flags  = write_permissions_and_key_size(properties)
412*e22a2612SMatthias Ringwald        flags |= property_flags['READ']
413*e22a2612SMatthias Ringwald        flags |= property_flags['WRITE']
414*e22a2612SMatthias Ringwald        flags |= property_flags['DYNAMIC']
415b3fcedb9SMatthias Ringwald        size = 2 + 2 + 2 + 2 + 2
416b3fcedb9SMatthias Ringwald        write_indent(fout)
417b3fcedb9SMatthias Ringwald        fout.write('// 0x%04x CLIENT_CHARACTERISTIC_CONFIGURATION\n' % (handle))
418b3fcedb9SMatthias Ringwald        write_indent(fout)
419b3fcedb9SMatthias Ringwald        write_16(fout, size)
420*e22a2612SMatthias Ringwald        write_16(fout, flags)
421b3fcedb9SMatthias Ringwald        write_16(fout, handle)
422b3fcedb9SMatthias Ringwald        write_16(fout, 0x2902)
423b3fcedb9SMatthias Ringwald        write_16(fout, 0)
424b3fcedb9SMatthias Ringwald        fout.write("\n")
425729074c4SMatthias Ringwald        defines_for_characteristics.append('#define ATT_CHARACTERISTIC_%s_CLIENT_CONFIGURATION_HANDLE 0x%04x' % (current_characteristic_uuid_string, handle))
426b3fcedb9SMatthias Ringwald        handle = handle + 1
427b3fcedb9SMatthias Ringwald
428b3fcedb9SMatthias Ringwald    if properties & property_flags['RELIABLE_WRITE']:
429b3fcedb9SMatthias Ringwald        size = 2 + 2 + 2 + 2 + 2
430b3fcedb9SMatthias Ringwald        write_indent(fout)
431b3fcedb9SMatthias Ringwald        fout.write('// 0x%04x CHARACTERISTIC_EXTENDED_PROPERTIES\n' % (handle))
432b3fcedb9SMatthias Ringwald        write_indent(fout)
433b3fcedb9SMatthias Ringwald        write_16(fout, size)
434*e22a2612SMatthias Ringwald        write_16(fout, read_only_anybody_flags)
435b3fcedb9SMatthias Ringwald        write_16(fout, handle)
436b3fcedb9SMatthias Ringwald        write_16(fout, 0x2900)
437b3fcedb9SMatthias Ringwald        write_16(fout, 1)   # Reliable Write
438b3fcedb9SMatthias Ringwald        fout.write("\n")
439b3fcedb9SMatthias Ringwald        handle = handle + 1
440b3fcedb9SMatthias Ringwald
441b3fcedb9SMatthias Ringwalddef parseCharacteristicUserDescription(fout, parts):
442b3fcedb9SMatthias Ringwald    global handle
443b3fcedb9SMatthias Ringwald    global total_size
444b3fcedb9SMatthias Ringwald    global current_characteristic_uuid_string
445b3fcedb9SMatthias Ringwald
446b3fcedb9SMatthias Ringwald    properties = parseProperties(parts[1])
447b3fcedb9SMatthias Ringwald    value      = parts[2]
448b3fcedb9SMatthias Ringwald
449b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2
450b3fcedb9SMatthias Ringwald    if is_string(value):
451b3fcedb9SMatthias Ringwald        size = size + len(value) - 2
452b3fcedb9SMatthias Ringwald    else:
453b3fcedb9SMatthias Ringwald        size = size + len(value.split())
454b3fcedb9SMatthias Ringwald
455*e22a2612SMatthias Ringwald    # use write, write permissions and encryption key size from attribute value and set READ_ANYBODY
456*e22a2612SMatthias Ringwald    flags  = write_permissions_and_key_size(properties)
457*e22a2612SMatthias Ringwald    flags |= properties & property_flags['WRITE']
458*e22a2612SMatthias Ringwald    flags |= property_flags['READ']
459*e22a2612SMatthias Ringwald
460b3fcedb9SMatthias Ringwald    write_indent(fout)
461b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x CHARACTERISTIC_USER_DESCRIPTION-%s\n' % (handle, '-'.join(parts[1:])))
462b3fcedb9SMatthias Ringwald    write_indent(fout)
463b3fcedb9SMatthias Ringwald    write_16(fout, size)
464*e22a2612SMatthias Ringwald    write_16(fout, flags)
465b3fcedb9SMatthias Ringwald    write_16(fout, handle)
466b3fcedb9SMatthias Ringwald    write_16(fout, 0x2901)
467b3fcedb9SMatthias Ringwald    if is_string(value):
468b3fcedb9SMatthias Ringwald        write_string(fout, value)
469b3fcedb9SMatthias Ringwald    else:
470b3fcedb9SMatthias Ringwald        write_sequence(fout,value)
471b3fcedb9SMatthias Ringwald    fout.write("\n")
472729074c4SMatthias Ringwald    defines_for_characteristics.append('#define ATT_CHARACTERISTIC_%s_USER_DESCRIPTION_HANDLE 0x%04x' % (current_characteristic_uuid_string, handle))
473b3fcedb9SMatthias Ringwald    handle = handle + 1
474b3fcedb9SMatthias Ringwald
475b3fcedb9SMatthias Ringwalddef parseServerCharacteristicConfiguration(fout, parts):
476b3fcedb9SMatthias Ringwald    global handle
477b3fcedb9SMatthias Ringwald    global total_size
478b3fcedb9SMatthias Ringwald    global current_characteristic_uuid_string
479b3fcedb9SMatthias Ringwald
480b3fcedb9SMatthias Ringwald    properties = parseProperties(parts[1])
481b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2
482b3fcedb9SMatthias Ringwald
483*e22a2612SMatthias Ringwald    # use write permissions and encryption key size from attribute value and set READ, WRITE, DYNAMIC, READ_ANYBODY
484*e22a2612SMatthias Ringwald    flags  = write_permissions_and_key_size(properties)
485*e22a2612SMatthias Ringwald    flags |= property_flags['READ']
486*e22a2612SMatthias Ringwald    flags |= property_flags['WRITE']
487*e22a2612SMatthias Ringwald    flags |= property_flags['DYNAMIC']
488*e22a2612SMatthias Ringwald
489b3fcedb9SMatthias Ringwald    write_indent(fout)
490b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x SERVER_CHARACTERISTIC_CONFIGURATION-%s\n' % (handle, '-'.join(parts[1:])))
491b3fcedb9SMatthias Ringwald    write_indent(fout)
492b3fcedb9SMatthias Ringwald    write_16(fout, size)
493*e22a2612SMatthias Ringwald    write_16(fout, flags)
494b3fcedb9SMatthias Ringwald    write_16(fout, handle)
495b3fcedb9SMatthias Ringwald    write_16(fout, 0x2903)
496b3fcedb9SMatthias Ringwald    fout.write("\n")
497729074c4SMatthias Ringwald    defines_for_characteristics.append('#define ATT_CHARACTERISTIC_%s_SERVER_CONFIGURATION_HANDLE 0x%04x' % (current_characteristic_uuid_string, handle))
498b3fcedb9SMatthias Ringwald    handle = handle + 1
499b3fcedb9SMatthias Ringwald
500b3fcedb9SMatthias Ringwalddef parseCharacteristicFormat(fout, parts):
501b3fcedb9SMatthias Ringwald    global handle
502b3fcedb9SMatthias Ringwald    global total_size
503b3fcedb9SMatthias Ringwald
504*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
505b3fcedb9SMatthias Ringwald
506b3fcedb9SMatthias Ringwald    identifier = parts[1]
507b3fcedb9SMatthias Ringwald    presentation_formats[identifier] = handle
508b3fcedb9SMatthias Ringwald    # print("format '%s' with handle %d\n" % (identifier, handle))
509b3fcedb9SMatthias Ringwald
510b3fcedb9SMatthias Ringwald    format     = parts[2]
511b3fcedb9SMatthias Ringwald    exponent   = parts[3]
512b3fcedb9SMatthias Ringwald    unit       = parseUUID(parts[4])
513b3fcedb9SMatthias Ringwald    name_space = parts[5]
514b3fcedb9SMatthias Ringwald    description = parseUUID(parts[6])
515b3fcedb9SMatthias Ringwald
516b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + 7
517b3fcedb9SMatthias Ringwald
518b3fcedb9SMatthias Ringwald    write_indent(fout)
519b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x CHARACTERISTIC_FORMAT-%s\n' % (handle, '-'.join(parts[1:])))
520b3fcedb9SMatthias Ringwald    write_indent(fout)
521b3fcedb9SMatthias Ringwald    write_16(fout, size)
522*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
523b3fcedb9SMatthias Ringwald    write_16(fout, handle)
524b3fcedb9SMatthias Ringwald    write_16(fout, 0x2904)
525b3fcedb9SMatthias Ringwald    write_sequence(fout, format)
526b3fcedb9SMatthias Ringwald    write_sequence(fout, exponent)
527b3fcedb9SMatthias Ringwald    write_uuid(unit)
528b3fcedb9SMatthias Ringwald    write_sequence(fout, name_space)
529b3fcedb9SMatthias Ringwald    write_uuid(description)
530b3fcedb9SMatthias Ringwald    fout.write("\n")
531b3fcedb9SMatthias Ringwald    handle = handle + 1
532b3fcedb9SMatthias Ringwald
533b3fcedb9SMatthias Ringwald
534b3fcedb9SMatthias Ringwalddef parseCharacteristicAggregateFormat(fout, parts):
535b3fcedb9SMatthias Ringwald    global handle
536b3fcedb9SMatthias Ringwald    global total_size
537b3fcedb9SMatthias Ringwald
538*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
539b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + (len(parts)-1) * 2
540b3fcedb9SMatthias Ringwald
541b3fcedb9SMatthias Ringwald    write_indent(fout)
542b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x CHARACTERISTIC_AGGREGATE_FORMAT-%s\n' % (handle, '-'.join(parts[1:])))
543b3fcedb9SMatthias Ringwald    write_indent(fout)
544b3fcedb9SMatthias Ringwald    write_16(fout, size)
545*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
546b3fcedb9SMatthias Ringwald    write_16(fout, handle)
547b3fcedb9SMatthias Ringwald    write_16(fout, 0x2905)
548b3fcedb9SMatthias Ringwald    for identifier in parts[1:]:
549b3fcedb9SMatthias Ringwald        format_handle = presentation_formats[identifier]
550b3fcedb9SMatthias Ringwald        if format == 0:
551b3fcedb9SMatthias Ringwald            print("ERROR: identifier '%s' in CHARACTERISTIC_AGGREGATE_FORMAT undefined" % identifier)
552b3fcedb9SMatthias Ringwald            sys.exit(1)
553b3fcedb9SMatthias Ringwald        write_16(fout, format_handle)
554b3fcedb9SMatthias Ringwald    fout.write("\n")
555b3fcedb9SMatthias Ringwald    handle = handle + 1
556b3fcedb9SMatthias Ringwald
557b3fcedb9SMatthias Ringwalddef parseReportReference(fout, parts):
558b3fcedb9SMatthias Ringwald    global handle
559b3fcedb9SMatthias Ringwald    global total_size
560b3fcedb9SMatthias Ringwald
561*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
562b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + 1 + 1
563b3fcedb9SMatthias Ringwald
564231a3c5dSMatthias Ringwald    report_id = parts[2]
565231a3c5dSMatthias Ringwald    report_type = parts[3]
566b3fcedb9SMatthias Ringwald
567b3fcedb9SMatthias Ringwald    write_indent(fout)
568b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x REPORT_REFERENCE-%s\n' % (handle, '-'.join(parts[1:])))
569b3fcedb9SMatthias Ringwald    write_indent(fout)
570b3fcedb9SMatthias Ringwald    write_16(fout, size)
571*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
572b3fcedb9SMatthias Ringwald    write_16(fout, handle)
573b3fcedb9SMatthias Ringwald    write_16(fout, 0x2908)
574b3fcedb9SMatthias Ringwald    write_sequence(fout, report_id)
575b3fcedb9SMatthias Ringwald    write_sequence(fout, report_type)
576b3fcedb9SMatthias Ringwald    fout.write("\n")
577b3fcedb9SMatthias Ringwald    handle = handle + 1
578b3fcedb9SMatthias Ringwald
579b3fcedb9SMatthias Ringwald
580b3fcedb9SMatthias Ringwalddef parseNumberOfDigitals(fout, parts):
581b3fcedb9SMatthias Ringwald    global handle
582b3fcedb9SMatthias Ringwald    global total_size
583b3fcedb9SMatthias Ringwald
584*e22a2612SMatthias Ringwald    read_only_anybody_flags = property_flags['READ'];
585b3fcedb9SMatthias Ringwald    size = 2 + 2 + 2 + 2 + 1
586b3fcedb9SMatthias Ringwald
587b3fcedb9SMatthias Ringwald    no_of_digitals = parts[1]
588b3fcedb9SMatthias Ringwald
589b3fcedb9SMatthias Ringwald    write_indent(fout)
590b3fcedb9SMatthias Ringwald    fout.write('// 0x%04x NUMBER_OF_DIGITALS-%s\n' % (handle, '-'.join(parts[1:])))
591b3fcedb9SMatthias Ringwald    write_indent(fout)
592b3fcedb9SMatthias Ringwald    write_16(fout, size)
593*e22a2612SMatthias Ringwald    write_16(fout, read_only_anybody_flags)
594b3fcedb9SMatthias Ringwald    write_16(fout, handle)
595b3fcedb9SMatthias Ringwald    write_16(fout, 0x2909)
596b3fcedb9SMatthias Ringwald    write_sequence(fout, no_of_digitals)
597b3fcedb9SMatthias Ringwald    fout.write("\n")
598b3fcedb9SMatthias Ringwald    handle = handle + 1
599b3fcedb9SMatthias Ringwald
60060b51a4cSMatthias Ringwalddef parseLines(fname_in, fin, fout):
601b3fcedb9SMatthias Ringwald    global handle
602b3fcedb9SMatthias Ringwald    global total_size
603b3fcedb9SMatthias Ringwald
604b165f97bSMatthias Ringwald    line_count = 0;
605b3fcedb9SMatthias Ringwald    for line in fin:
606b3fcedb9SMatthias Ringwald        line = line.strip("\n\r ")
607b165f97bSMatthias Ringwald        line_count += 1
608b3fcedb9SMatthias Ringwald
609b165f97bSMatthias Ringwald        if line.startswith("//"):
610b165f97bSMatthias Ringwald            fout.write("    //" + line.lstrip('/') + '\n')
611b165f97bSMatthias Ringwald            continue
612b165f97bSMatthias Ringwald
61360b51a4cSMatthias Ringwald        if line.startswith("#import"):
61460b51a4cSMatthias Ringwald            imported_file = ''
61560b51a4cSMatthias Ringwald            parts = re.match('#import\s+<(.*)>\w*',line)
61660b51a4cSMatthias Ringwald            if parts and len(parts.groups()) == 1:
61785a677ecSMatthias Ringwald                imported_file = btstack_root+'/src/ble/gatt-service/' + parts.groups()[0]
61860b51a4cSMatthias Ringwald            parts = re.match('#import\s+"(.*)"\w*',line)
61960b51a4cSMatthias Ringwald            if parts and len(parts.groups()) == 1:
62060b51a4cSMatthias Ringwald                imported_file = os.path.abspath(os.path.dirname(fname_in) + '/'+parts.groups()[0])
62160b51a4cSMatthias Ringwald            if len(imported_file) == 0:
62260b51a4cSMatthias Ringwald                print('ERROR: #import in file %s - line %u neither <name.gatt> nor "name.gatt" form', (fname_in, line_count))
62360b51a4cSMatthias Ringwald                continue
62460b51a4cSMatthias Ringwald
62560b51a4cSMatthias Ringwald            print("Importing %s" % imported_file)
62660b51a4cSMatthias Ringwald            try:
62760b51a4cSMatthias Ringwald                imported_fin = codecs.open (imported_file, encoding='utf-8')
62860b51a4cSMatthias Ringwald                fout.write('    // ' + line + ' -- BEGIN\n')
62960b51a4cSMatthias Ringwald                parseLines(imported_file, imported_fin, fout)
63060b51a4cSMatthias Ringwald                fout.write('    // ' + line + ' -- END\n')
63160b51a4cSMatthias Ringwald            except IOError as e:
63260b51a4cSMatthias Ringwald                print('ERROR: Import failed. Please check path.')
63360b51a4cSMatthias Ringwald
63460b51a4cSMatthias Ringwald            continue
63560b51a4cSMatthias Ringwald
63660b51a4cSMatthias Ringwald        if line.startswith("#TODO"):
63760b51a4cSMatthias Ringwald            print ("WARNING: #TODO in file %s - line %u not handled, skipping declaration:" % (fname_in, line_count))
638b165f97bSMatthias Ringwald            print ("'%s'" % line)
639b165f97bSMatthias Ringwald            fout.write("// " + line + '\n')
640b3fcedb9SMatthias Ringwald            continue
641b3fcedb9SMatthias Ringwald
642b3fcedb9SMatthias Ringwald        if len(line) == 0:
643b3fcedb9SMatthias Ringwald            continue
644b3fcedb9SMatthias Ringwald
645b3fcedb9SMatthias Ringwald        f = io.StringIO(line)
646b3fcedb9SMatthias Ringwald        parts_list = csv.reader(f, delimiter=',', quotechar='"')
647b3fcedb9SMatthias Ringwald
648b3fcedb9SMatthias Ringwald        for parts in parts_list:
649b3fcedb9SMatthias Ringwald            for index, object in enumerate(parts):
650b3fcedb9SMatthias Ringwald                parts[index] = object.strip().lstrip('"').rstrip('"')
651b3fcedb9SMatthias Ringwald
652b3fcedb9SMatthias Ringwald            if parts[0] == 'PRIMARY_SERVICE':
653b3fcedb9SMatthias Ringwald                parsePrimaryService(fout, parts)
654b3fcedb9SMatthias Ringwald                continue
655b3fcedb9SMatthias Ringwald
656b3fcedb9SMatthias Ringwald            if parts[0] == 'SECONDARY_SERVICE':
657b3fcedb9SMatthias Ringwald                parseSecondaryService(fout, parts)
658b3fcedb9SMatthias Ringwald                continue
659b3fcedb9SMatthias Ringwald
660b3fcedb9SMatthias Ringwald            if parts[0] == 'INCLUDE_SERVICE':
661b3fcedb9SMatthias Ringwald                parseIncludeService(fout, parts)
662b3fcedb9SMatthias Ringwald                continue
663b3fcedb9SMatthias Ringwald
664b3fcedb9SMatthias Ringwald            # 2803
665b3fcedb9SMatthias Ringwald            if parts[0] == 'CHARACTERISTIC':
666b3fcedb9SMatthias Ringwald                parseCharacteristic(fout, parts)
667b3fcedb9SMatthias Ringwald                continue
668b3fcedb9SMatthias Ringwald
669b3fcedb9SMatthias Ringwald            # 2900 Characteristic Extended Properties
670b3fcedb9SMatthias Ringwald
671b3fcedb9SMatthias Ringwald            # 2901
672b3fcedb9SMatthias Ringwald            if parts[0] == 'CHARACTERISTIC_USER_DESCRIPTION':
673b3fcedb9SMatthias Ringwald                parseCharacteristicUserDescription(fout, parts)
674b3fcedb9SMatthias Ringwald                continue
675b3fcedb9SMatthias Ringwald
676b165f97bSMatthias Ringwald
677b165f97bSMatthias Ringwald            # 2902 Client Characteristic Configuration - automatically included in Characteristic if
678b3fcedb9SMatthias Ringwald            # notification / indication is supported
679231a3c5dSMatthias Ringwald            if parts[0] == 'CLIENT_CHARACTERISTIC_CONFIGURATION':
680b165f97bSMatthias Ringwald                continue
681b3fcedb9SMatthias Ringwald
682b3fcedb9SMatthias Ringwald            # 2903
683b3fcedb9SMatthias Ringwald            if parts[0] == 'SERVER_CHARACTERISTIC_CONFIGURATION':
684b3fcedb9SMatthias Ringwald                parseServerCharacteristicConfiguration(fout, parts)
685b3fcedb9SMatthias Ringwald                continue
686b3fcedb9SMatthias Ringwald
687b3fcedb9SMatthias Ringwald            # 2904
688b3fcedb9SMatthias Ringwald            if parts[0] == 'CHARACTERISTIC_FORMAT':
689b3fcedb9SMatthias Ringwald                parseCharacteristicFormat(fout, parts)
690b3fcedb9SMatthias Ringwald                continue
691b3fcedb9SMatthias Ringwald
692b3fcedb9SMatthias Ringwald            # 2905
693b3fcedb9SMatthias Ringwald            if parts[0] == 'CHARACTERISTIC_AGGREGATE_FORMAT':
694b3fcedb9SMatthias Ringwald                parseCharacteristicAggregateFormat(fout, parts)
695b3fcedb9SMatthias Ringwald                continue
696b3fcedb9SMatthias Ringwald
697b3fcedb9SMatthias Ringwald            # 2906
698b3fcedb9SMatthias Ringwald            if parts[0] == 'VALID_RANGE':
699b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
700b3fcedb9SMatthias Ringwald                continue
701b3fcedb9SMatthias Ringwald
702b3fcedb9SMatthias Ringwald            # 2907
703b3fcedb9SMatthias Ringwald            if parts[0] == 'EXTERNAL_REPORT_REFERENCE':
704b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
705b3fcedb9SMatthias Ringwald                continue
706b3fcedb9SMatthias Ringwald
707b3fcedb9SMatthias Ringwald            # 2908
708b3fcedb9SMatthias Ringwald            if parts[0] == 'REPORT_REFERENCE':
709b3fcedb9SMatthias Ringwald                parseReportReference(fout, parts)
710b3fcedb9SMatthias Ringwald                continue
711b3fcedb9SMatthias Ringwald
712b3fcedb9SMatthias Ringwald            # 2909
713b3fcedb9SMatthias Ringwald            if parts[0] == 'NUMBER_OF_DIGITALS':
714b3fcedb9SMatthias Ringwald                parseNumberOfDigitals(fout, parts)
715b3fcedb9SMatthias Ringwald                continue
716b3fcedb9SMatthias Ringwald
717b3fcedb9SMatthias Ringwald            # 290A
718b3fcedb9SMatthias Ringwald            if parts[0] == 'VALUE_TRIGGER_SETTING':
719b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
720b3fcedb9SMatthias Ringwald                continue
721b3fcedb9SMatthias Ringwald
722b3fcedb9SMatthias Ringwald            # 290B
723b3fcedb9SMatthias Ringwald            if parts[0] == 'ENVIRONMENTAL_SENSING_CONFIGURATION':
724b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
725b3fcedb9SMatthias Ringwald                continue
726b3fcedb9SMatthias Ringwald
727b3fcedb9SMatthias Ringwald            # 290C
728b3fcedb9SMatthias Ringwald            if parts[0] == 'ENVIRONMENTAL_SENSING_MEASUREMENT':
729b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
730b3fcedb9SMatthias Ringwald                continue
731b3fcedb9SMatthias Ringwald
732b3fcedb9SMatthias Ringwald            # 290D
733b3fcedb9SMatthias Ringwald            if parts[0] == 'ENVIRONMENTAL_SENSING_TRIGGER_SETTING':
734b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
735b3fcedb9SMatthias Ringwald                continue
736b3fcedb9SMatthias Ringwald
737b3fcedb9SMatthias Ringwald            # 2906
738b3fcedb9SMatthias Ringwald            if parts[0] == 'VALID_RANGE':
739b3fcedb9SMatthias Ringwald                print("WARNING: %s not implemented yet\n" % (parts[0]))
740b3fcedb9SMatthias Ringwald                continue
741b3fcedb9SMatthias Ringwald
742b3fcedb9SMatthias Ringwald            print("WARNING: unknown token: %s\n" % (parts[0]))
743b3fcedb9SMatthias Ringwald
74460b51a4cSMatthias Ringwalddef parse(fname_in, fin, fname_out, fout):
74560b51a4cSMatthias Ringwald    global handle
74660b51a4cSMatthias Ringwald    global total_size
74760b51a4cSMatthias Ringwald
74860b51a4cSMatthias Ringwald    fout.write(header.format(fname_out, fname_in))
74960b51a4cSMatthias Ringwald    fout.write('{\n')
75060b51a4cSMatthias Ringwald
75160b51a4cSMatthias Ringwald    parseLines(fname_in, fin, fout)
75260b51a4cSMatthias Ringwald
753729074c4SMatthias Ringwald    serviceDefinitionComplete(fout)
754b3fcedb9SMatthias Ringwald    write_indent(fout)
755b3fcedb9SMatthias Ringwald    fout.write("// END\n");
756b3fcedb9SMatthias Ringwald    write_indent(fout)
757b3fcedb9SMatthias Ringwald    write_16(fout,0)
758b3fcedb9SMatthias Ringwald    fout.write("\n")
759b3fcedb9SMatthias Ringwald    total_size = total_size + 2
760b3fcedb9SMatthias Ringwald
761b3fcedb9SMatthias Ringwald    fout.write("}; // total size %u bytes \n" % total_size);
762b3fcedb9SMatthias Ringwald
763b3fcedb9SMatthias Ringwalddef listHandles(fout):
764b3fcedb9SMatthias Ringwald    fout.write('\n\n')
765b3fcedb9SMatthias Ringwald    fout.write('//\n')
766729074c4SMatthias Ringwald    fout.write('// list service handle ranges\n')
767729074c4SMatthias Ringwald    fout.write('//\n')
768729074c4SMatthias Ringwald    for define in defines_for_services:
769729074c4SMatthias Ringwald        fout.write(define)
770729074c4SMatthias Ringwald        fout.write('\n')
771729074c4SMatthias Ringwald    fout.write('\n')
772729074c4SMatthias Ringwald    fout.write('//\n')
773b3fcedb9SMatthias Ringwald    fout.write('// list mapping between characteristics and handles\n')
774b3fcedb9SMatthias Ringwald    fout.write('//\n')
775729074c4SMatthias Ringwald    for define in defines_for_characteristics:
776b3fcedb9SMatthias Ringwald        fout.write(define)
777b3fcedb9SMatthias Ringwald        fout.write('\n')
778b3fcedb9SMatthias Ringwald
779b3fcedb9SMatthias Ringwaldif (len(sys.argv) < 3):
780b3fcedb9SMatthias Ringwald    print(usage)
781b3fcedb9SMatthias Ringwald    sys.exit(1)
782b3fcedb9SMatthias Ringwaldtry:
783b165f97bSMatthias Ringwald    # read defines from bluetooth_gatt.h
784b165f97bSMatthias Ringwald    btstack_root = os.path.abspath(os.path.dirname(sys.argv[0]) + '/..')
785b165f97bSMatthias Ringwald    gen_path = btstack_root + '/src/bluetooth_gatt.h'
786b165f97bSMatthias Ringwald    bluetooth_gatt = read_defines(gen_path)
787b165f97bSMatthias Ringwald
788b3fcedb9SMatthias Ringwald    filename = sys.argv[2]
789b3fcedb9SMatthias Ringwald    fin  = codecs.open (sys.argv[1], encoding='utf-8')
790b3fcedb9SMatthias Ringwald    fout = open (filename, 'w')
791b3fcedb9SMatthias Ringwald    parse(sys.argv[1], fin, filename, fout)
792b3fcedb9SMatthias Ringwald    listHandles(fout)
793b3fcedb9SMatthias Ringwald    fout.close()
794b165f97bSMatthias Ringwald    print('Created %s' % filename)
795b3fcedb9SMatthias Ringwald
796b3fcedb9SMatthias Ringwaldexcept IOError as e:
797*e22a2612SMatthias Ringwald
798b3fcedb9SMatthias Ringwald    print(usage)
799b3fcedb9SMatthias Ringwald    sys.exit(1)
800b3fcedb9SMatthias Ringwald
801b3fcedb9SMatthias Ringwaldprint('Compilation successful!\n')
802