/*********************************************************************** * QueueEvent (WINEDOS.@) */ void WINAPI DOSVM_QueueEvent( INT irq, INT priority, DOSRELAY relay, LPVOID data) { LPDOSEVENT event, cur, prev; BOOL old_pending; if (MZ_Current()) { event = malloc(sizeof(DOSEVENT)); if (!event) { ERR("out of memory allocating event entry\n"); return; } event->irq = irq; event->priority = priority; event->relay = relay; event->data = data; EnterCriticalSection(&qcrit); old_pending = DOSVM_HasPendingEvents(); /* insert event into linked list, in order *after* * all earlier events of higher or equal priority */ cur = pending_event; prev = NULL; while (cur && cur->priority<=priority) { prev = cur; cur = cur->next; } event->next = cur; if (prev) prev->next = event; else pending_event = event; if (!old_pending && DOSVM_HasPendingEvents()) { TRACE("new event queued, signalling (time=%d)\n", GetTickCount()); /* Alert VM86 thread about the new event. */ kill(dosvm_pid,SIGUSR2); /* Wake up DOSVM_Wait so that it can serve pending events. */ SetEvent(event_notifier); } else { TRACE("new event queued (time=%d)\n", GetTickCount()); } LeaveCriticalSection(&qcrit); } else { /* DOS subsystem not running */ /* (this probably means that we're running a win16 app * which uses DPMI to thunk down to DOS services) */ if (irq<0) { /* callback event, perform it with dummy context */ CONTEXT86 context; memset(&context,0,sizeof(context)); (*relay)(&context,data); } else { ERR("IRQ without DOS task: should not happen\n"); } } }
/********************************************************************** * DPMI_CallRMProc * * This routine does the hard work of calling a real mode procedure. */ int DPMI_CallRMProc( CONTEXT86 *context, LPWORD stack, int args, int iret ) { LPWORD stack16; LPVOID addr = NULL; /* avoid gcc warning */ RMCB *CurrRMCB; int alloc = 0, already = 0; BYTE *code; TRACE("EAX=%08lx EBX=%08lx ECX=%08lx EDX=%08lx\n", context->Eax, context->Ebx, context->Ecx, context->Edx ); TRACE("ESI=%08lx EDI=%08lx ES=%04lx DS=%04lx CS:IP=%04lx:%04x, %d WORD arguments, %s\n", context->Esi, context->Edi, context->SegEs, context->SegDs, context->SegCs, LOWORD(context->Eip), args, iret?"IRET":"FAR" ); callrmproc_again: /* there might be some code that just jumps to RMCBs or the like, in which case following the jumps here might get us to a shortcut */ code = CTX_SEG_OFF_TO_LIN(context, context->SegCs, context->Eip); switch (*code) { case 0xe9: /* JMP NEAR */ context->Eip += 3 + *(WORD *)(code+1); /* yeah, I know these gotos don't look good... */ goto callrmproc_again; case 0xea: /* JMP FAR */ context->Eip = *(WORD *)(code+1); context->SegCs = *(WORD *)(code+3); /* ...but since the label is there anyway... */ goto callrmproc_again; case 0xeb: /* JMP SHORT */ context->Eip += 2 + *(signed char *)(code+1); /* ...because of other gotos below, so... */ goto callrmproc_again; } /* shortcut for chaining to internal interrupt handlers */ if ((context->SegCs == 0xF000) && iret) { DOSVM_CallBuiltinHandler( context, LOWORD(context->Eip)/4 ); return 0; } /* shortcut for RMCBs */ CurrRMCB = FirstRMCB; while (CurrRMCB && (HIWORD(CurrRMCB->address) != context->SegCs)) CurrRMCB = CurrRMCB->next; if (!CurrRMCB && !MZ_Current()) { FIXME("DPMI real-mode call using DOS VM task system, not fully tested!\n"); TRACE("creating VM86 task\n"); MZ_AllocDPMITask(); } if (!already) { if (!context->SegSs) { alloc = 1; /* allocate default stack */ stack16 = addr = DOSMEM_AllocBlock( 64, (UINT16 *)&(context->SegSs) ); context->Esp = 64-2; stack16 += 32-1; if (!addr) { ERR("could not allocate default stack\n"); return 1; } } else { stack16 = CTX_SEG_OFF_TO_LIN(context, context->SegSs, context->Esp); } context->Esp -= (args + (iret?1:0)) * sizeof(WORD); stack16 -= args; if (args) memcpy(stack16, stack, args*sizeof(WORD) ); /* push flags if iret */ if (iret) { stack16--; args++; *stack16 = LOWORD(context->EFlags); } /* push return address (return to interrupt wrapper) */ *(--stack16) = DOSVM_dpmi_segments->wrap_seg; *(--stack16) = 0; /* adjust stack */ context->Esp -= 2*sizeof(WORD); already = 1; } if (CurrRMCB) { /* RMCB call, invoke protected-mode handler directly */ DPMI_CallRMCBProc(context, CurrRMCB, dpmi_flag); /* check if we returned to where we thought we would */ if ((context->SegCs != DOSVM_dpmi_segments->wrap_seg) || (LOWORD(context->Eip) != 0)) { /* we need to continue at different address in real-mode space, so we need to set it all up for real mode again */ goto callrmproc_again; } } else { TRACE("entering real mode...\n"); DOSVM_Enter( context ); TRACE("returned from real-mode call\n"); } if (alloc) DOSMEM_FreeBlock( addr ); return 0; }