| Index: compile |
| =================================================================== |
| new file mode 100755 |
| --- /dev/null |
| +++ b/compile |
| @@ -0,0 +1,265 @@ |
| +#!/usr/bin/env python |
| + |
| +import json |
| +import os |
| +import re |
| +import subprocess |
| +import warnings |
| + |
| +EMSCRIPTEN_PATH = '../emscripten' |
| +SOURCE_DIR = './compiled' |
| +SOURCE_FILES = [ |
| + os.path.join(SOURCE_DIR, f) |
| + for f in os.listdir(SOURCE_DIR) |
| + if f.endswith('.cpp') |
| +] |
| +API_FILE = os.path.join(SOURCE_DIR, 'api.cpp') |
| +API_OUTPUT = os.path.join(SOURCE_DIR, 'api.js') |
| +COMPILER_OUTPUT = './lib/compiled.js' |
| +GENERATION_PARAMS = { |
| + 'SHELL_FILE': "'%s'" % os.path.abspath(os.path.join(SOURCE_DIR, 'shell.js')), |
| + 'TOTAL_MEMORY': 16*1024*1024, |
| + 'TOTAL_STACK': 1*1024*1024, |
| + 'ALLOW_MEMORY_GROWTH': 1, |
| + 'NO_EXIT_RUNTIME': 1, |
| + 'DISABLE_EXCEPTION_CATCHING': 0, |
| + 'NO_DYNAMIC_EXECUTION': 1, |
| + 'NO_BROWSER': 1, |
| + 'NO_FILESYSTEM': 1, |
| + 'INVOKE_RUN': 0, |
| + 'NODE_STDOUT_FLUSH_WORKAROUND': 0, |
| +} |
| +DEFINES = ['DEBUG'] |
| +ADDITIONAL_PARAMS = ['-O3', '-m32', '-std=gnu++11', '--memory-init-file', '0', |
| + '--emit-symbol-map'] |
| + |
| +def getenv(): |
| + path = [] |
| + env = {} |
| + output = subprocess.check_output([ |
| + '/bin/sh', '-c', os.path.join(EMSCRIPTEN_PATH, 'emsdk_env.sh')]) |
| + for line in output.splitlines(): |
| + match = re.search(r'^\s*PATH\s*\+=\s*(.*)', line) |
| + if match: |
| + path.append(match.group(1)) |
| + match = re.search(r'^\s*(\w+)\s*=\s*(.*)', line) |
| + if match: |
| + env[match.group(1)] = match.group(2) |
| + env['PATH'] = ':'.join([os.environ['PATH']] + path) |
| + return env |
| + |
| +def generate_api(env): |
| + params = [os.path.join(env['EMSCRIPTEN'], 'emcc'), '-E', API_FILE] |
| + params.extend(ADDITIONAL_PARAMS) |
| + output = subprocess.check_output(params, env=env) |
| + |
| + differentiators = {} |
| + differentiator = None |
| + |
| + cls = None |
| + method = None |
| + |
| + def wrap_call(func, is_instance, result_type, string_args): |
| + prefix = '''\ |
| +function() |
| +{ |
| + var params = Array.prototype.slice.apply(arguments); |
| +''' |
| + suffix = '''\ |
| + return result; |
| +}''' |
| + |
| + if result_type != 'primitive' or len(string_args): |
| + prefix += ' var sp = Runtime.stackSave();\n' |
| + suffix = ' Runtime.stackRestore(sp);\n' + suffix |
| + |
| + for pos in string_args: |
| + prefix += ' params[%i] = createString(params[%i]);\n' % (pos, pos) |
| + |
| + if is_instance: |
| + prefix += ' params.unshift(this._pointer);\n' |
| + |
| + if result_type == 'primitive': |
| + prefix += ' var result = _%s.apply(null, params);\n' % func |
| + elif result_type == 'string': |
| + prefix += ' params.unshift(createString());\n' |
| + prefix += ' _%s.apply(null, params);\n' % func |
| + prefix += ' var result = getStringData(params[0]);\n' |
| + else: |
| + prefix += ' params.unshift(_CreatePointer());\n' |
| + prefix += ' _%s.apply(null, params);\n' % func |
| + if result_type in differentiators: |
| + prefix += ' var type = _%s(params[0]);\n' % differentiator['func'] |
| + prefix += ' if (type in %s_mapping)\n' % result_type |
| + prefix += ' var result = new (exports[%s_mapping[type]])(params[0]);\n' % result_type |
| + prefix += ' else\n' |
| + prefix += ' throw new Error("Unexpected %s type: " + type);\n' % result_type |
| + else: |
| + prefix += ' var result = %s(params[0]);\n' % result_type |
| + |
| + return prefix + suffix |
| + |
| + def property_descriptor(property): |
| + if property['type'] == 'static': |
| + return 'value: %s' % property['getter'] |
| + |
| + result = 'get: %s' % wrap_call(property['getter'], True, property['type'], []) |
| + if property['setter'] is not None: |
| + string_args = [0] if property['type'] == 'string' else [] |
| + result += ', set: %s' % wrap_call(property['setter'], True, 'primitive', string_args) |
| + return result |
| + |
| + def method_descriptor(method): |
| + return wrap_call(method['func'], method['type'] == 'instance', |
| + method['result'], method['string_args']) |
| + |
| + def write_class(file, cls): |
| + name = cls['name'] |
| + if name in differentiators: |
| + print >>file, 'var %s_mapping = %s;' % ( |
| + name, json.dumps(differentiators[name]['mapping'], sort_keys=True)) |
| + |
| + print >>file, 'exports.%s = createClass(%s);' % ( |
| + name, 'exports.' + cls['superclass'] if cls['superclass'] else '') |
| + for property in cls['properties']: |
| + print >>file, 'Object.defineProperty(exports.%s.prototype, "%s", {%s});' % ( |
| + name, property['name'], property_descriptor(property)) |
| + for method in cls['methods']: |
| + obj = ('exports.%s' if method['type'] == 'class' else 'exports.%s.prototype') % name |
| + print >>file, '%s.%s = %s;' % ( |
| + obj, method['name'], method_descriptor(method)) |
| + for initializer in cls['class_initializers']: |
| + print >>file, '_%s();' % initializer |
| + |
| + def handle_class(name): |
| + if cls is not None: |
| + write_class(file, cls) |
| + differentiator = None |
| + return { |
| + 'name': command[1], |
| + 'superclass': None, |
| + 'properties': [], |
| + 'methods': [], |
| + 'class_initializers': [], |
| + } |
| + |
| + def handle_superclass(name): |
| + if cls is None: |
| + warnings.warn('Superclass declared outside a class: ' + name) |
| + return |
| + differentiator = None |
| + cls['superclass'] = name |
| + |
| + def handle_class_init(func): |
| + if cls is None: |
| + warnings.warn('Class initializer declared outside a class: ' + func) |
| + return |
| + differentiator = None |
| + method = None |
| + cls['class_initializers'].append(func) |
| + |
| + def handle_differentiator(cls, func): |
| + differentiator = {'func': func, 'mapping': {}} |
| + differentiators[cls] = differentiator |
| + return differentiator |
| + |
| + def handle_differentiator_mapping(value, subclass): |
| + if differentiator is None: |
| + warnings.warn('Differentiator mapping declared without a differentiator: ' + subclass) |
| + return |
| + differentiator['mapping'][value] = subclass |
| + |
| + def handle_property(type, name, getter, setter): |
| + if cls is None: |
| + warnings.warn('Property declared outside a class: ' + name) |
| + return |
| + method = None |
| + differentiator = None |
| + cls['properties'].append({ |
| + 'type': type, |
| + 'name': name, |
| + 'getter': getter, |
| + 'setter': setter, |
| + }) |
| + |
| + def handle_method(type, name, func): |
| + if cls is None: |
| + warnings.warn('Method declared outside a class: ' + name) |
| + return |
| + differentiator = None |
| + method = { |
| + 'type': type, |
| + 'name': name, |
| + 'func': func, |
| + 'result': 'primitive', |
| + 'string_args': [], |
| + } |
| + cls['methods'].append(method) |
| + return method |
| + |
| + def handle_method_result(type): |
| + if method is None: |
| + warnings.warn('Method result declared without a method definition') |
| + return |
| + method['result'] = type |
| + |
| + def handle_string_arg(pos): |
| + if method is None: |
| + warnings.warn('Argument type declared without a method definition') |
| + return |
| + method['string_args'].append(int(pos)) |
| + |
| + with open(API_OUTPUT, 'w') as file: |
| + for line in output.splitlines(): |
| + match = re.search(r'#pragma\s+comment\((.+?)\)', line) |
| + if match: |
| + command = match.group(1).strip().split() |
| + if command[0] == 'class': |
| + cls = handle_class(command[1]) |
| + elif command[0] == 'augments': |
| + handle_superclass(command[1]) |
| + elif command[0] == 'class_init': |
| + handle_class_init(command[1]) |
| + elif command[0] == 'differentiator': |
| + differentiator = handle_differentiator(command[1], command[2]) |
| + elif command[0] == 'differentiator_mapping': |
| + handle_differentiator_mapping(command[1], command[2]) |
| + elif command[0] == 'property': |
| + handle_property('primitive', command[1], command[2], command[3] if len(command) > 3 else None) |
| + elif command[0] == 'string_property': |
| + handle_property('string', command[1], command[2], command[3] if len(command) > 3 else None) |
| + elif command[0] == 'static_property': |
| + handle_property('static', command[1], command[2], None) |
| + elif command[0] == 'method': |
| + method = handle_method('instance', command[1], command[2]) |
| + elif command[0] == 'class_method': |
| + method = handle_method('class', command[1], command[2]) |
| + elif command[0] == 'string_result': |
| + handle_method_result('string') |
| + elif command[0] == 'pointer_result': |
| + handle_method_result(command[1]) |
| + elif command[0] == 'string_arg': |
| + handle_string_arg(command[1]) |
| + else: |
| + warnings.warn('Unknown declaration: ' + str(command)) |
| + |
| + if cls is not None: |
| + write_class(file, cls) |
| + |
| +def run_compiler(env): |
| + params = [ |
| + os.path.join(env['EMSCRIPTEN'], 'emcc'), |
| + '-o', COMPILER_OUTPUT, |
| + '--post-js', API_OUTPUT, |
| + ] |
| + params.extend(SOURCE_FILES) |
| + params.extend('-D' + flag for flag in DEFINES) |
| + for key, value in GENERATION_PARAMS.iteritems(): |
| + params.extend(['-s', '%s=%s' % (key, str(value))]) |
| + params.extend(ADDITIONAL_PARAMS) |
| + subprocess.check_call(params, env=env) |
| + |
| +if __name__ == '__main__': |
| + env = getenv() |
| + generate_api(env) |
| + run_compiler(env) |