15c544019SMatthias Ringwald#!/usr/bin/env python3 23338b9aeSMilanka Ringwald# BlueKitchen GmbH (c) 2014 33338b9aeSMilanka Ringwald 43338b9aeSMilanka Ringwaldimport re 53338b9aeSMilanka Ringwaldimport os 63338b9aeSMilanka Ringwaldimport sys 73338b9aeSMilanka Ringwald 83338b9aeSMilanka Ringwald# paths 93338b9aeSMilanka Ringwaldbluetooth_h_path = 'src/bluetooth.h' 103338b9aeSMilanka Ringwaldbtstack_defines_h_path = 'src/btstack_defines.h' 113338b9aeSMilanka Ringwalddaemon_cmds_c_path = 'platform/daemon/src/daemon_cmds.c' 12*9b8ab7daSMatthias Ringwalddaemon_cmds_h_path = 'platform/daemon/src/daemon_cmds.h' 133338b9aeSMilanka Ringwaldhci_cmds_c_path = 'src/hci_cmd.c' 143338b9aeSMilanka Ringwaldhci_cmds_h_path = 'src/hci_cmd.h' 153338b9aeSMilanka Ringwaldhci_h_path = 'src/hci.h' 163338b9aeSMilanka Ringwald 173338b9aeSMilanka Ringwaldbtstack_root = os.path.abspath(os.path.dirname(sys.argv[0]) + '/..') 183338b9aeSMilanka Ringwaldprint ("BTstack root %s" % btstack_root) 193338b9aeSMilanka Ringwald 203338b9aeSMilanka Ringwalddef set_btstack_root(path): 213338b9aeSMilanka Ringwald global btstack_root 223338b9aeSMilanka Ringwald btstack_root = path 233338b9aeSMilanka Ringwald 243338b9aeSMilanka Ringwalddef assert_dir(path): 253338b9aeSMilanka Ringwald if not os.access(path, os.R_OK): 263338b9aeSMilanka Ringwald os.makedirs(path) 273338b9aeSMilanka Ringwald 283338b9aeSMilanka Ringwalddef cap(x): 293338b9aeSMilanka Ringwald if x.lower() == 'btstack': 303338b9aeSMilanka Ringwald return 'BTstack' 313338b9aeSMilanka Ringwald acronyms = ['ATT', 'GAP', 'GATT', 'HCI', 'L2CAP', 'LE', 'RFCOMM', 'SM', 'SDP', 'UUID16', 'UUID128', 'HSP', 'HFP', 'ANCS'] 323338b9aeSMilanka Ringwald if x.upper() in acronyms: 333338b9aeSMilanka Ringwald return x.upper() 343338b9aeSMilanka Ringwald return x.capitalize() 353338b9aeSMilanka Ringwald 363338b9aeSMilanka Ringwalddef camel_case(name): 373338b9aeSMilanka Ringwald return ''.join(map(cap, name.split('_'))) 383338b9aeSMilanka Ringwald 393338b9aeSMilanka Ringwalddef camel_case_var(name): 403338b9aeSMilanka Ringwald if name in ['uuid128', 'uuid16']: 413338b9aeSMilanka Ringwald return name 423338b9aeSMilanka Ringwald camel = camel_case(name) 433338b9aeSMilanka Ringwald return camel[0].lower() + camel[1:] 443338b9aeSMilanka Ringwald 453338b9aeSMilanka Ringwalddef read_defines(infile): 463338b9aeSMilanka Ringwald defines = dict() 473338b9aeSMilanka Ringwald with open (infile, 'rt') as fin: 483338b9aeSMilanka Ringwald for line in fin: 493338b9aeSMilanka Ringwald parts = re.match('#define\s+(\w+)\s+(\w*)',line) 503338b9aeSMilanka Ringwald if parts and len(parts.groups()) == 2: 513338b9aeSMilanka Ringwald (key, value) = parts.groups() 523338b9aeSMilanka Ringwald defines[key] = value 533338b9aeSMilanka Ringwald return defines 543338b9aeSMilanka Ringwald 553338b9aeSMilanka Ringwalddef parse_defines(): 563338b9aeSMilanka Ringwald global btstack_root 573338b9aeSMilanka Ringwald defines = dict() 583338b9aeSMilanka Ringwald defines.update(read_defines(btstack_root + '/' + hci_cmds_h_path)) 593338b9aeSMilanka Ringwald defines.update(read_defines(btstack_root + '/' + hci_h_path)) 603338b9aeSMilanka Ringwald defines.update(read_defines(btstack_root + '/' + bluetooth_h_path)) 613338b9aeSMilanka Ringwald defines.update(read_defines(btstack_root + '/' + btstack_defines_h_path)) 623338b9aeSMilanka Ringwald return defines 633338b9aeSMilanka Ringwald 643338b9aeSMilanka Ringwalddef my_parse_events(path): 653338b9aeSMilanka Ringwald events = [] 663338b9aeSMilanka Ringwald subevents = [] 673338b9aeSMilanka Ringwald params = [] 683338b9aeSMilanka Ringwald event_types = set() 693338b9aeSMilanka Ringwald format = None 703338b9aeSMilanka Ringwald with open (path, 'rt') as fin: 713338b9aeSMilanka Ringwald for line in fin: 723338b9aeSMilanka Ringwald parts = re.match('.*@format\s*(\w*)\s*', line) 733338b9aeSMilanka Ringwald if parts and len(parts.groups()) == 1: 743338b9aeSMilanka Ringwald format = parts.groups()[0] 753338b9aeSMilanka Ringwald parts = re.match('.*@param\s*(\w*)\s*', line) 763338b9aeSMilanka Ringwald if parts and len(parts.groups()) == 1: 773338b9aeSMilanka Ringwald param = parts.groups()[0] 783338b9aeSMilanka Ringwald params.append(param) 793338b9aeSMilanka Ringwald parts = re.match('\s*#define\s+(\w+)\s+(\w*)',line) 803338b9aeSMilanka Ringwald if parts and len(parts.groups()) == 2: 813338b9aeSMilanka Ringwald (key, value) = parts.groups() 823338b9aeSMilanka Ringwald if format != None: 833338b9aeSMilanka Ringwald # renaming needed by Java Binding (... subevents are just enumerated with others due to event factory) 843338b9aeSMilanka Ringwald if "_subevent_" in key.lower(): 853338b9aeSMilanka Ringwald subevents.append((value, key, format, params)) 863338b9aeSMilanka Ringwald else: 873338b9aeSMilanka Ringwald events.append((value, key, format, params)) 883338b9aeSMilanka Ringwald event_types.add(key) 893338b9aeSMilanka Ringwald params = [] 903338b9aeSMilanka Ringwald format = None 913338b9aeSMilanka Ringwald return (events, subevents, event_types) 923338b9aeSMilanka Ringwald 933338b9aeSMilanka Ringwalddef parse_events(): 943338b9aeSMilanka Ringwald global btstack_root 953338b9aeSMilanka Ringwald 963338b9aeSMilanka Ringwald # parse bluetooth.h to get used events 973338b9aeSMilanka Ringwald (bluetooth_events, bluetooth_subevents, bluetooth_event_types) = my_parse_events(btstack_root + '/' + bluetooth_h_path) 983338b9aeSMilanka Ringwald 993338b9aeSMilanka Ringwald # parse btstack_defines to get events 1003338b9aeSMilanka Ringwald (btstack_events, btstack_subevents, btstack_event_types) = my_parse_events(btstack_root + '/' + btstack_defines_h_path) 1013338b9aeSMilanka Ringwald 1023338b9aeSMilanka Ringwald # concat lists 1033338b9aeSMilanka Ringwald (events, subvents, event_types) = (bluetooth_events + btstack_events, bluetooth_subevents + btstack_subevents, bluetooth_event_types | btstack_event_types) 1043338b9aeSMilanka Ringwald 1053338b9aeSMilanka Ringwald return (events, subvents, event_types) 106*9b8ab7daSMatthias Ringwalddef my_parse_opcodes(infile, convert_to_camel_case): 107*9b8ab7daSMatthias Ringwald opcodes = {} 108*9b8ab7daSMatthias Ringwald with open (infile, 'rt') as fin: 109*9b8ab7daSMatthias Ringwald for line in fin: 110*9b8ab7daSMatthias Ringwald definition = re.match('\s*(DAEMON_OPCODE_\w+)\s*=\s*DAEMON_OPCODE\s*\(\s*(\w+)\s*\).*', line) 111*9b8ab7daSMatthias Ringwald if definition: 112*9b8ab7daSMatthias Ringwald (opcode, daemon_ocf) = definition.groups() 113*9b8ab7daSMatthias Ringwald # opcodes.append((opcode, 'OGF_BTSTACK', daemon_ocf)) 114*9b8ab7daSMatthias Ringwald opcodes[opcode] = ('OGF_BTSTACK', daemon_ocf) 115*9b8ab7daSMatthias Ringwald definition = re.match('\s*(HCI_OPCODE_\w+)\s*=\s*HCI_OPCODE\s*\(\s*(\w+)\s*,\s*(\w+)\s*\).*', line) 116*9b8ab7daSMatthias Ringwald if definition: 117*9b8ab7daSMatthias Ringwald (opcode, ogf, ocf) = definition.groups() 118*9b8ab7daSMatthias Ringwald # opcodes.append((opcode, ogf, ocf)) 119*9b8ab7daSMatthias Ringwald opcodes[opcode] = (ogf, ocf) 120*9b8ab7daSMatthias Ringwald return opcodes 1213338b9aeSMilanka Ringwald 122*9b8ab7daSMatthias Ringwalddef parse_opcodes(camel_case=True): 123*9b8ab7daSMatthias Ringwald global btstack_root 124*9b8ab7daSMatthias Ringwald opcodes = {} 125*9b8ab7daSMatthias Ringwald opcodes.update(my_parse_opcodes(btstack_root + '/' + hci_cmds_h_path, camel_case)) 126*9b8ab7daSMatthias Ringwald opcodes.update(my_parse_opcodes(btstack_root + '/' + daemon_cmds_h_path, camel_case)) 127*9b8ab7daSMatthias Ringwald return opcodes 128*9b8ab7daSMatthias Ringwald 129*9b8ab7daSMatthias Ringwalddef my_parse_commands(infile, opcodes, convert_to_camel_case): 1303338b9aeSMilanka Ringwald commands = [] 1313338b9aeSMilanka Ringwald with open (infile, 'rt') as fin: 1323338b9aeSMilanka Ringwald 1333338b9aeSMilanka Ringwald params = [] 1343338b9aeSMilanka Ringwald for line in fin: 1353338b9aeSMilanka Ringwald 1363338b9aeSMilanka Ringwald parts = re.match('.*@param\s*(\w*)\s*', line) 1373338b9aeSMilanka Ringwald if parts and len(parts.groups()) == 1: 1383338b9aeSMilanka Ringwald param = parts.groups()[0] 1399c806232SMatthias Ringwald if convert_to_camel_case: 1403ddaf9c7SMatthias Ringwald param = camel_case_var(param) 1413ddaf9c7SMatthias Ringwald else: 1423ddaf9c7SMatthias Ringwald param = param.lower() 1433ddaf9c7SMatthias Ringwald params.append(param) 1443338b9aeSMilanka Ringwald continue 1453338b9aeSMilanka Ringwald 1463338b9aeSMilanka Ringwald declaration = re.match('const\s+hci_cmd_t\s+(\w+)[\s=]+', line) 1473338b9aeSMilanka Ringwald if declaration: 1483ddaf9c7SMatthias Ringwald command_name = declaration.groups()[0] 1493ddaf9c7SMatthias Ringwald # drop _cmd suffix for daemon commands 1503ddaf9c7SMatthias Ringwald if command_name.endswith('_cmd'): 1513ddaf9c7SMatthias Ringwald command_name = command_name[:-len('_cmd')] 1529c806232SMatthias Ringwald if convert_to_camel_case: 1533ddaf9c7SMatthias Ringwald command_name = camel_case(command_name) 1543ddaf9c7SMatthias Ringwald else: 1553ddaf9c7SMatthias Ringwald command_name = command_name.lower() 1563338b9aeSMilanka Ringwald continue 1573338b9aeSMilanka Ringwald 158*9b8ab7daSMatthias Ringwald # HCI_OPCODE or DAEMON_OPCODE definition 159*9b8ab7daSMatthias Ringwald definition = re.match('\s*(HCI_OPCODE_\w+|DAEMON_OPCODE_\w+)\s*,\s\\"(\w*)\\".*', line) 1603338b9aeSMilanka Ringwald if definition: 161*9b8ab7daSMatthias Ringwald (opcode, format) = definition.groups() 1623338b9aeSMilanka Ringwald if len(params) != len(format): 1633338b9aeSMilanka Ringwald params = [] 1643338b9aeSMilanka Ringwald arg_counter = 1 1653338b9aeSMilanka Ringwald for f in format: 1663338b9aeSMilanka Ringwald arg_name = 'arg%u' % arg_counter 1673338b9aeSMilanka Ringwald params.append(arg_name) 1683338b9aeSMilanka Ringwald arg_counter += 1 169*9b8ab7daSMatthias Ringwald (ogf, ocf) = opcodes[opcode] 1703338b9aeSMilanka Ringwald commands.append((command_name, ogf, ocf, format, params)) 1713338b9aeSMilanka Ringwald params = [] 1723338b9aeSMilanka Ringwald continue 173*9b8ab7daSMatthias Ringwald 1743338b9aeSMilanka Ringwald return commands 1753338b9aeSMilanka Ringwald 1763ddaf9c7SMatthias Ringwalddef parse_commands(camel_case=True): 1773338b9aeSMilanka Ringwald global btstack_root 178*9b8ab7daSMatthias Ringwald opcodes = parse_opcodes(camel_case) 179*9b8ab7daSMatthias Ringwald 1803338b9aeSMilanka Ringwald commands = [] 181*9b8ab7daSMatthias Ringwald commands += my_parse_commands(btstack_root + '/' + hci_cmds_c_path, opcodes, camel_case) 182*9b8ab7daSMatthias Ringwald commands += my_parse_commands(btstack_root + '/' + daemon_cmds_c_path, opcodes, camel_case) 1833338b9aeSMilanka Ringwald return commands 184*9b8ab7daSMatthias Ringwald 185*9b8ab7daSMatthias Ringwalddef parse_daemon_commands(camel_case=True): 186*9b8ab7daSMatthias Ringwald global btstack_root 187*9b8ab7daSMatthias Ringwald opcodes = parse_opcodes(camel_case) 188*9b8ab7daSMatthias Ringwald return my_parse_commands(btstack_root + '/' + daemon_cmds_c_path, opcodes, camel_case) 189*9b8ab7daSMatthias Ringwald 190*9b8ab7daSMatthias Ringwalddef print_opcode_enum(commands): 191*9b8ab7daSMatthias Ringwald print("typedef enum {") 192*9b8ab7daSMatthias Ringwald for command in commands: 193*9b8ab7daSMatthias Ringwald (name, opcode, format, params) = command 194*9b8ab7daSMatthias Ringwald print(" DAEMON_OPCODE_%s = HCI_OPCODE (%s, %s)," % (name.upper(), ogf, ocf)) 195*9b8ab7daSMatthias Ringwald print("} daemon_opcode_t;") 196*9b8ab7daSMatthias Ringwald 197*9b8ab7daSMatthias Ringwald 198