From f46c856dc5a64d178bfd080cba322c18b6de8b49 Mon Sep 17 00:00:00 2001 From: Jameson Nash Date: Mon, 14 Aug 2017 13:44:26 -0400 Subject: [PATCH] win: avoid stack overflow in initial deserialization fix #22866 [release-0.6 only] --- src/dump.c | 47 ++++++++++++++++++++++++++++++++++++++++++++++- src/threading.c | 2 +- 2 files changed, 47 insertions(+), 2 deletions(-) diff --git a/src/dump.c b/src/dump.c index 6cbcd93c2bfa2..baa4729c98be3 100644 --- a/src/dump.c +++ b/src/dump.c @@ -2595,11 +2595,51 @@ JL_DLLEXPORT void jl_set_sysimg_so(void *handle) jl_sysimg_handle = handle; } +#ifdef _OS_WINDOWS_ +static unsigned __stdcall _jl_restore_system_image_from_stream(void*); +extern DWORD jl_tls_key; +#endif + static void jl_restore_system_image_from_stream(ios_t *f) { +#ifdef _OS_WINDOWS_ + // on Windows, the default stack size is insufficient + // for this serializer design, + // use a separate thread to run this function + // as a workaround + int en = jl_gc_enable(0); + void *main_ptls = TlsGetValue(jl_tls_key); + void *args[] = { + (void*)f, + (void*)main_ptls + }; + HANDLE hThread = (HANDLE)_beginthreadex( + NULL, + 8 * 1024 * 1024, /* 8 MB */ + &_jl_restore_system_image_from_stream, + (void*)args, + 0, + NULL); + WaitForSingleObject(hThread, INFINITE); + CloseHandle(hThread); + + jl_gc_reset_alloc_count(); + jl_gc_enable(en); + jl_update_all_fptrs(); +} + +static unsigned __stdcall _jl_restore_system_image_from_stream(void* threadargs) +{ + ios_t *f = (ios_t*)((void**)threadargs)[0]; + void *old_ptls = TlsGetValue(jl_tls_key); + void *main_ptls = ((void**)threadargs)[1]; + TlsSetValue(jl_tls_key, main_ptls); +#else + int en = jl_gc_enable(0); +#endif + JL_TIMING(SYSIMG_LOAD); jl_ptls_t ptls = jl_get_ptls_states(); - int en = jl_gc_enable(0); arraylist_new(&backref_list, 250000); jl_serializer_state s = { f, MODE_SYSTEM_IMAGE, @@ -2663,9 +2703,14 @@ static void jl_restore_system_image_from_stream(ios_t *f) //jl_printf(JL_STDERR, "backref_list.len = %d\n", backref_list.len); arraylist_free(&backref_list); +#ifdef _OS_WINDOWS_ + TlsSetValue(jl_tls_key, old_ptls); + return 0; +#else jl_gc_reset_alloc_count(); jl_gc_enable(en); jl_update_all_fptrs(); +#endif } JL_DLLEXPORT void jl_restore_system_image(const char *fname) diff --git a/src/threading.c b/src/threading.c index baf4c2f808a4b..abf183ae3576d 100644 --- a/src/threading.c +++ b/src/threading.c @@ -94,7 +94,7 @@ jl_get_ptls_states_func jl_get_ptls_states_getter(void) // Apparently windows doesn't have a static TLS model (or one that can be // reliably used from a shared library) either..... Use `TLSAlloc` instead. -static DWORD jl_tls_key; +DWORD jl_tls_key; // Put this here for now. We can move this out later if we find more use for it. BOOLEAN WINAPI DllMain(IN HINSTANCE hDllHandle, IN DWORD nReason,