# Copyright (C) 2014-2016 Intel Corporation. All Rights Reserved. # # Permission is hereby granted, free of charge, to any person obtaining a # copy of this software and associated documentation files (the "Software"), # to deal in the Software without restriction, including without limitation # the rights to use, copy, modify, merge, publish, distribute, sublicense, # and/or sell copies of the Software, and to permit persons to whom the # Software is furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice (including the next # paragraph) shall be included in all copies or substantial portions of the # Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL # THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS # IN THE SOFTWARE. # Python source from __future__ import print_function import os import sys import re from gen_common import * def parse_event_fields(lines, idx, event_dict): """ Parses lines from a proto file that contain an event definition and stores it in event_dict """ fields = [] end_of_event = False # record all fields in event definition. # note: we don't check if there's a leading brace. while not end_of_event and idx < len(lines): line = lines[idx].rstrip() idx += 1 match = re.match(r'(\s*)([\w\*]+)(\s+)(counter\s+)*([\w]+)(\[\d+\])*', line) if match: field = { "type": match.group(2), "name": match.group(5), "size": int(match.group(6)[1:-1]) if match.group(6) else 1, "counter": True if match.group(4) else False } fields.append(field) end_of_event = re.match(r'(\s*)};', line) event_dict['fields'] = fields event_dict['num_fields'] = len(fields) return idx def parse_enums(lines, idx, event_dict): """ Parses lines from a proto file that contain an enum definition and stores it in event_dict """ enum_names = [] end_of_enum = False # record all enum values in enumeration # note: we don't check if there's a leading brace. while not end_of_enum and idx < len(lines): line = lines[idx].rstrip() idx += 1 preprocessor = re.search(r'#if|#endif', line) if not preprocessor: enum = re.match(r'(\s*)(\w+)(\s*)', line) if enum: enum_names.append(line) end_of_enum = re.match(r'(\s*)};', line) event_dict['names'] = enum_names return idx def parse_protos(files, verbose=False): """ Parses a proto file and returns a dictionary of event definitions """ protos = {} protos['events'] = {} # event dictionary containing events with their fields protos['event_names'] = [] # needed to keep events in order parsed. dict is not ordered. protos['event_map'] = {} # dictionary to map event ids to event names protos['enums'] = {} protos['enum_names'] = [] eventId = 0 if type(files) is not list: files = [files] for filename in files: if verbose: print("Parsing proto file: %s" % os.path.normpath(filename)) with open(filename, 'r') as f: lines=f.readlines() idx = 0 raw_text = [] while idx < len(lines): line = lines[idx].rstrip() idx += 1 # search for event definitions. match = re.match(r'(\s*)event(\s*)(\w+)', line) if match: eventId += 1 event_name = match.group(3) protos["event_names"].append(event_name) protos["events"][event_name] = {} protos["events"][event_name]["event_id"] = eventId protos["event_map"][eventId] = event_name idx = parse_event_fields(lines, idx, protos["events"][event_name]) # search for enums. match = re.match(r'(\s*)enum(\s*)(\w+)', line) if match: enum_name = match.group(3) protos["enum_names"].append(enum_name) protos["enums"][enum_name] = {} idx = parse_enums(lines, idx, protos["enums"][enum_name]) return protos def main(): # Parse args... parser = ArgumentParser() parser.add_argument("--proto", "-p", dest="protos", nargs='+', help="Path to all proto file(s) to process. Accepts one or more paths (i.e. events.proto and events_private.proto)", required=True) parser.add_argument("--output-dir", help="Output dir (defaults to ./codegen). Will create folder if it does not exist.", required=False, default="codegen") parser.add_argument("--verbose", "-v", help="Verbose", action="store_true") args = parser.parse_args() if not os.path.exists(args.output_dir): MakeDir(args.output_dir) for f in args.protos: if not os.path.exists(f): print('Error: Could not find proto file %s' % f, file=sys.stderr) return 1 # Parse each proto file and add to protos container protos = parse_protos(args.protos, args.verbose) files = [ ["gen_ar_event.hpp", ""], ["gen_ar_event.cpp", ""], ["gen_ar_eventhandler.hpp", "gen_ar_event.hpp"], ["gen_ar_eventhandlerfile.hpp", "gen_ar_eventhandler.hpp"] ] rval = 0 try: # Delete existing files for f in files: filename = f[0] output_fullpath = os.path.join(args.output_dir, filename) if os.path.exists(output_fullpath): if args.verbose: print("Deleting existing file: %s" % output_fullpath) os.remove(output_fullpath) # Generate files from templates print("Generating c++ from proto files...") for f in files: filename = f[0] event_header = f[1] curdir = os.path.dirname(os.path.abspath(__file__)) template_file = os.path.join(curdir, 'templates', filename) output_fullpath = os.path.join(args.output_dir, filename) if args.verbose: print("Generating: %s" % output_fullpath) MakoTemplateWriter.to_file(template_file, output_fullpath, cmdline=sys.argv, filename=filename, protos=protos, event_header=event_header) except Exception as e: print(e) rval = 1 return rval if __name__ == '__main__': sys.exit(main())