summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--.gitignore2
-rw-r--r--.gitmodules3
-rw-r--r--test/binary/basic.txt23
-rwxr-xr-xtest/gen-wasm.py461
-rwxr-xr-xtest/run-gen-wasm.py73
-rwxr-xr-xtest/run-roundtrip.py2
-rwxr-xr-xtest/run-tests.py7
-rw-r--r--test/utils.py16
m---------third_party/ply0
9 files changed, 577 insertions, 10 deletions
diff --git a/.gitignore b/.gitignore
index be8d47aa..41a84217 100644
--- a/.gitignore
+++ b/.gitignore
@@ -3,6 +3,8 @@
/third_party/v8/depot_tools
/fuzz-out
/emscripten
+/test/parser.out
+/test/parsetab.py
*.pyc
#Visual Studio Products
diff --git a/.gitmodules b/.gitmodules
index 706771c8..55067ee9 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -13,3 +13,6 @@
[submodule "third_party/CMakeXFind"]
path = third_party/CMakeXFind
url = https://github.com/julp/CMakeXFind
+[submodule "third_party/ply"]
+ path = third_party/ply
+ url = https://github.com/dabeaz/ply
diff --git a/test/binary/basic.txt b/test/binary/basic.txt
new file mode 100644
index 00000000..928f9e02
--- /dev/null
+++ b/test/binary/basic.txt
@@ -0,0 +1,23 @@
+;;; TOOL: run-gen-wasm
+magic
+version
+section("signatures") { count[1] params[0] result[i32] }
+section("function_signatures") { count[1] type[0] }
+section("export_table") { count[1] func[0] str("main") }
+section("function_bodies") {
+ count[1]
+ func {
+ locals[0]
+ return
+ i32.const
+ leb_i32(-420)
+ }
+}
+(;; STDOUT ;;;
+(module
+ (type (;0;) (func (result i32)))
+ (func (;0;) (type 0) (result i32)
+ (return
+ (i32.const -420)))
+ (export "main" 0))
+;;; STDOUT ;;)
diff --git a/test/gen-wasm.py b/test/gen-wasm.py
new file mode 100755
index 00000000..de0fa858
--- /dev/null
+++ b/test/gen-wasm.py
@@ -0,0 +1,461 @@
+#!/usr/bin/env python
+#
+# Copyright 2016 WebAssembly Community Group participants
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+import argparse
+import cStringIO
+import os
+import struct
+import sys
+
+from utils import Error, Hexdump
+
+SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
+ROOT_DIR = os.path.dirname(SCRIPT_DIR)
+PLY_DIR = os.path.join(ROOT_DIR, 'third_party', 'ply')
+
+sys.path.append(PLY_DIR)
+
+try:
+ import ply.lex as lex
+ import ply.yacc as yacc
+except ImportError:
+ raise Error('Unable to import ply. Did you run "git submodule update"?')
+
+
+## ply stuff ###################################################################
+NAMED_VALUES = {
+ 'void': 0,
+ 'i32': 1,
+ 'i64': 2,
+ 'f32': 3,
+ 'f64': 4,
+ 'magic': (0, 0x61, 0x73, 0x6d),
+ 'version': (0xa, 0, 0, 0),
+
+ "nop": 0x00,
+ "block": 0x01,
+ "loop": 0x02,
+ "if": 0x03,
+ "if_else": 0x04,
+ "select": 0x05,
+ "br": 0x06,
+ "br_if": 0x07,
+ "br_table": 0x08,
+ "i8.const": 0x09,
+ "i32.const": 0x0a,
+ "i64.const": 0x0b,
+ "f64.const": 0x0c,
+ "f32.const": 0x0d,
+ "get_local": 0x0e,
+ "set_local": 0x0f,
+ "call": 0x12,
+ "call_indirect": 0x13,
+ "return": 0x14,
+ "unreachable": 0x15,
+ "call_import": 0x1f,
+ "i32.load8_s": 0x20,
+ "i32.load8_u": 0x21,
+ "i32.load16_s": 0x22,
+ "i32.load16_u": 0x23,
+ "i64.load8_s": 0x24,
+ "i64.load8_u": 0x25,
+ "i64.load16_s": 0x26,
+ "i64.load16_u": 0x27,
+ "i64.load32_s": 0x28,
+ "i64.load32_u": 0x29,
+ "i32.load": 0x2a,
+ "i64.load": 0x2b,
+ "f32.load": 0x2c,
+ "f64.load": 0x2d,
+ "i32.store8": 0x2e,
+ "i32.store16": 0x2f,
+ "i64.store8": 0x30,
+ "i64.store16": 0x31,
+ "i64.store32": 0x32,
+ "i32.store": 0x33,
+ "i64.store": 0x34,
+ "f32.store": 0x35,
+ "f64.store": 0x36,
+ "memory_size": 0x3b,
+ "grow_memory": 0x39,
+ "i32.add": 0x40,
+ "i32.sub": 0x41,
+ "i32.mul": 0x42,
+ "i32.div_s": 0x43,
+ "i32.div_u": 0x44,
+ "i32.rem_s": 0x45,
+ "i32.rem_u": 0x46,
+ "i32.and": 0x47,
+ "i32.or": 0x48,
+ "i32.xor": 0x49,
+ "i32.shl": 0x4a,
+ "i32.shr_u": 0x4b,
+ "i32.shr_s": 0x4c,
+ "i32.eq": 0x4d,
+ "i32.ne": 0x4e,
+ "i32.lt_s": 0x4f,
+ "i32.le_s": 0x50,
+ "i32.lt_u": 0x51,
+ "i32.le_u": 0x52,
+ "i32.gt_s": 0x53,
+ "i32.ge_s": 0x54,
+ "i32.gt_u": 0x55,
+ "i32.ge_u": 0x56,
+ "i32.clz": 0x57,
+ "i32.ctz": 0x58,
+ "i32.popcnt": 0x59,
+ "i32.eqz": 0x5a,
+ "i64.add": 0x5b,
+ "i64.sub": 0x5c,
+ "i64.mul": 0x5d,
+ "i64.div_s": 0x5e,
+ "i64.div_u": 0x5f,
+ "i64.rem_s": 0x60,
+ "i64.rem_u": 0x61,
+ "i64.and": 0x62,
+ "i64.or": 0x63,
+ "i64.xor": 0x64,
+ "i64.shl": 0x65,
+ "i64.shr_u": 0x66,
+ "i64.shr_s": 0x67,
+ "i64.eq": 0x68,
+ "i64.ne": 0x69,
+ "i64.lt_s": 0x6a,
+ "i64.le_s": 0x6b,
+ "i64.lt_u": 0x6c,
+ "i64.le_u": 0x6d,
+ "i64.gt_s": 0x6e,
+ "i64.ge_s": 0x6f,
+ "i64.gt_u": 0x70,
+ "i64.ge_u": 0x71,
+ "i64.clz": 0x72,
+ "i64.ctz": 0x73,
+ "i64.popcnt": 0x74,
+ "f32.add": 0x75,
+ "f32.sub": 0x76,
+ "f32.mul": 0x77,
+ "f32.div": 0x78,
+ "f32.min": 0x79,
+ "f32.max": 0x7a,
+ "f32.abs": 0x7b,
+ "f32.neg": 0x7c,
+ "f32.copysign": 0x7d,
+ "f32.ceil": 0x7e,
+ "f32.floor": 0x7f,
+ "f32.trunc": 0x80,
+ "f32.nearest": 0x81,
+ "f32.sqrt": 0x82,
+ "f32.eq": 0x83,
+ "f32.ne": 0x84,
+ "f32.lt": 0x85,
+ "f32.le": 0x86,
+ "f32.gt": 0x87,
+ "f32.ge": 0x88,
+ "f64.add": 0x89,
+ "f64.sub": 0x8a,
+ "f64.mul": 0x8b,
+ "f64.div": 0x8c,
+ "f64.min": 0x8d,
+ "f64.max": 0x8e,
+ "f64.abs": 0x8f,
+ "f64.neg": 0x90,
+ "f64.copysign": 0x91,
+ "f64.ceil": 0x92,
+ "f64.floor": 0x93,
+ "f64.trunc": 0x94,
+ "f64.nearest": 0x95,
+ "f64.sqrt": 0x96,
+ "f64.eq": 0x97,
+ "f64.ne": 0x98,
+ "f64.lt": 0x99,
+ "f64.le": 0x9a,
+ "f64.gt": 0x9b,
+ "f64.ge": 0x9c,
+ "i32.trunc_s/f32": 0x9d,
+ "i32.trunc_s/f64": 0x9e,
+ "i32.trunc_u/f32": 0x9f,
+ "i32.trunc_u/f64": 0xa0,
+ "i32.wrap/i64": 0xa1,
+ "i64.trunc_s/f32": 0xa2,
+ "i64.trunc_s/f64": 0xa3,
+ "i64.trunc_u/f32": 0xa4,
+ "i64.trunc_u/f64": 0xa5,
+ "i64.extend_s/i32": 0xa6,
+ "i64.extend_u/i32": 0xa7,
+ "f32.convert_s/i32": 0xa8,
+ "f32.convert_u/i32": 0xa9,
+ "f32.convert_s/i64": 0xaa,
+ "f32.convert_u/i64": 0xab,
+ "f32.demote/f64": 0xac,
+ "f32.reinterpret/i32": 0xad,
+ "f64.convert_s/i32": 0xae,
+ "f64.convert_u/i32": 0xaf,
+ "f64.convert_s/i64": 0xb0,
+ "f64.convert_u/i64": 0xb1,
+ "f64.promote/f32": 0xb2,
+ "f64.reinterpret/i64": 0xb3,
+ "i32.reinterpret/f32": 0xb4,
+ "i64.reinterpret/f64": 0xb5,
+ "i32.rotr": 0xb6,
+ "i32.rotl": 0xb7,
+ "i64.rotr": 0xb8,
+ "i64.rotl": 0xb9,
+ 'i64.eqz': 0xba,
+}
+
+keywords = {
+ 'func': 'FUNC',
+ 'section': 'SECTION',
+ 'leb_i32': 'LEB_I32',
+ 'leb_i64': 'LEB_I64',
+ 'leb_u32': 'LEB_U32',
+ 'f32': 'F32',
+ 'f64': 'F64',
+ 'str': 'STR',
+}
+
+## lexer ###
+
+tokens = tuple(keywords.values()) + (
+ 'BYTE',
+ 'INT',
+ 'FLOAT',
+ 'STRING',
+ 'NAME',
+ 'NAMED_VALUE',
+ 'LPAREN',
+ 'RPAREN',
+ 'LBRACE',
+ 'RBRACE',
+ 'LBRACKET',
+ 'RBRACKET',
+)
+
+t_LPAREN = r'\('
+t_RPAREN = r'\)'
+t_LBRACE = r'{'
+t_RBRACE = r'}'
+t_LBRACKET = r'\['
+t_RBRACKET = r'\]'
+t_ignore = ' \t'
+
+def t_COMMENT(t):
+ r'\#.*'
+ pass
+
+def t_INT(t):
+ r'\-?([0-9]+|0x[0-9a-fA-F]+)'
+ t.value = int(t.value)
+ if 0 <= t.value < 256:
+ t.type = 'BYTE'
+ return t
+
+def t_FLOAT(t):
+ r'\-?([0-9]*\.?[0-9]+|[0-9]+\.?[0-9]*)([eE][0-9]+)?'
+ t.value = float(t.value)
+ return t
+
+def t_STRING(t):
+ r'\'[^\']*\'|\"[^\"]*\"'
+ t.value = t.value[1:-1]
+ return t
+
+def t_NAME(t):
+ r'[a-zA-Z][a-zA-Z0-9_\.\/]*'
+ if t.value in NAMED_VALUES:
+ t.type = 'NAMED_VALUE'
+ t.value = NAMED_VALUES[t.value]
+ elif t.value in keywords:
+ t.type = keywords[t.value]
+ return t
+
+def t_newline(t):
+ r'\n+'
+ t.lexer.lineno += len(t.value)
+
+def t_error(t):
+ print "Illegal character '%s'" % t.value[0]
+ t.lexer.skip(1)
+
+lexer = lex.lex()
+
+## parser ###
+
+def LebLoop(data, v, cond):
+ while True:
+ byte = v & 0x7f
+ v >>= 7
+ if cond(v, byte):
+ data.append(byte)
+ break
+ else:
+ data.append(byte | 0x80)
+
+def WriteUnsignedLeb(data, v, max_size):
+ result = []
+ LebLoop(result, v, lambda v, byte: v == 0)
+ assert len(result) <= max_size
+ data.extend(result)
+
+def WriteLebU32(data, v):
+ WriteUnsignedLeb(data, v, 5)
+
+def WriteSignedLeb(data, v, max_size):
+ result = []
+ if v < 0:
+ LebLoop(result, v, lambda v, byte: v == -1 and byte & 0x40)
+ else:
+ LebLoop(result, v, lambda v, byte: v == 0 and not byte & 0x40)
+ assert len(result) <= max_size
+ data.extend(result)
+
+def WriteLebI32(data, v):
+ WriteSignedLeb(data, v, 5)
+
+def WriteLebI64(data, v):
+ WriteSignedLeb(data, v, 10)
+
+def WriteF32(data, v):
+ data.extend(ord(b) for b in struct.pack('<f', v))
+
+def WriteF64(data, v):
+ data.extend(ord(b) for b in struct.pack('<d', v))
+
+def WriteString(data, s):
+ data.extend(ord(c) for c in s)
+
+def p_data_byte(p):
+ 'data : data BYTE'
+ p[0] = p[1]
+ p[0].append(p[2])
+
+def p_data_name(p):
+ '''data : data NAME LBRACKET data RBRACKET
+ | data FUNC LBRACKET data RBRACKET'''
+ p[0] = p[1]
+ # name is only used for documentation
+ p[0].extend(p[4])
+
+def p_data_named_value(p):
+ 'data : data NAMED_VALUE'
+ p[0] = p[1]
+ if type(p[2]) is tuple:
+ p[0].extend(p[2])
+ else:
+ p[0].append(p[2])
+
+def p_data_section(p):
+ 'data : data SECTION LPAREN STRING RPAREN LBRACE data RBRACE'
+ p[0] = p[1]
+ name = p[4]
+ contents = p[7]
+ section_data = []
+ WriteLebU32(section_data, len(name))
+ WriteString(section_data, name)
+ section_data.extend(contents)
+ WriteLebU32(p[0], len(section_data))
+ p[0].extend(section_data)
+
+def p_data_func(p):
+ 'data : data FUNC LBRACE data RBRACE'
+ p[0] = p[1]
+ func_data = p[4]
+ WriteLebU32(p[0], len(func_data))
+ p[0].extend(func_data)
+
+def p_data_str(p):
+ 'data : data STR LPAREN STRING RPAREN'
+ p[0] = p[1]
+ s = p[4]
+ WriteLebU32(p[0], len(s))
+ WriteString(p[0], s)
+
+def p_data_leb_i32(p):
+ '''data : data LEB_I32 LPAREN INT RPAREN
+ | data LEB_I32 LPAREN BYTE RPAREN'''
+ p[0] = p[1]
+ WriteLebI32(p[0], p[4])
+
+def p_data_leb_i64(p):
+ '''data : data LEB_I64 LPAREN INT RPAREN
+ | data LEB_I64 LPAREN BYTE RPAREN'''
+ p[0] = p[1]
+ WriteLebI64(p[0], p[4])
+
+def p_data_leb_u32(p):
+ '''data : data LEB_U32 LPAREN INT RPAREN
+ | data LEB_U32 LPAREN BYTE RPAREN'''
+ p[0] = p[1]
+ WriteLebU32(p[0], p[4])
+
+def p_data_f32(p):
+ 'data : data F32 LPAREN FLOAT RPAREN'
+ p[0] = p[1]
+ WriteF32(p[0], p[4])
+
+def p_data_f64(p):
+ 'data : data F64 LPAREN FLOAT RPAREN'
+ p[0] = p[1]
+ WriteF64(p[0], p[4])
+
+def p_data_string(p):
+ 'data : data STRING'
+ p[0] = p[1]
+ WriteString(p[0], p[2])
+
+def p_data_empty(p):
+ 'data :'
+ p[0] = []
+
+def p_error(p):
+ print '%d: syntax error, %s' % (p.lineno, p)
+
+parser = yacc.yacc()
+
+################################################################################
+
+def Run(input_file_name):
+ with open(input_file_name) as input_file:
+ input_data = input_file.read()
+ data = parser.parse(input_data)
+ # convert to string
+ data = ''.join(chr(x) for x in data)
+ return data
+
+def main(args):
+ arg_parser = argparse.ArgumentParser()
+ arg_parser.add_argument('-o', '--output', metavar='PATH', help='output file.')
+ arg_parser.add_argument('-v', '--verbose',
+ help='print more diagnotic messages.',
+ action='store_true')
+ arg_parser.add_argument('file', help='input file.')
+ options = arg_parser.parse_args(args)
+ data = Run(options.file)
+ if options.output:
+ with open(options.output, 'wb') as output_file:
+ output_file.write(data)
+ else:
+ sys.stdout.writelines(Hexdump(data))
+ return 0
+
+
+if __name__ == '__main__':
+ try:
+ sys.exit(main(sys.argv[1:]))
+ except Error as e:
+ sys.stderr.write(str(e) + '\n')
+ sys.exit(1)
diff --git a/test/run-gen-wasm.py b/test/run-gen-wasm.py
new file mode 100755
index 00000000..068c432a
--- /dev/null
+++ b/test/run-gen-wasm.py
@@ -0,0 +1,73 @@
+#!/usr/bin/env python
+#
+# Copyright 2016 WebAssembly Community Group participants
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+import argparse
+import os
+import shutil
+import subprocess
+import sys
+import tempfile
+
+import find_exe
+import utils
+from utils import Error
+
+SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
+GEN_WASM_PY = os.path.join(SCRIPT_DIR, 'gen-wasm.py')
+
+
+def main(args):
+ parser = argparse.ArgumentParser()
+ parser.add_argument('-v', '--verbose', help='print more diagnotic messages.',
+ action='store_true')
+ parser.add_argument('-o', '--out-dir', metavar='PATH',
+ help='output directory for files.')
+ parser.add_argument('--wasm-wast-executable', metavar='PATH',
+ help='set the wasm-wast executable to use.')
+ parser.add_argument('--no-error-cmdline',
+ help='don\'t display the subprocess\'s commandline when' +
+ ' an error occurs', dest='error_cmdline',
+ action='store_false')
+ parser.add_argument('--use-libc-allocator', action='store_true')
+ parser.add_argument('--debug-names', action='store_true')
+ parser.add_argument('--generate-names', action='store_true')
+ parser.add_argument('file', help='test file.')
+ options = parser.parse_args(args)
+
+ gen_wasm = utils.Executable(
+ sys.executable, GEN_WASM_PY, error_cmdline=options.error_cmdline)
+
+ wasm_wast = utils.Executable(
+ find_exe.GetWasmWastExecutable(options.wasm_wast_executable),
+ error_cmdline=options.error_cmdline)
+ wasm_wast.AppendOptionalArgs({
+ '--debug-names': options.debug_names,
+ '--generate-names': options.generate_names,
+ '--use-libc-allocator': options.use_libc_allocator
+ })
+
+ with utils.TempDirectory(options.out_dir, 'run-gen-wasm-') as out_dir:
+ out_file = utils.ChangeDir(utils.ChangeExt(options.file, '.wasm'), out_dir)
+ gen_wasm.RunWithArgs(options.file, '-o', out_file)
+ wasm_wast.RunWithArgs(out_file)
+
+if __name__ == '__main__':
+ try:
+ sys.exit(main(sys.argv[1:]))
+ except Error as e:
+ sys.stderr.write(str(e) + '\n')
+ sys.exit(1)
diff --git a/test/run-roundtrip.py b/test/run-roundtrip.py
index 124583e1..f6966b8a 100755
--- a/test/run-roundtrip.py
+++ b/test/run-roundtrip.py
@@ -28,8 +28,6 @@ import findtests
import utils
from utils import Error
-SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
-
OK = 0
ERROR = 1
SKIPPED = 2
diff --git a/test/run-tests.py b/test/run-tests.py
index 1b7c17c5..6e61df1b 100755
--- a/test/run-tests.py
+++ b/test/run-tests.py
@@ -91,6 +91,13 @@ TOOLS = {
'--spec',
'--no-error-cmdline',
])
+ },
+ 'run-gen-wasm': {
+ 'EXE': 'test/run-gen-wasm.py',
+ 'FLAGS': ' '.join([
+ '--wasm-wast-executable=%(wasm-wast)s',
+ '--no-error-cmdline',
+ ])
}
}
diff --git a/test/utils.py b/test/utils.py
index a2036c4a..6cd91954 100644
--- a/test/utils.py
+++ b/test/utils.py
@@ -33,16 +33,16 @@ class Error(Exception):
class Executable(object):
- def __init__(self, exe, error_cmdline=True, clean_stdout=None,
- clean_stderr=None):
+ def __init__(self, exe, *before_args, **kwargs):
self.exe = exe
- self.extra_args = []
- self.error_cmdline = error_cmdline
- self.clean_stdout = clean_stdout
- self.clean_stderr = clean_stderr
+ self.before_args = list(before_args)
+ self.after_args = []
+ self.error_cmdline = kwargs.get('error_cmdline', True)
+ self.clean_stdout = kwargs.get('clean_stdout')
+ self.clean_stderr = kwargs.get('clean_stderr')
def RunWithArgs(self, *args):
- cmd = [self.exe] + list(args) + self.extra_args
+ cmd = [self.exe] + self.before_args + list(args) + self.after_args
cmd_str = ' '.join(cmd)
err_cmd_str = cmd_str
@@ -69,7 +69,7 @@ class Executable(object):
raise Error('Error running "%s": %s' % (err_cmd_str, str(e)))
def AppendArg(self, arg):
- self.extra_args.append(arg)
+ self.after_args.append(arg)
def AppendOptionalArgs(self, option_dict):
for option, value in option_dict.iteritems():
diff --git a/third_party/ply b/third_party/ply
new file mode 160000
+Subproject d776a2ece6c12bf8f8b6a0e65b48546ac607876