📄 api.xs
字号:
} else {
croak("Win32::API::Call: parameter %d must be an array reference!\n", i+1);
}
break;
case T_INTEGER:
params[i].t = T_NUMBER;
params[i].l = (long) (int) SvIV(ST(i+1));
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: params[%d].t=%d, .u=%d\n", i, params[i].t, params[i].l);
#endif
break;
case T_STRUCTURE:
{
MAGIC* mg;
params[i].t = T_STRUCTURE;
if(SvROK(ST(i+1))) {
mg = mg_find(SvRV(ST(i+1)), 'P');
if(mg != NULL) {
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: SvRV(ST(i+1)) has P magic\n");
#endif
origST[i] = mg->mg_obj;
// structs[i].object = mg->mg_obj;
} else {
origST[i] = ST(i+1);
// structs[i].object = ST(i+1);
}
}
}
break;
case T_CODE:
params[i].t = T_CODE;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: got a T_CODE, (SV=0x%08x) (SvPV='%s')\n", ST(i+1), SvPV_nolen(ST(i+1)));
#endif
if(SvROK(ST(i+1))) {
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: fetching code...\n");
#endif
code = hv_fetch((HV*) SvRV(ST(i+1)), "code", 4, 0);
if(code != NULL) {
params[i].l = SvIV(*code);
// callbacks[i].object = ST(i+1);
origST[i] = ST(i+1);
} else {
croak("Win32::API::Call: parameter %d must be a Win32::API::Callback object!\n", i+1);
}
} else {
croak("Win32::API::Call: parameter %d must be a Win32::API::Callback object!\n", i+1);
}
break;
}
}
/* #### SECOND PASS: fixup structures/callbacks/pointers... #### */
for(i = 0; i <= nin; i++) {
if(params[i].t == T_STRUCTURE) {
SV** buffer;
int count;
/*
ENTER;
SAVETMPS;
PUSHMARK(SP);
XPUSHs(sv_2mortal(newSVsv(structs[i].object)));
PUTBACK;
count = call_method("sizeof", G_SCALAR);
SPAGAIN;
structs[i].size = POPi;
PUTBACK;
FREETMPS;
LEAVE;
*/
ENTER;
SAVETMPS;
PUSHMARK(SP);
XPUSHs(sv_2mortal(newSVsv(origST[i])));
PUTBACK;
count = call_method("Pack", G_DISCARD);
PUTBACK;
FREETMPS;
LEAVE;
buffer = hv_fetch((HV*) SvRV(origST[i]), "buffer", 6, 0);
if(buffer != NULL) {
params[i].p = (char *) (LPBYTE) SvPV_nolen(*buffer);
} else {
params[i].p = NULL;
}
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: params[%d].t=%d, .u=%s (0x%08x)\n", i, params[i].t, params[i].p, params[i].p);
#endif
}
if(params[i].t == T_CODE) {
int count;
ENTER;
SAVETMPS;
PUSHMARK(SP);
XPUSHs(origST[i]);
PUTBACK;
count = call_method("PushSelf", G_DISCARD);
PUTBACK;
FREETMPS;
LEAVE;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: params[%d].t=%d, .u=0x%x\n", i, params[i].t, params[i].l);
#endif
}
}
/* #### PUSH THE PARAMETER ON THE (ASSEMBLER) STACK #### */
for(i = nin; i >= 0; i--) {
switch(params[i].t) {
case T_POINTER:
case T_STRUCTURE:
pParam = params[i].p;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (P) is %s\n", i, pParam);
#endif
_asm {
mov eax, dword ptr pParam
push eax
}
break;
case T_POINTERPOINTER:
ppParam = params[i].b;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (P) is %s\n", i, ppParam);
#endif
_asm {
mov eax, dword ptr ppParam
push eax
}
break;
case T_NUMBER:
case T_CHAR:
lParam = params[i].l;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (N) is %ld\n", i, lParam);
#endif
_asm {
mov eax, lParam
push eax
}
break;
case T_FLOAT:
fParam = params[i].f;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (F) is %f\n", i, fParam);
#endif
_asm {
mov eax, fParam
push eax
}
break;
case T_DOUBLE:
dParam = params[i].d;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (D) is %f\n", i, dParam);
#endif
_asm {
mov eax, dword ptr [dParam + 4]
push eax
mov eax, dword ptr [dParam]
push eax
}
break;
case T_CODE:
lParam = params[i].l;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: parameter %d (K) is 0x%x\n", i, lParam);
#endif
_asm {
mov eax, lParam
push eax
}
break;
}
}
}
/* #### NOW CALL THE FUNCTION #### */
switch(tout) {
case T_NUMBER:
ApiFunctionNumber = (ApiNumber *) ApiFunction;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionNumber()\n");
#endif
lReturn = ApiFunctionNumber();
break;
case T_FLOAT:
ApiFunctionFloat = (ApiFloat *) ApiFunction;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionFloat()\n");
#endif
// _asm {
// call dword ptr [ApiFunctionFloat]
// fstp qword ptr [fReturn]
// }
fReturn = ApiFunctionFloat();
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: ApiFunctionFloat returned %f\n", fReturn);
#endif
break;
case T_DOUBLE:
ApiFunctionDouble = (ApiDouble *) ApiFunction;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionDouble()\n");
#endif
_asm {
call dword ptr [ApiFunctionDouble]
fstp qword ptr [dReturn]
}
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: ApiFunctionDouble returned %f\n", dReturn);
#endif
break;
case T_POINTER:
ApiFunctionPointer = (ApiPointer *) ApiFunction;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionPointer()\n");
#endif
pReturn = ApiFunctionPointer();
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: ApiFunctionPointer returned 0x%x '%s'\n", pReturn, pReturn);
#endif
/* #### only works with strings... #### */
cReturn = (char *) safemalloc(strlen(pReturn));
strcpy(cReturn, pReturn);
break;
case T_INTEGER:
ApiFunctionInteger = (ApiInteger *) ApiFunction;
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionInteger()\n");
#endif
iReturn = ApiFunctionInteger();
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: ApiFunctionInteger returned %d\n", iReturn);
#endif
break;
case T_VOID:
default:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: Calling ApiFunctionVoid() (tout=%d)\n", tout);
#endif
ApiFunctionVoid = (ApiVoid *) ApiFunction;
ApiFunctionVoid();
break;
}
/* #### THIRD PASS: postfix pointers/structures #### */
for(i = 0; i <= nin; i++) {
if(params[i].t == T_POINTER && has_proto) {
pointerCallUnpack(origST[i], i, intypes);
}
if(params[i].t == T_STRUCTURE) {
ENTER;
SAVETMPS;
PUSHMARK(SP);
// XPUSHs(sv_2mortal(newSVsv(origST[i])));
XPUSHs(origST[i]);
PUTBACK;
call_method("Unpack", G_DISCARD);
PUTBACK;
FREETMPS;
LEAVE;
}
if(params[i].t == T_POINTERPOINTER) {
pparray = (AV*) SvRV(origST[i]);
av_extend(pparray, 2);
av_store(pparray, 1, newSViv(*(params[i].b)));
}
}
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: freeing memory...\n");
#endif
if(nin >= 0) {
safefree(params);
safefree(origST);
}
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning to caller.\n");
#endif
/* #### NOW PUSH THE RETURN VALUE ON THE (PERL) STACK #### */
EXTEND(SP, 1);
switch(tout) {
case T_NUMBER:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning %d.\n", lReturn);
#endif
XSRETURN_IV(lReturn);
break;
case T_FLOAT:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning %f.\n", fReturn);
#endif
XSRETURN_NV((double) fReturn);
break;
case T_DOUBLE:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning %f.\n", dReturn);
#endif
XSRETURN_NV(dReturn);
break;
case T_POINTER:
if(pReturn == NULL) {
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning NULL.\n");
#endif
XSRETURN_IV(0);
} else {
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning 0x%x '%s'\n", cReturn, cReturn);
#endif
XSRETURN_PV(cReturn);
}
break;
case T_INTEGER:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning %d.\n", iReturn);
#endif
XSRETURN_IV(iReturn);
break;
case T_VOID:
default:
#ifdef WIN32_API_DEBUG
printf("(XS)Win32::API::Call: returning UNDEF.\n");
#endif
XSRETURN_UNDEF;
break;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -