| # Copyright 2014 The Emscripten Authors. All rights reserved. |
| # Emscripten is available under two separate licenses, the MIT license and the |
| # University of Illinois/NCSA Open Source License. Both these licenses can be |
| # found in the LICENSE file. |
| |
| import json |
| import os |
| import shutil |
| |
| if __name__ == '__main__': |
| raise Exception('do not run this file directly; do something like: test/runner.py interactive') |
| |
| from browser_common import BrowserCore |
| from common import copy_asset, create_file, test_file |
| from decorators import also_with_minimal_runtime, parameterized |
| |
| from tools.utils import WINDOWS |
| |
| |
| class interactive(BrowserCore): |
| @classmethod |
| def setUpClass(cls): |
| super().setUpClass() |
| cls.browser_timeout = 60 |
| print() |
| print('Running the interactive tests. Make sure the browser allows popups from localhost.') |
| print() |
| |
| def test_html5_core(self): |
| self.btest_exit('test_html5_core.c', cflags=['-DKEEP_ALIVE']) |
| |
| def test_html5_fullscreen(self): |
| self.btest('test_html5_fullscreen.c', expected='0', cflags=['-sDISABLE_DEPRECATED_FIND_EVENT_TARGET_BEHAVIOR', '-sEXPORTED_FUNCTIONS=_requestFullscreen,_enterSoftFullscreen,_main', '--shell-file', test_file('browser/test_html5_fullscreen.html')]) |
| |
| def test_html5_emscripten_exit_with_escape(self): |
| self.btest('test_html5_emscripten_exit_fullscreen.c', expected='1', cflags=['-DEXIT_WITH_F']) |
| |
| def test_html5_emscripten_exit_fullscreen(self): |
| self.btest('test_html5_emscripten_exit_fullscreen.c', expected='1') |
| |
| def test_html5_mouse(self): |
| self.btest_exit('test_html5_mouse.c', cflags=['-sDISABLE_DEPRECATED_FIND_EVENT_TARGET_BEHAVIOR']) |
| |
| def test_html5_pointerlockerror(self): |
| self.btest('test_html5_pointerlockerror.c', expected='0', cflags=['-sDISABLE_DEPRECATED_FIND_EVENT_TARGET_BEHAVIOR']) |
| |
| def test_sdl_mousewheel(self): |
| self.btest_exit('test_sdl_mousewheel.c') |
| |
| def test_sdl_touch(self): |
| self.btest('test_sdl_touch.c', cflags=['-O2', '-g1', '--closure=1'], expected='0') |
| |
| def test_sdl_wm_togglefullscreen(self): |
| self.btest_exit('test_sdl_wm_togglefullscreen.c') |
| |
| def test_sdl_key_events(self): |
| self.btest_exit('test_sdl_key_events.c', cflags=['-sRUNTIME_DEBUG']) |
| |
| def test_sdl_fullscreen_samecanvassize(self): |
| self.btest_exit('test_sdl_fullscreen_samecanvassize.c') |
| |
| def test_sdl2_togglefullscreen(self): |
| self.btest_exit('browser/test_sdl_togglefullscreen.c', cflags=['-sUSE_SDL=2']) |
| |
| def test_sdl_audio(self): |
| copy_asset('sounds/alarmvictory_1.ogg', 'sound.ogg') |
| copy_asset('sounds/alarmcreatemiltaryfoot_1.wav', 'sound2.wav') |
| copy_asset('sounds/noise.ogg') |
| copy_asset('sounds/the_entertainer.ogg') |
| create_file('bad.ogg', 'I claim to be audio, but am lying') |
| |
| # use closure to check for a possible bug with closure minifying away newer Audio() attributes |
| self.btest_exit('test_sdl_audio.c', cflags=['--preload-file', 'sound.ogg', '--preload-file', 'sound2.wav', '--embed-file', 'the_entertainer.ogg', '--preload-file', 'noise.ogg', '--preload-file', 'bad.ogg']) |
| |
| # print('SDL2') |
| # check sdl2 as well |
| # FIXME: restore this test once we have proper SDL2 audio (existing test |
| # depended on fragile SDL1/SDL2 mixing, which stopped working with |
| # 7a5744d754e00bec4422405a1a94f60b8e53c8fc (which just uncovered |
| # the existing problem) |
| # self.run_process([EMCC, '-O1', '--closure', '0', '--minify=0', 'sdl_audio.c', '--preload-file', 'sound.ogg', '--preload-file', 'sound2.wav', '--embed-file', 'the_entertainer.ogg', '--preload-file', 'noise.ogg', '--preload-file', 'bad.ogg', '-o', 'page.html', '-sEXPORTED_FUNCTIONS=[_main,_play,_play2', '-sUSE_SDL=2', '-DUSE_SDL2']).communicate() |
| # self.run_browser('page.html', '', '/report_result?1') |
| |
| @parameterized({ |
| '': ([],), |
| 'wasmfs': (['-sWASMFS'],), |
| }) |
| def test_sdl_audio_mix_channels(self, args): |
| copy_asset('sounds/noise.ogg') |
| self.btest_exit('test_sdl_audio_mix_channels.c', cflags=['-O2', '--minify=0', '--preload-file', 'sound.ogg'] + args) |
| |
| def test_sdl_audio_mix_channels_halt(self): |
| copy_asset('sounds/the_entertainer.ogg') |
| self.btest_exit('test_sdl_audio_mix_channels_halt.c', cflags=['-O2', '--minify=0', '--preload-file', 'the_entertainer.ogg']) |
| |
| def test_sdl_audio_mix_playing(self): |
| copy_asset('sounds/noise.ogg') |
| self.btest_exit('test_sdl_audio_mix_playing.c', cflags=['-O2', '--minify=0', '--preload-file', 'noise.ogg']) |
| |
| @parameterized({ |
| '': ([],), |
| 'wasmfs': (['-sWASMFS'],), |
| }) |
| def test_sdl_audio_mix(self, args): |
| copy_asset('sounds/pluck.ogg', 'sound.ogg') |
| copy_asset('sounds/the_entertainer.ogg', 'music.ogg') |
| copy_asset('sounds/noise.ogg', 'noise.ogg') |
| |
| self.btest_exit('test_sdl_audio_mix.c', cflags=['-O2', '--minify=0', '--preload-file', 'sound.ogg', '--preload-file', 'music.ogg', '--preload-file', 'noise.ogg'] + args) |
| |
| def test_sdl_audio_panning(self): |
| copy_asset('sounds/the_entertainer.wav') |
| |
| # use closure to check for a possible bug with closure minifying away newer Audio() attributes |
| self.btest_exit('test_sdl_audio_panning.c', cflags=['-O2', '--closure=1', '--minify=0', '--preload-file', 'the_entertainer.wav', '-sEXPORTED_FUNCTIONS=_main,_play']) |
| |
| def test_sdl_audio_beeps(self): |
| # use closure to check for a possible bug with closure minifying away newer Audio() attributes |
| self.btest_exit('test_sdl_audio_beep.cpp', cflags=['-O2', '--closure=1', '--minify=0', '-sDISABLE_EXCEPTION_CATCHING=0', '-o', 'page.html']) |
| |
| def test_sdl2_mixer_wav(self): |
| copy_asset('sounds/the_entertainer.wav', 'sound.wav') |
| self.btest_exit('browser/test_sdl2_mixer_wav.c', cflags=[ |
| '-O2', |
| '-sUSE_SDL=2', |
| '-sUSE_SDL_MIXER=2', |
| '-sINITIAL_MEMORY=33554432', |
| '--preload-file', 'sound.wav', |
| ]) |
| |
| @parameterized({ |
| 'wav': ([], '0', 'the_entertainer.wav'), |
| 'ogg': (['ogg'], 'MIX_INIT_OGG', 'alarmvictory_1.ogg'), |
| 'mp3': (['mp3'], 'MIX_INIT_MP3', 'pudinha.mp3'), |
| }) |
| def test_sdl2_mixer_music(self, formats, flags, music_name): |
| copy_asset('sounds', music_name) |
| self.btest('browser/test_sdl2_mixer_music.c', expected='exit:0', cflags=[ |
| '-O2', |
| '--minify=0', |
| '--preload-file', music_name, |
| '-DSOUND_PATH=' + json.dumps(music_name), |
| '-DFLAGS=' + flags, |
| '-sUSE_SDL=2', |
| '-sUSE_SDL_MIXER=2', |
| '-sSDL2_MIXER_FORMATS=' + json.dumps(formats), |
| '-sINITIAL_MEMORY=33554432', |
| ]) |
| |
| def test_sdl2_audio_beeps(self): |
| # use closure to check for a possible bug with closure minifying away newer Audio() attributes |
| # TODO: investigate why this does not pass |
| self.btest_exit('browser/test_sdl2_audio_beep.cpp', cflags=['-O2', '--closure=1', '--minify=0', '-sDISABLE_EXCEPTION_CATCHING=0', '-sUSE_SDL=2']) |
| |
| @parameterized({ |
| '': ([],), |
| 'proxy_to_pthread': (['-sPROXY_TO_PTHREAD', '-pthread'],), |
| }) |
| def test_openal_playback(self, args): |
| copy_asset('sounds/audio.wav') |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-O2', '--preload-file', 'audio.wav'] + args) |
| |
| def test_openal_buffers(self): |
| self.btest_exit('openal/test_openal_buffers.c', cflags=['-DTEST_INTERACTIVE=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/']) |
| |
| def test_openal_buffers_animated_pitch(self): |
| self.btest_exit('openal/test_openal_buffers.c', cflags=['-DTEST_INTERACTIVE=1', '-DTEST_ANIMATED_PITCH=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/']) |
| |
| def test_openal_looped_pitched_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_looped_seek_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_LOOPED_SEEK_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_animated_looped_pitched_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ANIMATED_LOOPED_PITCHED_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_animated_looped_distance_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ANIMATED_LOOPED_DISTANCE_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_animated_looped_doppler_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ANIMATED_LOOPED_DOPPLER_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_animated_looped_panned_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ANIMATED_LOOPED_PANNED_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_animated_looped_relative_playback(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ANIMATED_LOOPED_RELATIVE_PLAYBACK=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_al_soft_loop_points(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_AL_SOFT_LOOP_POINTS=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_alc_soft_pause_device(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_ALC_SOFT_PAUSE_DEVICE=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_al_soft_source_spatialize(self): |
| self.btest('openal/test_openal_playback.c', '1', cflags=['-DTEST_AL_SOFT_SOURCE_SPATIALIZE=1', '-DTEST_LOOPED_PLAYBACK=1', '--preload-file', test_file('sounds/the_entertainer.wav') + '@/audio.wav']) |
| |
| def test_openal_capture(self): |
| self.btest_exit('openal/test_openal_capture.c') |
| |
| def get_freealut_library(self): |
| self.cflags += ['-Wno-pointer-sign'] |
| if WINDOWS and shutil.which('cmake'): |
| return self.get_library(os.path.join('third_party', 'freealut'), 'libalut.a', configure=['cmake', '.'], configure_args=['-DBUILD_TESTS=ON']) |
| else: |
| return self.get_library(os.path.join('third_party', 'freealut'), os.path.join('src', '.libs', 'libalut.a'), configure_args=['--disable-shared']) |
| |
| def test_freealut(self): |
| src = test_file('third_party/freealut/examples/hello_world.c') |
| inc = test_file('third_party/freealut/include') |
| self.btest_exit(src, cflags=['-O2', '-I' + inc] + self.get_freealut_library()) |
| |
| def test_glfw_cursor_disabled(self): |
| self.btest_exit('interactive/test_glfw_cursor_disabled.c', cflags=['-sUSE_GLFW=3', '-lglfw', '-lGL']) |
| |
| def test_glfw_dropfile(self): |
| self.btest_exit('interactive/test_glfw_dropfile.c', cflags=['-sUSE_GLFW=3', '-lglfw', '-lGL']) |
| |
| def test_glfw_fullscreen(self): |
| self.btest_exit('interactive/test_glfw_fullscreen.c', cflags=['-sUSE_GLFW=3']) |
| |
| def test_glfw_get_key_stuck(self): |
| self.btest_exit('interactive/test_glfw_get_key_stuck.c', cflags=['-sUSE_GLFW=3']) |
| |
| def test_glfw_joystick(self): |
| self.btest_exit('interactive/test_glfw_joystick.c', cflags=['-sUSE_GLFW=3']) |
| |
| def test_glfw_pointerlock(self): |
| self.btest_exit('interactive/test_glfw_pointerlock.c', cflags=['-sUSE_GLFW=3']) |
| |
| def test_glut_fullscreen(self): |
| self.skipTest('looks broken') |
| self.btest_exit('glut_fullscreen.c', cflags=['-lglut', '-lGL']) |
| |
| def test_cpuprofiler_memoryprofiler(self): |
| self.btest('hello_world_gles.c', expected='0', cflags=['-DLONGTEST=1', '-DTEST_MEMORYPROFILER_ALLOCATIONS_MAP=1', '-O2', '--cpuprofiler', '--memoryprofiler']) |
| |
| def test_threadprofiler(self): |
| args = ['-O2', '--threadprofiler', |
| '-pthread', |
| '-sASSERTIONS', |
| '-DTEST_THREAD_PROFILING=1', |
| '-sPTHREAD_POOL_SIZE=16', |
| '-sINITIAL_MEMORY=64mb', |
| '--shell-file', test_file('pthread/test_pthread_mandelbrot_shell.html')] |
| self.btest_exit('pthread/test_pthread_mandelbrot.cpp', cflags=args) |
| |
| # Test that event backproxying works. |
| def test_html5_callbacks_on_calling_thread(self): |
| # TODO: Make this automatic by injecting mouse event in e.g. shell html file. |
| for args in ([], ['-DTEST_SYNC_BLOCKING_LOOP=1']): |
| self.btest('html5_callbacks_on_calling_thread.c', expected='1', cflags=args + ['-sDISABLE_DEPRECATED_FIND_EVENT_TARGET_BEHAVIOR', '-pthread', '-sPROXY_TO_PTHREAD']) |
| |
| # Test that it is possible to register HTML5 event callbacks on either main browser thread, or |
| # application main thread, and that the application can manually proxy the event from main browser |
| # thread to the application main thread, to implement event suppression capabilities. |
| @parameterized({ |
| '': ([],), |
| 'pthread': (['-pthread'],), |
| 'proxy_to_pthread': (['-pthread', '-sPROXY_TO_PTHREAD'],), |
| }) |
| def test_html5_event_callback_in_two_threads(self, args): |
| # TODO: Make this automatic by injecting enter key press in e.g. shell html file. |
| self.btest_exit('html5_event_callback_in_two_threads.c', cflags=args) |
| |
| # Test that emscripten_hide_mouse() is callable from pthreads (and proxies to main |
| # thread to obtain the proper window.devicePixelRatio value). |
| @parameterized({ |
| '': ([],), |
| 'threads': (['-pthread'],), |
| }) |
| def test_emscripten_hide_mouse(self, args): |
| self.btest('emscripten_hide_mouse.c', expected='0', cflags=args) |
| |
| # Tests that WebGL can be run on another thread after first having run it on one thread (and that |
| # thread has exited). The intent of this is to stress graceful deinit semantics, so that it is not |
| # possible to "taint" a Canvas to a bad state after a rendering thread in a program quits and |
| # restarts. (perhaps e.g. between level loads, or subsystem loads/restarts or something like that) |
| @parameterized({ |
| '': (['-sOFFSCREENCANVAS_SUPPORT', '-DTEST_OFFSCREENCANVAS=1'],), |
| 'ofb': (['-sOFFSCREEN_FRAMEBUFFER'],), |
| }) |
| def test_webgl_offscreen_canvas_in_two_pthreads(self, args): |
| self.btest('gl_in_two_pthreads.c', expected='1', cflags=args + ['-pthread', '-lGL', '-sGL_DEBUG', '-sPROXY_TO_PTHREAD']) |
| |
| # Tests creating a Web Audio context using Emscripten library_webaudio.js feature. |
| @also_with_minimal_runtime |
| def test_web_audio(self): |
| self.btest('webaudio/create_webaudio.c', expected='0', cflags=['-lwebaudio.js']) |
| |
| # Tests simple AudioWorklet noise generation |
| @also_with_minimal_runtime |
| def test_audio_worklet(self): |
| self.btest('webaudio/audioworklet.c', expected='0', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS', '--preload-file', test_file('hello_world.c') + '@/']) |
| self.btest('webaudio/audioworklet.c', expected='0', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS', '-pthread']) |
| |
| # Tests a second AudioWorklet example: sine wave tone generator. |
| def test_audio_worklet_tone_generator(self): |
| self.btest('webaudio/audio_worklet_tone_generator.c', expected='0', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests that AUDIO_WORKLET+MINIMAL_RUNTIME+MODULARIZE combination works together. |
| def test_audio_worklet_modularize(self): |
| self.btest('webaudio/audioworklet.c', expected='0', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS', '-sMINIMAL_RUNTIME', '-sMODULARIZE']) |
| |
| # Tests an AudioWorklet with multiple stereo inputs mixing in the processor to a single stereo output (4kB stack) |
| def test_audio_worklet_stereo_io(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_in_out_stereo.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests an AudioWorklet with multiple stereo inputs copying in the processor to multiple stereo outputs (6kB stack) |
| def test_audio_worklet_2x_stereo_io(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_2x_in_out_stereo.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests an AudioWorklet with multiple mono inputs mixing in the processor to a single mono output (2kB stack) |
| def test_audio_worklet_mono_io(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat-mono.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass-mono.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_in_out_mono.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests an AudioWorklet with multiple mono inputs copying in the processor to L+R stereo outputs (3kB stack) |
| def test_audio_worklet_2x_hard_pan_io(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat-mono.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass-mono.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_2x_in_hard_pan.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests an AudioWorklet with multiple stereo inputs mixing in the processor via a parameter to a single stereo output (6kB stack) |
| def test_audio_worklet_params_mixing(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_params_mixing.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Mixing test above with hard-pans to verify left and right ordering |
| def test_audio_worklet_hard_pans(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat-right.mp3', 'audio_files/emscripten-beat.mp3') |
| copy_asset('webaudio/audio_files/emscripten-bass-left.mp3', 'audio_files/emscripten-bass.mp3') |
| self.btest_exit('webaudio/audioworklet_params_mixing.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS']) |
| |
| # Tests an AudioWorklet with a growable heap |
| def test_audio_worklet_memory_growth(self): |
| os.mkdir('audio_files') |
| copy_asset('webaudio/audio_files/emscripten-beat.mp3', 'audio_files/') |
| copy_asset('webaudio/audio_files/emscripten-bass.mp3', 'audio_files/') |
| self.btest_exit('webaudio/audioworklet_memory_growth.c', cflags=['-sAUDIO_WORKLET', '-sWASM_WORKERS', '-sALLOW_MEMORY_GROWTH']) |
| |
| def test_html_source_map(self): |
| # browsers will try to 'guess' the corresponding original line if a |
| # generated line is unmapped, so if we want to make sure that our |
| # numbering is correct, we need to provide a couple of 'possible wrong |
| # answers'. thus, we add some printf calls so that the cpp file gets |
| # multiple mapped lines. in other words, if the program consists of a |
| # single 'throw' statement, browsers may just map any thrown exception to |
| # that line, because it will be the only mapped line. |
| create_file('src.cpp', r''' |
| #include <cstdio> |
| |
| int main() { |
| printf("Starting test\n"); |
| try { |
| throw 42; // line 8 |
| } catch (int e) { } |
| printf("done\n"); |
| return 0; |
| } |
| ''') |
| # use relative paths when calling emcc, because file:// URIs can only load |
| # sourceContent when the maps are relative paths |
| self.compile_btest('src.cpp', ['-o', 'src.html', '-gsource-map']) |
| self.assertExists('src.html') |
| self.assertExists('src.wasm.map') |
| print(''' |
| If manually bisecting: |
| Open the file with ./emrun out/test/src.html |
| Check that you see src.cpp among the page sources. |
| Even better, add a breakpoint, e.g. on the printf, then reload, then step |
| through and see the print (best to run with --save-dir for the reload). |
| ''') |
| |
| |
| class interactive64(interactive): |
| def setUp(self): |
| super().setUp() |
| self.set_setting('MEMORY64') |
| self.require_wasm64() |
| |
| |
| class interactive64_4gb(interactive): |
| def setUp(self): |
| super().setUp() |
| self.set_setting('MEMORY64') |
| self.set_setting('INITIAL_MEMORY', '4200mb') |
| self.set_setting('GLOBAL_BASE', '4gb') |
| self.require_wasm64() |
| |
| |
| class interactive_2gb(interactive): |
| def setUp(self): |
| super().setUp() |
| self.set_setting('INITIAL_MEMORY', '2200mb') |
| self.set_setting('GLOBAL_BASE', '2gb') |