[PATCH v10 0/4] MR9265: crt: Run MSVC constructors and destructors
There are several motivations to enable ctor/dtor support: 1. ASan. This is not just about my wine-asan work, this is important for @bernhardu's PE asan work as well. The compiler generates ctors that poisons memory around global variables so misuses of them can be caught. This is the same both of our implementations of ASan. 2. Building third-party codebases against wine's CRT. This is not necessary for mingw-gcc or llvm-mingw, but the MSVC frontend of clang is another option for cross-compiling to Windows. However right now that requires Windows SDK. Bringing wine CRT to the point of being functional enough will give us another option. Could be useful for things like Proton. 3. (Controversial) Many dlls perform init/deinit tasks in `DllMain`. Some of them might benefit from a conversion to ctor/dtor. Since nothing in wine uses ctor/dtor right now, this change shouldn't have any impact by itself. -- v10: crt: Run MSVC constructors and destructors. winecrtend: New staticlib hosting fallbacks of initterm and initterm_e. winegcc: Merge .CRT sections for windows targets. winebuild: Generate start and end symbols for .CRT sections. https://gitlab.winehq.org/wine/wine/-/merge_requests/9265
From: Yuxuan Shui <yshui@codeweavers.com> --- tools/winebuild/build.h | 1 + tools/winebuild/spec16.c | 1 + tools/winebuild/spec32.c | 27 +++++++++++++++++++++++++++ 3 files changed, 29 insertions(+) diff --git a/tools/winebuild/build.h b/tools/winebuild/build.h index 3acf987aab2..0cabc4648db 100644 --- a/tools/winebuild/build.h +++ b/tools/winebuild/build.h @@ -301,6 +301,7 @@ extern void output_imports( DLLSPEC *spec ); extern void output_import_lib( DLLSPEC *spec, struct strarray files ); extern void output_static_lib( const char *output_name, struct strarray files, int create ); extern void output_exports( DLLSPEC *spec ); +extern void output_crt_sections(void); extern int load_res32_file( const char *name, DLLSPEC *spec ); extern void output_resources( DLLSPEC *spec ); extern void output_bin_resources( DLLSPEC *spec, unsigned int start_rva ); diff --git a/tools/winebuild/spec16.c b/tools/winebuild/spec16.c index face1c551d9..d16e97eabe6 100644 --- a/tools/winebuild/spec16.c +++ b/tools/winebuild/spec16.c @@ -790,6 +790,7 @@ void output_spec16_file( DLLSPEC *spec16 ) output_stubs( spec16 ); output_exports( spec32 ); output_imports( spec16 ); + output_crt_sections(); if (!strcmp( spec16->dll_name, "kernel" )) output_asm_relays16(); if (needs_get_pc_thunk) output_get_pc_thunk(); if (spec16->main_module) diff --git a/tools/winebuild/spec32.c b/tools/winebuild/spec32.c index 7099e82e0f6..ef280b7f51e 100644 --- a/tools/winebuild/spec32.c +++ b/tools/winebuild/spec32.c @@ -703,6 +703,32 @@ static void output_load_config(void) } +void output_crt_sections(void) +{ + /* Generate the start/end symbols for .CRT$X?? sections. The start symbol is put into + * .CRT$X?A, the end .CRT$X?Z. Since the linker sort .CRT$X?? sections by name, these symbols + * will end up at the right location.*/ + static const char sections[] = "ict"; + int i; + for (i = 0; sections[i]; i++) + { + char *symbol_name = strmake( "__x%c_a", sections[i] ); + output( "\t.section .CRT$X%cA\n", toupper( sections[i] ) ); + output( "\t.globl %s\n", asm_name( symbol_name ) ); + output( "\t.balign %u\n", get_ptr_size() ); + output( "%s:\n", asm_name( symbol_name ) ); + output( "\t%s 0\n", get_asm_ptr_keyword() ); + + symbol_name = strmake( "__x%c_z", sections[i] ); + output( "\t.section .CRT$X%cZ\n", toupper( sections[i] ) ); + output( "\t.globl %s\n", asm_name( symbol_name ) ); + output( "\t.balign %u\n", get_ptr_size() ); + output( "%s:\n", asm_name( symbol_name ) ); + output( "\t%s 0\n", get_asm_ptr_keyword() ); + } +} + + /******************************************************************* * output_module * @@ -841,6 +867,7 @@ void output_spec32_file( DLLSPEC *spec ) output_imports( spec ); if (needs_get_pc_thunk) output_get_pc_thunk(); output_load_config(); + output_crt_sections(); output_resources( spec ); output_gnu_stack_note(); close_output_file(); -- GitLab https://gitlab.winehq.org/wine/wine/-/merge_requests/9265
From: Yuxuan Shui <yshui@codeweavers.com> On mingw targets this is already done by ld's linker script. --- tools/winegcc/winegcc.c | 3 +++ 1 file changed, 3 insertions(+) diff --git a/tools/winegcc/winegcc.c b/tools/winegcc/winegcc.c index f8651804fa2..f5eb2a81d83 100644 --- a/tools/winegcc/winegcc.c +++ b/tools/winegcc/winegcc.c @@ -628,6 +628,9 @@ static struct strarray get_link_args( const char *output_name ) strarray_add( &link_args, strmake( "-Wl,-filealign:%s,-align:%s,-driver", file_align, section_align )); + /* Merge .CRT sections into .rdata */ + strarray_add( &link_args, "-Wl,-merge:.CRT=.rdata" ); + strarray_addall( &link_args, flags ); return link_args; -- GitLab https://gitlab.winehq.org/wine/wine/-/merge_requests/9265
From: Yuxuan Shui <yshui@codeweavers.com> --- configure | 2 ++ configure.ac | 1 + dlls/winecrtend/Makefile.in | 4 ++++ dlls/winecrtend/initterm.c | 18 ++++++++++++++++++ dlls/winecrtend/initterm_e.c | 16 ++++++++++++++++ tools/makedep.c | 2 ++ tools/winegcc/winegcc.c | 1 + 7 files changed, 44 insertions(+) create mode 100644 dlls/winecrtend/Makefile.in create mode 100644 dlls/winecrtend/initterm.c create mode 100644 dlls/winecrtend/initterm_e.c diff --git a/configure b/configure index a6fb235e2ed..6201d7eabc9 100755 --- a/configure +++ b/configure @@ -1541,6 +1541,7 @@ enable_winebth_sys enable_winebus_sys enable_winecoreaudio_drv enable_winecrt0 +enable_winecrtend enable_wined3d enable_winedmo enable_winegstreamer @@ -23217,6 +23218,7 @@ wine_fn_config_makefile dlls/winebth.sys enable_winebth_sys wine_fn_config_makefile dlls/winebus.sys enable_winebus_sys wine_fn_config_makefile dlls/winecoreaudio.drv enable_winecoreaudio_drv wine_fn_config_makefile dlls/winecrt0 enable_winecrt0 +wine_fn_config_makefile dlls/winecrtend enable_winecrtend wine_fn_config_makefile dlls/wined3d enable_wined3d wine_fn_config_makefile dlls/winedmo enable_winedmo wine_fn_config_makefile dlls/winegstreamer enable_winegstreamer diff --git a/configure.ac b/configure.ac index b8ae3db64a0..8c3cbad16f9 100644 --- a/configure.ac +++ b/configure.ac @@ -3359,6 +3359,7 @@ WINE_CONFIG_MAKEFILE(dlls/winebth.sys) WINE_CONFIG_MAKEFILE(dlls/winebus.sys) WINE_CONFIG_MAKEFILE(dlls/winecoreaudio.drv) WINE_CONFIG_MAKEFILE(dlls/winecrt0) +WINE_CONFIG_MAKEFILE(dlls/winecrtend) WINE_CONFIG_MAKEFILE(dlls/wined3d) WINE_CONFIG_MAKEFILE(dlls/winedmo) WINE_CONFIG_MAKEFILE(dlls/winegstreamer) diff --git a/dlls/winecrtend/Makefile.in b/dlls/winecrtend/Makefile.in new file mode 100644 index 00000000000..d0f04e50baa --- /dev/null +++ b/dlls/winecrtend/Makefile.in @@ -0,0 +1,4 @@ +STATICLIB = libwinecrtend.a +SOURCES = \ + initterm.c \ + initterm_e.c diff --git a/dlls/winecrtend/initterm.c b/dlls/winecrtend/initterm.c new file mode 100644 index 00000000000..448456e0f9f --- /dev/null +++ b/dlls/winecrtend/initterm.c @@ -0,0 +1,18 @@ +#include <process.h> +#include "minwindef.h" + +/* This is a fallback version of msvcrt's _initterm. Since _initterm is used in winecrt0, + * if the final object is linked with a version of msvcrt without the _initterm function, this + * version will be used. */ +void CDECL _initterm( _PVFV *start,_PVFV *end ) +{ + _PVFV* current = start; + + while (current<end) + { + if (*current) + (**current)(); + current++; + } +} + diff --git a/dlls/winecrtend/initterm_e.c b/dlls/winecrtend/initterm_e.c new file mode 100644 index 00000000000..f1d9ee65a61 --- /dev/null +++ b/dlls/winecrtend/initterm_e.c @@ -0,0 +1,16 @@ +typedef int (__cdecl *_PIFV)(void); + +/* This is a fallback version of msvcrt's _initterm_e. Since _initterm_e is used in winecrt0, + * if the final object is linked with a version of msvcrt without the _initterm_e function, this + * version will be used. */ +int __cdecl _initterm_e(_PIFV *table, _PIFV *end) +{ + int res = 0; + + while (!res && table < end) { + if (*table) + res = (**table)(); + table++; + } + return res; +} diff --git a/tools/makedep.c b/tools/makedep.c index efc7517f0c8..f491ecf1930 100644 --- a/tools/makedep.c +++ b/tools/makedep.c @@ -2279,6 +2279,7 @@ static struct strarray get_default_imports( const struct makefile *make, struct STRARRAY_FOR_EACH( imp, &imports ) if (!strcmp( imp, "winecrt0" )) return ret; strarray_add( &ret, "winecrt0" ); if (compiler_rt) strarray_add( &ret, compiler_rt ); + strarray_add( &ret, "winecrtend" ); return ret; } @@ -2293,6 +2294,7 @@ static struct strarray get_default_imports( const struct makefile *make, struct strarray_add( &ret, "kernel32" ); strarray_add( &ret, "ntdll" ); + strarray_add( &ret, "winecrtend" ); return ret; } diff --git a/tools/winegcc/winegcc.c b/tools/winegcc/winegcc.c index f5eb2a81d83..b1f36a29a67 100644 --- a/tools/winegcc/winegcc.c +++ b/tools/winegcc/winegcc.c @@ -1337,6 +1337,7 @@ static void build(struct strarray input_files, const char *output) if (is_win16_app) add_library(lib_dirs, &files, "kernel"); add_library(lib_dirs, &files, "kernel32"); add_library(lib_dirs, &files, "ntdll"); + add_library(lib_dirs, &files, "winecrtend"); } /* set default entry point, if needed */ -- GitLab https://gitlab.winehq.org/wine/wine/-/merge_requests/9265
From: Yuxuan Shui <yshui@codeweavers.com> --- dlls/msvcr100/Makefile.in | 1 + dlls/msvcr110/Makefile.in | 1 + dlls/msvcr120/Makefile.in | 1 + dlls/msvcr70/Makefile.in | 1 + dlls/msvcr71/Makefile.in | 1 + dlls/msvcr80/Makefile.in | 1 + dlls/msvcr90/Makefile.in | 1 + dlls/msvcrt/Makefile.in | 1 + dlls/msvcrt/crt_init.c | 36 ++++++++++++++++++++++++++++++++++++ dlls/msvcrt/crt_main.c | 2 ++ dlls/msvcrt/crt_wmain.c | 2 ++ dlls/ucrtbase/Makefile.in | 1 + dlls/winecrt0/crt_dllmain.c | 30 +++++++++++++++++++++++++++++- 13 files changed, 78 insertions(+), 1 deletion(-) create mode 100644 dlls/msvcrt/crt_init.c diff --git a/dlls/msvcr100/Makefile.in b/dlls/msvcr100/Makefile.in index a0a3f64d6bf..34182e69f19 100644 --- a/dlls/msvcr100/Makefile.in +++ b/dlls/msvcr100/Makefile.in @@ -10,6 +10,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr110/Makefile.in b/dlls/msvcr110/Makefile.in index e6798427795..fdbcba054d4 100644 --- a/dlls/msvcr110/Makefile.in +++ b/dlls/msvcr110/Makefile.in @@ -10,6 +10,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr120/Makefile.in b/dlls/msvcr120/Makefile.in index b233753fa2e..3465d77a31e 100644 --- a/dlls/msvcr120/Makefile.in +++ b/dlls/msvcr120/Makefile.in @@ -10,6 +10,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr70/Makefile.in b/dlls/msvcr70/Makefile.in index 9413738ae1f..be93d88764b 100644 --- a/dlls/msvcr70/Makefile.in +++ b/dlls/msvcr70/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr71/Makefile.in b/dlls/msvcr71/Makefile.in index 6e0b855cac3..c59ce97e8e4 100644 --- a/dlls/msvcr71/Makefile.in +++ b/dlls/msvcr71/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr80/Makefile.in b/dlls/msvcr80/Makefile.in index 2a84383f79d..78accf161e9 100644 --- a/dlls/msvcr80/Makefile.in +++ b/dlls/msvcr80/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcr90/Makefile.in b/dlls/msvcr90/Makefile.in index d476d2e3080..3dfc080e554 100644 --- a/dlls/msvcr90/Makefile.in +++ b/dlls/msvcr90/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcrt/Makefile.in b/dlls/msvcrt/Makefile.in index aa068b62e51..a4023568d71 100644 --- a/dlls/msvcrt/Makefile.in +++ b/dlls/msvcrt/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/msvcrt/crt_init.c b/dlls/msvcrt/crt_init.c new file mode 100644 index 00000000000..fed6d821a68 --- /dev/null +++ b/dlls/msvcrt/crt_init.c @@ -0,0 +1,36 @@ +/* Utility functions for calling MSVC/MingW ctors & dtors from mainCRTStartup. */ + +#if 0 +#pragma makedep implib +#endif + +#include <process.h> +#include <stdlib.h> +#include "minwindef.h" + +extern _PIFV __xi_a[]; +extern _PIFV __xi_z[]; +extern _PVFV __xc_a[]; +extern _PVFV __xc_z[]; +extern _PVFV __xt_a[]; +extern _PVFV __xt_z[]; + +int CDECL _initterm_e(_PIFV *table, _PIFV *end); +void CDECL _initterm(_PVFV *,_PVFV *); + +static __cdecl void do_global_dtors(void) +{ + _initterm(__xt_a, __xt_z); +} + +void do_global_ctors(void) +{ + if (_initterm_e(__xi_a, __xi_z) != 0) return; + _initterm(__xc_a, __xc_z); + +#ifdef _UCRT + _crt_atexit(do_global_dtors); +#else + _onexit((_onexit_t)do_global_dtors); +#endif +} diff --git a/dlls/msvcrt/crt_main.c b/dlls/msvcrt/crt_main.c index eb785e30bb0..5eb9e5032a0 100644 --- a/dlls/msvcrt/crt_main.c +++ b/dlls/msvcrt/crt_main.c @@ -31,6 +31,7 @@ #include "winternl.h" int __cdecl main(int argc, char **argv, char **env); +void do_global_ctors( void ); static const IMAGE_NT_HEADERS *get_nt_header( void ) { @@ -54,6 +55,7 @@ int __cdecl mainCRTStartup(void) __getmainargs(&argc, &argv, &env, 0, &new_mode); #endif _set_app_type(get_nt_header()->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_GUI ? _crt_gui_app : _crt_console_app); + do_global_ctors(); ret = main(argc, argv, env); diff --git a/dlls/msvcrt/crt_wmain.c b/dlls/msvcrt/crt_wmain.c index 79755d1a6e9..71cc0e48ec7 100644 --- a/dlls/msvcrt/crt_wmain.c +++ b/dlls/msvcrt/crt_wmain.c @@ -31,6 +31,7 @@ #include "winternl.h" int __cdecl wmain(int argc, WCHAR **argv, WCHAR **env); +void do_global_ctors(void); static const IMAGE_NT_HEADERS *get_nt_header( void ) { @@ -54,6 +55,7 @@ int __cdecl wmainCRTStartup(void) __wgetmainargs(&argc, &argv, &env, 0, &new_mode); #endif _set_app_type(get_nt_header()->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_GUI ? _crt_gui_app : _crt_console_app); + do_global_ctors(); ret = wmain(argc, argv, env); diff --git a/dlls/ucrtbase/Makefile.in b/dlls/ucrtbase/Makefile.in index a3bb54750af..1bd1073dcf1 100644 --- a/dlls/ucrtbase/Makefile.in +++ b/dlls/ucrtbase/Makefile.in @@ -9,6 +9,7 @@ SOURCES = \ console.c \ cpp.c \ crt_gccmain.c \ + crt_init.c \ crt_main.c \ crt_winmain.c \ crt_wmain.c \ diff --git a/dlls/winecrt0/crt_dllmain.c b/dlls/winecrt0/crt_dllmain.c index 181760c884a..4266a3737a5 100644 --- a/dlls/winecrt0/crt_dllmain.c +++ b/dlls/winecrt0/crt_dllmain.c @@ -25,9 +25,37 @@ #include "windef.h" #include "winbase.h" +typedef int (__cdecl *_PIFV)(void); +typedef void (__cdecl *_PVFV)(void); + +extern _PIFV __xi_a[]; +extern _PIFV __xi_z[]; +extern _PVFV __xc_a[]; +extern _PVFV __xc_z[]; +extern _PVFV __xt_a[]; +extern _PVFV __xt_z[]; + +void CDECL _initterm( _PVFV *start,_PVFV *end ); +int CDECL _initterm_e( _PIFV *table, _PIFV *end ); + +static void do_global_ctors(void) +{ + if (_initterm_e( __xi_a, __xi_z ) != 0) return; + _initterm( __xc_a, __xc_z ); +} + +static void do_global_dtors(void) +{ + _initterm( __xt_a, __xt_z ); +} + BOOL WINAPI DllMainCRTStartup( HINSTANCE inst, DWORD reason, void *reserved ) { - return DllMain( inst, reason, reserved ); + BOOL ret; + if (reason == DLL_PROCESS_ATTACH) do_global_ctors(); + ret = DllMain( inst, reason, reserved ); + if (reason == DLL_PROCESS_DETACH) do_global_dtors(); + return ret; } #endif -- GitLab https://gitlab.winehq.org/wine/wine/-/merge_requests/9265
On Sat Dec 13 00:07:51 2025 +0000, Bernhard Übelacker wrote:
I have removed the "--enable-werror" in [this job](https://gitlab.winehq.org/bernhardu/wine/-/jobs/217175#L159), but there I see now also such lines `lld-link: warning: literal .edata sections override exports`, which did not appear with the previous version of this MR. hi, i made some adjustments. does that help?
-- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126496
On Wed Jan 7 14:35:34 2026 +0000, Yuxuan Shui wrote:
hi, i made some adjustments. does that help? @jacek do you have thoughts on this? `corecrt_startup.h` defines `_initterm_e` as dllimport, but the fallback version is not in a implib, and lld-link complains:
``` lld-link: error: undefined symbol: __declspec(dllimport) _initterm_e
referenced by libwinecrt0.a(crt_dllmain.o):(DllMainCRTStartup) NOTE: a relevant symbol '_initterm_e' is available in dlls/winecrtend/x86_64-windows/libwinecrtend.a but cannot be used because it is not an import library.
--
https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126498
On Wed Jan 7 14:56:22 2026 +0000, Yuxuan Shui wrote:
@jacek do you have thoughts on this? `corecrt_startup.h` defines `_initterm_e` as dllimport, but the fallback version is not in a implib, and lld-link complains: ``` lld-link: error: undefined symbol: __declspec(dllimport) _initterm_e
referenced by libwinecrt0.a(crt_dllmain.o):(DllMainCRTStartup) NOTE: a relevant symbol '_initterm_e' is available in dlls/winecrtend/x86_64-windows/libwinecrtend.a but cannot be used because it is not an import library.
Also, the "pulling symbol in only when needed" behavior is GNU ld specific. With lld-link I also get `duplicate symbol: _initterm_e` when it's also available from msvcrt. Now I start to think "use `_initterm_e` from msvcrt when available" is not possible...
-- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126499
On Wed Jan 7 15:05:18 2026 +0000, Yuxuan Shui wrote:
Also, the "pulling symbol in only when needed" behavior is GNU ld specific. With lld-link I also get `duplicate symbol: _initterm_e` when it's also available from msvcrt. Now I start to think "use `_initterm_e` from msvcrt when available" is not possible... Here is a version that seems to work for me: https://gitlab.winehq.org/jacek/wine/-/commits/ctors. `_initterm` and `_initterm_e` need to be in separate object files to avoid duplicate symbol (some crt version implement only one variant; if pulling in the other variant pulls both of them from a static lib, we get a duplicate symbol).
I also made other clean ups and avoided the need for a new static library by tweaking makedep and using `__ASM_GLOBAL_IMPORT`. How does it look for you? -- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126669
On Tue Jan 13 01:30:22 2026 +0000, Jacek Caban wrote:
Here is a version that seems to work for me: https://gitlab.winehq.org/jacek/wine/-/commits/ctors. `_initterm` and `_initterm_e` need to be in separate object files to avoid duplicate symbol (some crt version implement only one variant; if pulling in the other variant pulls both of them from a static lib, we get a duplicate symbol). I also made other clean ups and avoided the need for a new static library by tweaking makedep and using `__ASM_GLOBAL_IMPORT`. How does it look for you? Thanks. I've tested it and it works for me as well.
* * * Can you help me understand why my attempt doesn't work?
`_initterm` and `_initterm_e` need to be in separate object files to avoid duplicate symbol
Yeah I have [that](https://gitlab.winehq.org/wine/wine/-/merge_requests/9265/diffs?commit_id=e7...) as well.
using `__ASM_GLOBAL_IMPORT`
Thanks. Adding it fixed this error for me: ``` NOTE: a relevant symbol '_initterm' is available in dlls/winecrtend/x86_64-windows/libwinecrtend.a but cannot be used because it is not an import library. ``` But I am still getting `error: duplicate symbol: __initterm`. My failed attempt is [here](https://gitlab.winehq.org/yshui/wine/-/commits/ctor-dtor-dup-symbols) -- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126881
On Wed Jan 14 00:23:26 2026 +0000, Yuxuan Shui wrote:
Thanks. I've tested it and it works for me as well. * * * Can you help me understand why my attempt doesn't work?
`_initterm` and `_initterm_e` need to be in separate object files to avoid duplicate symbol Yeah I have [that](https://gitlab.winehq.org/wine/wine/-/merge_requests/9265/diffs?commit_id=e7...) as well. using `__ASM_GLOBAL_IMPORT` Thanks. Adding it fixed this error for me:
NOTE: a relevant symbol '_initterm' is available in dlls/winecrtend/x86_64-windows/libwinecrtend.a but cannot be used because it is not an import library.But I am still getting `error: duplicate symbol: __initterm`. My failed attempt is [here](https://gitlab.winehq.org/yshui/wine/-/commits/ctor-dtor-dup-symbols) Oh, I need to add `__ASM_GLOBAL_IMPORT` to data.c as well? But why does that fix the duplicate symbol problem?
Hmm, I think I understand now. `crt_dllmain.o` is pulling in the `__imp__initterm` symbol, not the `_initterm` symbol itself. So I need to provide that to prevent `initerm.o` from being linked. -- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126882
On Wed Jan 14 00:49:46 2026 +0000, Yuxuan Shui wrote:
Oh, I need to add `__ASM_GLOBAL_IMPORT` to data.c as well? But why does that fix the duplicate symbol problem? Hmm, I think I understand now. `crt_dllmain.o` is pulling in the `__imp__initterm` symbol, not the `_initterm` symbol itself. So I need to provide that to prevent `initerm.o` from being linked. Yes, that’s the reason. Whenever a function is pulled in as dllimport, the compiler accesses it via the `__imp_` symbol, which caused it to be pulled from winecrtend when building the crt DLLs, even though those DLLs also provide the implementation.
I misremembered your original version, so the comment about separate object files does not apply to it; it applied to my intermediate local version I had while experimenting. -- https://gitlab.winehq.org/wine/wine/-/merge_requests/9265#note_126883
participants (3)
-
Jacek Caban (@jacek) -
Yuxuan Shui -
Yuxuan Shui (@yshui)