This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
Extract a common closure.
[perl5.git] / ext / DynaLoader / dl_win32.xs
CommitLineData
0a753a76 1/* dl_win32.xs
2 *
3 * Platform: Win32 (Windows NT/Windows 95)
4 * Author: Wei-Yuen Tan (wyt@hip.com)
5 * Created: A warm day in June, 1995
6 *
7 * Modified:
8 * August 23rd 1995 - rewritten after losing everything when I
9 * wiped off my NT partition (eek!)
10 */
11
12/* Porting notes:
13
14I merely took Paul's dl_dlopen.xs, took out extraneous stuff and
15replaced the appropriate SunOS calls with the corresponding Win32
16calls.
17
18*/
19
20#define WIN32_LEAN_AND_MEAN
a835ef8a
NIS
21#ifdef __GNUC__
22#define Win32_Winsock
23#endif
0a753a76 24#include <windows.h>
25#include <string.h>
26
c5be433b
GS
27#define PERL_NO_GET_CONTEXT
28
0a753a76 29#include "EXTERN.h"
30#include "perl.h"
eda5ff31 31#include "win32.h"
565764a8 32
0a753a76 33#include "XSUB.h"
34
cdc73a10
JH
35typedef struct {
36 SV * x_error_sv;
37} my_cxtx_t; /* this *must* be named my_cxtx_t */
38
39#define DL_CXT_EXTRA /* ask for dl_cxtx to be defined in dlutils.c */
40#include "dlutils.c" /* SaveError() etc */
41
42#define dl_error_sv (dl_cxtx.x_error_sv)
eda5ff31
GS
43
44static char *
acfe0abc 45OS_Error_String(pTHX)
eda5ff31 46{
cdc73a10
JH
47 dMY_CXT;
48 DWORD err = GetLastError();
49 STRLEN len;
50 if (!dl_error_sv)
51 dl_error_sv = newSVpvn("",0);
52 PerlProc_GetOSError(dl_error_sv,err);
53 return SvPV(dl_error_sv,len);
eda5ff31
GS
54}
55
0a753a76 56static void
acfe0abc 57dl_private_init(pTHX)
0a753a76 58{
acfe0abc 59 (void)dl_generic_private_init(aTHX);
0a753a76 60}
61
26b3385c
DL
62/*
63 This function assumes the list staticlinkmodules
64 will be formed from package names with '::' replaced
65 with '/'. Thus Win32::OLE is in the list as Win32/OLE
66*/
0a753a76 67static int
68dl_static_linked(char *filename)
69{
6c6c7914 70 const char * const *p;
d2b25974 71 char *ptr, *hptr;
fe1c5936 72 static const char subStr[] = "/auto/";
26b3385c
DL
73 char szBuffer[MAX_PATH];
74
0e244b13
JD
75 /* avoid buffer overflow when called with invalid filenames */
76 if (strlen(filename) >= sizeof(szBuffer))
77 return 0;
78
26b3385c
DL
79 /* change all the '\\' to '/' */
80 strcpy(szBuffer, filename);
81 for(ptr = szBuffer; ptr = strchr(ptr, '\\'); ++ptr)
82 *ptr = '/';
83
84 /* delete the file name */
85 ptr = strrchr(szBuffer, '/');
86 if(ptr != NULL)
87 *ptr = '\0';
88
89 /* remove leading lib path */
90 ptr = strstr(szBuffer, subStr);
91 if(ptr != NULL)
92 ptr += sizeof(subStr)-1;
93 else
94 ptr = szBuffer;
95
6c6c7914 96 for (p = staticlinkmodules; *p;p++) {
d2b25974
VK
97 if (hptr = strstr(ptr, *p)) {
98 /* found substring, need more detailed check if module name match */
99 if (hptr==ptr) {
100 return strcmp(ptr, *p)==0;
101 }
102 if (hptr[strlen(*p)] == 0)
103 return hptr[-1]=='/';
104 }
68dc0745 105 };
106 return 0;
0a753a76 107}
108
109MODULE = DynaLoader PACKAGE = DynaLoader
110
111BOOT:
acfe0abc 112 (void)dl_private_init(aTHX);
0a753a76 113
63d7ac5f 114void
0a753a76 115dl_load_file(filename,flags=0)
116 char * filename
117 int flags
118 PREINIT:
63d7ac5f 119 void *retv;
0a753a76 120 CODE:
b9010385 121 {
bf49b057 122 DLDEBUG(1,PerlIO_printf(Perl_debug_log,"dl_load_file(%s):\n", filename));
7fac1903 123 if (dl_static_linked(filename) == 0) {
63d7ac5f 124 retv = PerlProc_DynaLoad(filename);
7fac1903 125 }
68dc0745 126 else
63d7ac5f
S
127 retv = (void*) Win_GetModuleHandle(NULL);
128 DLDEBUG(2,PerlIO_printf(Perl_debug_log," libref=%x\n", retv));
0a753a76 129 ST(0) = sv_newmortal() ;
63d7ac5f 130 if (retv == NULL)
acfe0abc
GS
131 SaveError(aTHX_ "load_file:%s",
132 OS_Error_String(aTHX)) ;
0a753a76 133 else
63d7ac5f 134 sv_setiv( ST(0), (IV)retv);
b9010385 135 }
0a753a76 136
6a57da86
D
137int
138dl_unload_file(libref)
139 void * libref
140 CODE:
141 DLDEBUG(1,PerlIO_printf(Perl_debug_log, "dl_unload_file(%lx):\n", PTR2ul(libref)));
142 RETVAL = FreeLibrary(libref);
143 if (!RETVAL)
144 SaveError(aTHX_ "unload_file:%s", OS_Error_String(aTHX)) ;
145 DLDEBUG(2,PerlIO_printf(Perl_debug_log, " retval = %d\n", RETVAL));
146 OUTPUT:
147 RETVAL
148
63d7ac5f 149void
0a753a76 150dl_find_symbol(libhandle, symbolname)
151 void * libhandle
152 char * symbolname
63d7ac5f
S
153 PREINIT:
154 void *retv;
0a753a76 155 CODE:
bf49b057 156 DLDEBUG(2,PerlIO_printf(Perl_debug_log,"dl_find_symbol(handle=%x, symbol=%s)\n",
68dc0745 157 libhandle, symbolname));
63d7ac5f
S
158 retv = (void*) GetProcAddress((HINSTANCE) libhandle, symbolname);
159 DLDEBUG(2,PerlIO_printf(Perl_debug_log," symbolref = %x\n", retv));
0a753a76 160 ST(0) = sv_newmortal() ;
63d7ac5f 161 if (retv == NULL)
acfe0abc
GS
162 SaveError(aTHX_ "find_symbol:%s",
163 OS_Error_String(aTHX)) ;
0a753a76 164 else
63d7ac5f 165 sv_setiv( ST(0), (IV)retv);
0a753a76 166
167
168void
169dl_undef_symbols()
63d7ac5f 170 CODE:
0a753a76 171
172
173
174# These functions should not need changing on any platform:
175
176void
177dl_install_xsub(perl_name, symref, filename="$Package")
178 char * perl_name
179 void * symref
180 char * filename
181 CODE:
bf49b057 182 DLDEBUG(2,PerlIO_printf(Perl_debug_log,"dl_install_xsub(name=%s, symref=%x)\n",
68dc0745 183 perl_name, symref));
4f63d024 184 ST(0) = sv_2mortal(newRV((SV*)newXS(perl_name,
acfe0abc 185 (void(*)(pTHX_ CV *))symref,
4f63d024 186 filename)));
0a753a76 187
188
189char *
190dl_error()
191 CODE:
cdc73a10
JH
192 dMY_CXT;
193 RETVAL = dl_last_error;
0a753a76 194 OUTPUT:
195 RETVAL
196
8c472fc1
CB
197#if defined(USE_ITHREADS)
198
199void
200CLONE(...)
201 CODE:
202 MY_CXT_CLONE;
203
204 /* MY_CXT_CLONE just does a memcpy on the whole structure, so to avoid
205 * using Perl variables that belong to another thread, we create our
206 * own for this thread.
207 */
208 MY_CXT.x_dl_last_error = newSVpvn("", 0);
209
210#endif
211
0a753a76 212# end.