diff --git a/test/common.py b/test/common.py index 036c89886e5c4..5b8bb7ec63e1b 100644 --- a/test/common.py +++ b/test/common.py @@ -833,6 +833,11 @@ def setUp(self): self.js_engines = config.JS_ENGINES.copy() self.settings_mods = {} self.emcc_args = ['-Wclosure', '-Werror', '-Wno-limited-postlink-optimizations'] + # TODO(https://github.com/emscripten-core/emscripten/issues/11121) + # For historical reasons emcc compiles and links as C++ by default. + # However we want to run our tests in a more strict manner. We can + # remove this if the issue above is ever fixed. + self.set_setting('NO_DEFAULT_TO_CXX') self.ldflags = [] # Increate stack trace limit to maximise usefulness of test failure reports self.node_args = ['--stack-trace-limit=50'] @@ -954,9 +959,13 @@ def has_changed_setting(self, key): def clear_setting(self, key): self.settings_mods.pop(key, None) - def serialize_settings(self): + def serialize_settings(self, ldflags=True): ret = [] + # Incomplete list of link-only settings + link_only_settings = ['NO_DEFAULT_TO_CXX', 'DEFAULT_LIBRARY_FUNCS_TO_INCLUDE'] for key, value in self.settings_mods.items(): + if not ldflags and key in link_only_settings: + continue if value == 1: ret.append(f'-s{key}') elif type(value) is list: @@ -994,7 +1003,7 @@ def get_emcc_args(self, main_file=False, ldflags=True): def is_ldflag(f): return any(f.startswith(s) for s in ['-sENVIRONMENT=', '--pre-js=', '--post-js=']) - args = self.serialize_settings() + self.emcc_args + args = self.serialize_settings(ldflags) + self.emcc_args if ldflags: args += self.ldflags else: @@ -2106,24 +2115,26 @@ def reftest(self, expected, manually_trigger=False): setupRefTest(); ''' % (reporting, basename, int(manually_trigger))) - def compile_btest(self, args, reporting=Reporting.FULL): + def compile_btest(self, filename, args, reporting=Reporting.FULL): # Inject support code for reporting results. This adds an include a header so testcases can # use REPORT_RESULT, and also adds a cpp file to be compiled alongside the testcase, which # contains the implementation of REPORT_RESULT (we can't just include that implementation in # the header as there may be multiple files being compiled here). if reporting != Reporting.NONE: # For basic reporting we inject JS helper funtions to report result back to server. - args += ['-DEMTEST_PORT_NUMBER=%d' % self.port, - '--pre-js', test_file('browser_reporting.js')] + args += ['--pre-js', test_file('browser_reporting.js')] if reporting == Reporting.FULL: # If C reporting (i.e. REPORT_RESULT macro) is required # also compile in report_result.c and forice-include report_result.h - args += ['-I' + TEST_ROOT, - '-include', test_file('report_result.h'), - test_file('report_result.c')] + self.run_process([EMCC, '-c', '-I' + TEST_ROOT, + '-DEMTEST_PORT_NUMBER=%d' % self.port, + test_file('report_result.c')] + self.get_emcc_args(ldflags=False)) + args += ['report_result.o', '-include', test_file('report_result.h')] if EMTEST_BROWSER == 'node': args.append('-DEMTEST_NODE') - self.run_process([EMCC] + self.get_emcc_args() + args) + if not os.path.exists(filename): + filename = test_file(filename) + self.run_process([compiler_for(filename), filename] + self.get_emcc_args() + args) def btest_exit(self, filename, assert_returncode=0, *args, **kwargs): """Special case of btest that reports its result solely via exiting @@ -2166,10 +2177,10 @@ def btest(self, filename, expected=None, reference=None, # manual_reference only makes sense for reference tests assert manual_reference is None outfile = output_basename + '.html' - args += [filename, '-o', outfile] + args += ['-o', outfile] # print('all args:', args) utils.delete_file(outfile) - self.compile_btest(args, reporting=reporting) + self.compile_btest(filename, args, reporting=reporting) self.assertExists(outfile) if post_build: post_build() diff --git a/test/test_browser.py b/test/test_browser.py index 70e1931b69b04..7b36b20fe18ba 100644 --- a/test/test_browser.py +++ b/test/test_browser.py @@ -261,8 +261,6 @@ def test_sdl1_es6(self): # as this test may take the focus away from the main test window # by opening a new window and possibly not closing it. def test_zzz_html_source_map(self): - if not has_browser(): - self.skipTest('need a browser') cpp_file = 'src.cpp' html_file = 'src.html' # browsers will try to 'guess' the corresponding original line if a @@ -288,9 +286,11 @@ def test_zzz_html_source_map(self): # sourceContent when the maps are relative paths delete_file(html_file) delete_file(html_file + '.map') - self.compile_btest(['src.cpp', '-o', 'src.html', '-gsource-map']) + self.compile_btest('src.cpp', ['-o', 'src.html', '-gsource-map']) self.assertExists(html_file) self.assertExists('src.wasm.map') + if not has_browser(): + self.skipTest('need a browser') webbrowser.open_new('file://' + html_file) print(''' If manually bisecting: @@ -436,7 +436,7 @@ def make_main_two_files(path1, path2, nonexistingpath): make_main('somefile.txt') # absolute becomes relative ensure_dir('dirrey') - self.compile_btest(['main.cpp', '--preload-file', absolute_src_path, '-o', 'dirrey/page.html'], reporting=Reporting.JS_ONLY) + self.compile_btest('main.cpp', ['--preload-file', absolute_src_path, '-o', 'dirrey/page.html'], reporting=Reporting.JS_ONLY) self.run_browser('dirrey/page.html', '/report_result?exit:0') # With FS.preloadFile @@ -455,11 +455,9 @@ def make_main_two_files(path1, path2, nonexistingpath): }) @requires_threads def test_preload_file_with_manual_data_download(self, args): - src = test_file('manual_download_data.cpp') - create_file('file.txt', '''Hello!''') - self.compile_btest([src, '-o', 'manual_download_data.js', '--preload-file', 'file.txt@/file.txt'] + args) + self.compile_btest('manual_download_data.cpp', ['-o', 'manual_download_data.js', '--preload-file', 'file.txt@/file.txt'] + args) shutil.copyfile(test_file('manual_download_data.html'), 'manual_download_data.html') # Move .data file out of server root to ensure that getPreloadedPackage is actually used @@ -504,7 +502,7 @@ def test_output_file_escaping(self): self.run_process([FILE_PACKAGER, data_file, '--use-preload-cache', '--indexedDB-name=testdb', '--preload', abs_txt + '@' + txt, '--js-output=' + data_js_file]) page_file = os.path.join(d, 'file with ' + tricky_part + '.html') abs_page_file = os.path.abspath(page_file) - self.compile_btest([cpp, '--pre-js', data_js_file, '-o', abs_page_file, '-sFORCE_FILESYSTEM'], reporting=Reporting.JS_ONLY) + self.compile_btest(cpp, ['--pre-js', data_js_file, '-o', abs_page_file, '-sFORCE_FILESYSTEM'], reporting=Reporting.JS_ONLY) self.run_browser(page_file, '/report_result?exit:0') @parameterized({ @@ -559,7 +557,7 @@ def test_preload_caching(self, extra_size): self.skipTest('chrome bug') create_file('somefile.txt', '''load me right before running the code please''' + ('_' * extra_size)) print('size:', os.path.getsize('somefile.txt')) - self.compile_btest(['main.c', '--use-preload-cache', '--js-library', 'test.js', '--preload-file', 'somefile.txt', '-o', 'page.html', '-sALLOW_MEMORY_GROWTH'], reporting=Reporting.JS_ONLY) + self.compile_btest('main.c', ['--use-preload-cache', '--js-library', 'test.js', '--preload-file', 'somefile.txt', '-o', 'page.html', '-sALLOW_MEMORY_GROWTH'], reporting=Reporting.JS_ONLY) self.run_browser('page.html', '/report_result?exit:0') self.run_browser('page.html', '/report_result?exit:1') @@ -609,7 +607,7 @@ def make_main(path): make_main('somefile.txt') self.run_process([FILE_PACKAGER, 'somefile.data', '--use-preload-cache', '--indexedDB-name=testdb', '--preload', 'somefile.txt', '--js-output=' + 'somefile.js']) - self.compile_btest(['main.c', '--js-library', 'test.js', '--pre-js', 'somefile.js', '-o', 'page.html', '-sFORCE_FILESYSTEM'], reporting=Reporting.JS_ONLY) + self.compile_btest('main.c', ['--js-library', 'test.js', '--pre-js', 'somefile.js', '-o', 'page.html', '-sFORCE_FILESYSTEM'], reporting=Reporting.JS_ONLY) self.run_browser('page.html', '/report_result?exit:0') self.run_browser('page.html', '/report_result?exit:1') @@ -649,7 +647,7 @@ def test_multifile(self): # by directory, and remove files to make sure self.set_setting('EXIT_RUNTIME') - self.compile_btest(['main.c', '--preload-file', 'subdirr', '-o', 'page.html'], reporting=Reporting.JS_ONLY) + self.compile_btest('main.c', ['--preload-file', 'subdirr', '-o', 'page.html'], reporting=Reporting.JS_ONLY) shutil.rmtree('subdirr') self.run_browser('page.html', '/report_result?exit:0') @@ -692,7 +690,7 @@ def test_custom_file_package_url(self): ''') self.set_setting('EXIT_RUNTIME') - self.compile_btest(['main.c', '--shell-file', 'shell.html', '--preload-file', 'subdirr/data1.txt', '-o', 'test.html'], reporting=Reporting.JS_ONLY) + self.compile_btest('main.c', ['--shell-file', 'shell.html', '--preload-file', 'subdirr/data1.txt', '-o', 'test.html'], reporting=Reporting.JS_ONLY) shutil.move('test.data', Path('cdn/test.data')) self.run_browser('test.html', '/report_result?exit:0') @@ -739,18 +737,18 @@ def setup(assetLocalization): def test(): # test test missing file should run xhr.onload with status different than 200, 304 or 206 setup("") - self.compile_btest(['main.cpp', '--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) + self.compile_btest('main.cpp', ['--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) shutil.move('test.data', 'missing.data') self.run_browser('test.html', '/report_result?1') # test unknown protocol should go through xhr.onerror setup("unknown_protocol://") - self.compile_btest(['main.cpp', '--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) + self.compile_btest('main.cpp', ['--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) self.run_browser('test.html', '/report_result?1') # test wrong protocol and port setup("https://localhost:8800/") - self.compile_btest(['main.cpp', '--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) + self.compile_btest('main.cpp', ['--shell-file', 'on_window_error_shell.html', '--preload-file', 'data.txt', '-o', 'test.html']) self.run_browser('test.html', '/report_result?1') test() @@ -890,7 +888,7 @@ def test_sdl_canvas_proxy(self): @requires_graphics_hardware def test_glgears_proxy_jstarget(self): # test .js target with --proxy-worker; emits 2 js files, client and worker - self.compile_btest([test_file('hello_world_gles_proxy.c'), '-o', 'test.js', '--proxy-to-worker', '-sGL_TESTING', '-lGL', '-lglut']) + self.compile_btest('hello_world_gles_proxy.c', ['-o', 'test.js', '--proxy-to-worker', '-sGL_TESTING', '-lGL', '-lglut']) shell_with_script('shell_minimal.html', 'test.html', '') self.reftest(test_file('gears.png')) self.post_manual_reftest() @@ -1152,7 +1150,7 @@ def test_sdl_mouse_offsets(self): ''') - self.compile_btest([test_file('browser/test_sdl_mouse.c'), '-DTEST_SDL_MOUSE_OFFSETS', '-O2', '--minify=0', '-o', 'sdl_mouse.js', '--pre-js', 'pre.js', '-lSDL', '-lGL', '-sEXIT_RUNTIME']) + self.compile_btest('browser/test_sdl_mouse.c', ['-DTEST_SDL_MOUSE_OFFSETS', '-O2', '--minify=0', '-o', 'sdl_mouse.js', '--pre-js', 'pre.js', '-lSDL', '-lGL', '-sEXIT_RUNTIME']) self.run_browser('page.html', '', '/report_result?exit:0') def test_glut_touchevents(self): @@ -1466,7 +1464,7 @@ def test_fs_lz4fs_package(self): print(' opts') self.btest_exit('fs/test_lz4fs.cpp', 2, args=['--pre-js', 'files.js', '-sLZ4', '-sFORCE_FILESYSTEM', '-O2']) print(' modularize') - self.compile_btest([test_file('fs/test_lz4fs.cpp'), '--pre-js', 'files.js', '-sLZ4', '-sFORCE_FILESYSTEM', '-sMODULARIZE', '-sEXIT_RUNTIME']) + self.compile_btest('fs/test_lz4fs.cpp', ['--pre-js', 'files.js', '-sLZ4', '-sFORCE_FILESYSTEM', '-sMODULARIZE', '-sEXIT_RUNTIME']) create_file('a.html', ''' ') self.run_browser('test.html', '/report_result?0') @@ -4476,7 +4472,7 @@ def test_in_flight_memfile_request(self, args, expected): self.set_setting('WASM', 0) print('plain html') - self.compile_btest([test_file('in_flight_memfile_request.c'), '-o', 'test.js']) + self.compile_btest('in_flight_memfile_request.c', ['-o', 'test.js']) create_file('test.html', '') # never when we provide our own HTML like this. self.run_browser('test.html', '/report_result?0') @@ -4539,7 +4535,7 @@ def test_async_compile(self): def test_manual_wasm_instantiate(self, args): if self.is_wasm64() and args: self.skipTest('TODO: ASAN in memory64') - self.compile_btest([test_file('manual_wasm_instantiate.cpp'), '-o', 'manual_wasm_instantiate.js'] + args) + self.compile_btest('manual_wasm_instantiate.cpp', ['-o', 'manual_wasm_instantiate.js'] + args) shutil.copyfile(test_file('manual_wasm_instantiate.html'), 'manual_wasm_instantiate.html') self.run_browser('manual_wasm_instantiate.html', '/report_result?1') @@ -4547,7 +4543,7 @@ def test_wasm_locate_file(self): # Test that it is possible to define "Module.locateFile(foo)" function to locate where worker.js will be loaded from. ensure_dir('cdn') create_file('shell2.html', read_file(path_from_root('src/shell.html')).replace('var Module = {', 'var Module = { locateFile: function(filename) { if (filename == "test.wasm") return "cdn/test.wasm"; else return filename; }, ')) - self.compile_btest([test_file('browser_test_hello_world.c'), '--shell-file', 'shell2.html', '-o', 'test.html']) + self.compile_btest('browser_test_hello_world.c', ['--shell-file', 'shell2.html', '-o', 'test.html']) shutil.move('test.wasm', Path('cdn/test.wasm')) self.run_browser('test.html', '/report_result?0') @@ -5057,7 +5053,7 @@ def test_pthread_reltime(self): def test_load_js_from_blob_with_pthreads(self): # TODO: enable this with wasm, currently pthreads/atomics have limitations self.set_setting('EXIT_RUNTIME') - self.compile_btest([test_file('pthread/hello_thread.c'), '-pthread', '-o', 'hello_thread_with_blob_url.js'], reporting=Reporting.JS_ONLY) + self.compile_btest('pthread/hello_thread.c', ['-pthread', '-o', 'hello_thread_with_blob_url.js'], reporting=Reporting.JS_ONLY) shutil.copyfile(test_file('pthread/main_js_as_blob_loader.html'), 'hello_thread_with_blob_url.html') self.run_browser('hello_thread_with_blob_url.html', '/report_result?exit:0') @@ -5123,7 +5119,7 @@ def test_single_file_locate_file(self, args): # Tests that SINGLE_FILE works as intended in a Worker in JS output def test_single_file_worker_js(self): - self.compile_btest([test_file('browser_test_hello_world.c'), '-o', 'test.js', '--proxy-to-worker', '-sSINGLE_FILE']) + self.compile_btest('browser_test_hello_world.c', ['-o', 'test.js', '--proxy-to-worker', '-sSINGLE_FILE']) create_file('test.html', '') self.run_browser('test.html', '/report_result?0') self.assertExists('test.js') @@ -5136,7 +5132,7 @@ def test_single_file_worker_js(self): @requires_threads def test_pthreads_started_in_worker(self): self.set_setting('EXIT_RUNTIME') - self.compile_btest([test_file('pthread/test_pthread_atomics.cpp'), '-o', 'test.js', '-sINITIAL_MEMORY=64MB', '-pthread', '-sPTHREAD_POOL_SIZE=8'], reporting=Reporting.JS_ONLY) + self.compile_btest('pthread/test_pthread_atomics.cpp', ['-o', 'test.js', '-sINITIAL_MEMORY=64MB', '-pthread', '-sPTHREAD_POOL_SIZE=8'], reporting=Reporting.JS_ONLY) create_file('test.html', '''