Changeset 778


Ignore:
Timestamp:
Jan 21, 2007, 1:29:17 AM (18 years ago)
Author:
bird
Message:

Attempt at dealing with deadlocked kmk on Ctrl-C by dispatching the SIGINT/SIGBREAK on the main thread instead of the ctrl-event thread (Windows).

Location:
trunk/src/gmake
Files:
3 edited

Legend:

Unmodified
Added
Removed
  • TabularUnified trunk/src/gmake/Makefile.kmk

    r775 r778  
    4646        BUILD_PLATFORM_ARCH=\"$(BUILD_TARGET_ARCH)\" \
    4747        BUILD_PLATFORM_CPU=\"$(BUILD_TARGET_CPU)\"
     48kmk_DEFS.win = CONFIG_NEW_WIN32_CTRL_EVENT
    4849
    4950kmk_SOURCES = \
  • TabularUnified trunk/src/gmake/commands.c

    r765 r778  
    442442  exit (10);
    443443#else /* not Amiga */
    444 #ifdef WINDOWS32
     444#if defined(WINDOWS32) && !defined(CONFIG_NEW_WIN32_CTRL_EVENT)
    445445  extern HANDLE main_thread;
    446446
     
    516516
    517517  remove_intermediates (1);
    518 
    519518#ifdef SIGQUIT
    520519  if (sig == SIGQUIT)
     
    525524
    526525#ifdef WINDOWS32
     526#ifndef CONFIG_NEW_WIN32_CTRL_EVENT
    527527  if (main_thread)
    528528    CloseHandle (main_thread);
     529#endif /* !CONFIG_NEW_WIN32_CTRL_EVENT */
    529530  /* Cannot call W32_kill with a pid (it needs a handle).  The exit
    530531     status of 130 emulates what happens in Bash.  */
  • TabularUnified trunk/src/gmake/main.c

    r765 r778  
    927927  return (sh_found);
    928928}
     929
     930/* bird: */
     931#ifdef CONFIG_NEW_WIN32_CTRL_EVENT
     932#include <process.h>
     933static UINT g_tidMainThread = 0;
     934static int g_sigPending = 0; /* lazy bird */
     935
     936static __declspec(naked) void dispatch_stub(void)
     937{
     938    __asm  {
     939        pushfd
     940        pushad
     941        cld
     942    }
     943    fflush(stdout);
     944    /*fprintf(stderr, "dbg: raising %s on the main thread (%d)\n", g_sigPending == SIGINT ? "SIGINT" : "SIGBREAK", _getpid());*/
     945    raise(g_sigPending);
     946    __asm  {
     947        popad
     948        popfd
     949        ret
     950    }
     951}
     952
     953static BOOL WINAPI ctrl_event(DWORD CtrlType)
     954{
     955    int sig = (CtrlType == CTRL_C_EVENT) ? SIGINT : SIGBREAK;
     956    HANDLE hThread;
     957    CONTEXT Ctx;
     958
     959    /* open the main thread and suspend it. */
     960    hThread = OpenThread(THREAD_ALL_ACCESS, FALSE, g_tidMainThread);
     961    SuspendThread(hThread);
     962
     963    /* Get the thread context and */
     964    memset(&Ctx, 0, sizeof(Ctx));
     965    Ctx.ContextFlags = CONTEXT_FULL;
     966    GetThreadContext(hThread, &Ctx);
     967
     968    /* If we've got a valid Esp, dispatch it on the main thread
     969       otherwise raise the signal in the ctrl-event thread (this). */
     970    if (Ctx.Esp >= 0x1000)
     971    {
     972        ((uintptr_t *)Ctx.Esp)[-1] = Ctx.Eip;
     973        Ctx.Esp -= sizeof(uintptr_t);
     974        Ctx.Eip = (uintptr_t)&dispatch_stub;
     975
     976        SetThreadContext(hThread, &Ctx);
     977        g_sigPending = sig;
     978        ResumeThread(hThread);
     979        CloseHandle(hThread);
     980    }
     981    else
     982    {
     983        fprintf(stderr, "dbg: raising %s on the ctrl-event thread (%d)\n", sig == SIGINT ? "SIGINT" : "SIGBREAK", _getpid());
     984        raise(sig);
     985        ResumeThread(hThread);
     986        CloseHandle(hThread);
     987        exit(130);
     988    }
     989
     990    Sleep(1);
     991    return TRUE;
     992}
     993#endif /* CONFIG_NEW_WIN32_CTRL_EVENT */
     994
    929995#endif  /* WINDOWS32 */
    930996
     
    10951161  FATAL_SIG (SIGXFSZ);
    10961162#endif
     1163
     1164#ifdef CONFIG_NEW_WIN32_CTRL_EVENT
     1165  /* bird: dispatch signals in our own way to try avoid deadlocks. */
     1166  g_tidMainThread = GetCurrentThreadId ();
     1167  SetConsoleCtrlHandler (ctrl_event, TRUE);
     1168#endif /* CONFIG_NEW_WIN32_CTRL_EVENT */
    10971169
    10981170#undef  FATAL_SIG
Note: See TracChangeset for help on using the changeset viewer.