1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314
|
/*
* The Road goes ever on and on
* Down from the door where it began.
*
* [Bilbo on p.35 of _The Lord of the Rings_, I/i: "A Long-Expected Party"]
* [Frodo on p.73 of _The Lord of the Rings_, I/iii: "Three Is Company"]
*/
#define PERLIO_NOT_STDIO 0
#define PERL_IN_WIN32_PERLLIB_C
#include "EXTERN.h"
#include "perl.h"
#include "XSUB.h"
#include <winuser.h>
#ifdef PERL_IMPLICIT_SYS
#include "win32iop.h"
#include <fcntl.h>
#endif /* PERL_IMPLICIT_SYS */
/* Register any extra external extensions */
const char * const staticlinkmodules[] = {
"DynaLoader",
/* other similar records will be included from "perllibst.h" */
#define STATIC1
#include "perllibst.h"
NULL,
};
EXTERN_C void boot_DynaLoader (pTHX_ CV* cv);
/* other similar records will be included from "perllibst.h" */
#define STATIC2
#include "perllibst.h"
static void
xs_init(pTHX)
{
const char *file = __FILE__;
dXSUB_SYS;
newXS("DynaLoader::boot_DynaLoader", boot_DynaLoader, file);
/* other similar records will be included from "perllibst.h" */
#define STATIC3
#include "perllibst.h"
}
#ifdef PERL_IMPLICIT_SYS
#include "perlhost.h"
void
win32_checkTLS(PerlInterpreter *host_perl)
{
/* GCurThdId() is lightweight, but b/c of the ctrl-c/signals sometimes firing
in other random WinOS threads, that make the TIDs go out of sync.
This isn't always an error, although high chance of a SEGV in the next
couple milliseconds b/c of "Day 1 of Win32 port" Ctrl-C vs Perl bugs.
Google it for details. So this code, if TIDs don't match, do the full heavy
TlsGetValue() + misc fn calls. Then resync TIDs to keep this fast for
future calls to this fn. */
DWORD tid = GetCurrentThreadId();
if(tid != host_perl->Isys_intern.cur_tid) {
dTHX; /* heavyweight */
if (host_perl != my_perl) {
int *nowhere = NULL;
abort();
}
host_perl->Isys_intern.cur_tid = tid;
}
}
EXTERN_C void
perl_get_host_info(struct IPerlMemInfo* perlMemInfo,
struct IPerlMemInfo* perlMemSharedInfo,
struct IPerlMemInfo* perlMemParseInfo,
struct IPerlEnvInfo* perlEnvInfo,
struct IPerlStdIOInfo* perlStdIOInfo,
struct IPerlLIOInfo* perlLIOInfo,
struct IPerlDirInfo* perlDirInfo,
struct IPerlSockInfo* perlSockInfo,
struct IPerlProcInfo* perlProcInfo)
{
if (perlMemInfo) {
Copy(&perlMem, &perlMemInfo->perlMemList, perlMemInfo->nCount, void*);
perlMemInfo->nCount = (sizeof(struct IPerlMem)/sizeof(void*));
}
if (perlMemSharedInfo) {
Copy(&perlMem, &perlMemSharedInfo->perlMemList, perlMemSharedInfo->nCount, void*);
perlMemSharedInfo->nCount = (sizeof(struct IPerlMem)/sizeof(void*));
}
if (perlMemParseInfo) {
Copy(&perlMem, &perlMemParseInfo->perlMemList, perlMemParseInfo->nCount, void*);
perlMemParseInfo->nCount = (sizeof(struct IPerlMem)/sizeof(void*));
}
if (perlEnvInfo) {
Copy(&perlEnv, &perlEnvInfo->perlEnvList, perlEnvInfo->nCount, void*);
perlEnvInfo->nCount = (sizeof(struct IPerlEnv)/sizeof(void*));
}
if (perlStdIOInfo) {
Copy(&perlStdIO, &perlStdIOInfo->perlStdIOList, perlStdIOInfo->nCount, void*);
perlStdIOInfo->nCount = (sizeof(struct IPerlStdIO)/sizeof(void*));
}
if (perlLIOInfo) {
Copy(&perlLIO, &perlLIOInfo->perlLIOList, perlLIOInfo->nCount, void*);
perlLIOInfo->nCount = (sizeof(struct IPerlLIO)/sizeof(void*));
}
if (perlDirInfo) {
Copy(&perlDir, &perlDirInfo->perlDirList, perlDirInfo->nCount, void*);
perlDirInfo->nCount = (sizeof(struct IPerlDir)/sizeof(void*));
}
if (perlSockInfo) {
Copy(&perlSock, &perlSockInfo->perlSockList, perlSockInfo->nCount, void*);
perlSockInfo->nCount = (sizeof(struct IPerlSock)/sizeof(void*));
}
if (perlProcInfo) {
Copy(&perlProc, &perlProcInfo->perlProcList, perlProcInfo->nCount, void*);
perlProcInfo->nCount = (sizeof(struct IPerlProc)/sizeof(void*));
}
}
EXTERN_C PerlInterpreter*
perl_alloc_override(const struct IPerlMem** ppMem, const struct IPerlMem** ppMemShared,
const struct IPerlMem** ppMemParse, const struct IPerlEnv** ppEnv,
const struct IPerlStdIO** ppStdIO, const struct IPerlLIO** ppLIO,
const struct IPerlDir** ppDir, const struct IPerlSock** ppSock,
const struct IPerlProc** ppProc)
{
PerlInterpreter *my_perl = NULL;
CPerlHost* pHost = new CPerlHost(ppMem, ppMemShared, ppMemParse, ppEnv,
ppStdIO, ppLIO, ppDir, ppSock, ppProc);
if (pHost) {
my_perl = perl_alloc_using(&pHost->m_pHostperlMem,
&pHost->m_pHostperlMemShared,
&pHost->m_pHostperlMemParse,
&pHost->m_pHostperlEnv,
&pHost->m_pHostperlStdIO,
&pHost->m_pHostperlLIO,
&pHost->m_pHostperlDir,
&pHost->m_pHostperlSock,
&pHost->m_pHostperlProc);
if (my_perl) {
w32_internal_host = pHost;
pHost->host_perl = my_perl;
}
}
return my_perl;
}
EXTERN_C PerlInterpreter*
perl_alloc(void)
{
PerlInterpreter* my_perl = NULL;
CPerlHost* pHost = new CPerlHost();
if (pHost) {
my_perl = perl_alloc_using(&pHost->m_pHostperlMem,
&pHost->m_pHostperlMemShared,
&pHost->m_pHostperlMemParse,
&pHost->m_pHostperlEnv,
&pHost->m_pHostperlStdIO,
&pHost->m_pHostperlLIO,
&pHost->m_pHostperlDir,
&pHost->m_pHostperlSock,
&pHost->m_pHostperlProc);
if (my_perl) {
w32_internal_host = pHost;
pHost->host_perl = my_perl;
}
}
return my_perl;
}
EXTERN_C void
win32_delete_internal_host(void *h)
{
CPerlHost *host = (CPerlHost*)h;
delete host;
}
#endif /* PERL_IMPLICIT_SYS */
EXTERN_C HANDLE w32_perldll_handle;
EXTERN_C DllExport int
RunPerl(int argc, char **argv, char **env)
{
int exitstatus;
PerlInterpreter *my_perl, *new_perl = NULL;
bool use_environ = (env == environ);
PERL_SYS_INIT(&argc,&argv);
if (!(my_perl = perl_alloc()))
return (1);
perl_construct(my_perl);
PL_exit_flags |= PERL_EXIT_DESTRUCT_END;
PL_perl_destruct_level = 0;
/* PERL_SYS_INIT() may update the environment, e.g. via ansify_path().
* This may reallocate the RTL environment block. Therefore we need
* to make sure that `env` continues to have the same value as `environ`
* if we have been called this way. If we have been called with any
* other value for `env` then all environment munging by PERL_SYS_INIT()
* will be lost again.
*/
if (use_environ)
env = environ;
if (!perl_parse(my_perl, xs_init, argc, argv, env)) {
#if defined(TOP_CLONE) && defined(USE_ITHREADS) /* XXXXXX testing */
new_perl = perl_clone(my_perl, 1);
(void) perl_run(new_perl);
PERL_SET_THX(my_perl);
#else
(void) perl_run(my_perl);
#endif
}
exitstatus = perl_destruct(my_perl);
perl_free(my_perl);
#ifdef USE_ITHREADS
if (new_perl) {
PERL_SET_THX(new_perl);
exitstatus = perl_destruct(new_perl);
perl_free(new_perl);
}
#endif
PERL_SYS_TERM();
return (exitstatus);
}
EXTERN_C void
set_w32_module_name(void);
EXTERN_C void
EndSockets(void);
#ifdef __MINGW32__
EXTERN_C /* GCC in C++ mode mangles the name, otherwise */
#endif
BOOL APIENTRY
DllMain(HINSTANCE hModule, /* DLL module handle */
DWORD fdwReason, /* reason called */
LPVOID lpvReserved) /* reserved */
{
switch (fdwReason) {
/* The DLL is attaching to a process due to process
* initialization or a call to LoadLibrary.
*/
case DLL_PROCESS_ATTACH:
DisableThreadLibraryCalls((HMODULE)hModule);
w32_perldll_handle = hModule;
set_w32_module_name();
break;
/* The DLL is detaching from a process due to
* process termination or call to FreeLibrary.
*/
case DLL_PROCESS_DETACH:
/* As long as we use TerminateProcess()/TerminateThread() etc. for mimicking kill()
anything here had better be harmless if:
A. Not called at all.
B. Called after memory allocation for Heap has been forcibly removed by OS.
PerlIO_cleanup() was done here but fails (B).
*/
EndSockets();
#if defined(USE_ITHREADS)
if (PL_curinterp)
FREE_THREAD_KEY;
#endif
break;
/* The attached process creates a new thread. */
case DLL_THREAD_ATTACH:
break;
/* The thread of the attached process terminates. */
case DLL_THREAD_DETACH:
break;
default:
break;
}
return TRUE;
}
#if defined(USE_ITHREADS) && defined(PERL_IMPLICIT_SYS)
EXTERN_C PerlInterpreter *
perl_clone_host(PerlInterpreter* proto_perl, UV flags) {
dTHX;
CPerlHost *h;
h = new CPerlHost(*(CPerlHost*)PL_sys_intern.internal_host);
proto_perl = perl_clone_using(proto_perl, flags,
&h->m_pHostperlMem,
&h->m_pHostperlMemShared,
&h->m_pHostperlMemParse,
&h->m_pHostperlEnv,
&h->m_pHostperlStdIO,
&h->m_pHostperlLIO,
&h->m_pHostperlDir,
&h->m_pHostperlSock,
&h->m_pHostperlProc
);
proto_perl->Isys_intern.internal_host = h;
h->host_perl = proto_perl;
return proto_perl;
}
#endif
|