| [25] | 1 | /* | 
|---|
|  | 2 | * tclUnixNotify.c -- | 
|---|
|  | 3 | * | 
|---|
|  | 4 | *      This file contains the implementation of the select()-based | 
|---|
|  | 5 | *      Unix-specific notifier, which is the lowest-level part of the Tcl | 
|---|
|  | 6 | *      event loop. This file works together with generic/tclNotify.c. | 
|---|
|  | 7 | * | 
|---|
|  | 8 | * Copyright (c) 1995-1997 Sun Microsystems, Inc. | 
|---|
|  | 9 | * | 
|---|
|  | 10 | * See the file "license.terms" for information on usage and redistribution of | 
|---|
|  | 11 | * this file, and for a DISCLAIMER OF ALL WARRANTIES. | 
|---|
|  | 12 | * | 
|---|
|  | 13 | * RCS: @(#) $Id: tclUnixNotfy.c,v 1.34 2008/03/11 22:23:50 das Exp $ | 
|---|
|  | 14 | */ | 
|---|
|  | 15 |  | 
|---|
|  | 16 | #include "tclInt.h" | 
|---|
|  | 17 | #ifndef HAVE_COREFOUNDATION     /* Darwin/Mac OS X CoreFoundation notifier is | 
|---|
|  | 18 | * in tclMacOSXNotify.c */ | 
|---|
|  | 19 | #include <signal.h> | 
|---|
|  | 20 |  | 
|---|
|  | 21 | /* | 
|---|
|  | 22 | * This code does deep stub magic to allow replacement of the notifier at | 
|---|
|  | 23 | * runtime. | 
|---|
|  | 24 | */ | 
|---|
|  | 25 |  | 
|---|
|  | 26 | extern TclStubs tclStubs; | 
|---|
|  | 27 | extern Tcl_NotifierProcs tclOriginalNotifier; | 
|---|
|  | 28 |  | 
|---|
|  | 29 | /* | 
|---|
|  | 30 | * This structure is used to keep track of the notifier info for a registered | 
|---|
|  | 31 | * file. | 
|---|
|  | 32 | */ | 
|---|
|  | 33 |  | 
|---|
|  | 34 | typedef struct FileHandler { | 
|---|
|  | 35 | int fd; | 
|---|
|  | 36 | int mask;                   /* Mask of desired events: TCL_READABLE, | 
|---|
|  | 37 | * etc. */ | 
|---|
|  | 38 | int readyMask;              /* Mask of events that have been seen since | 
|---|
|  | 39 | * the last time file handlers were invoked | 
|---|
|  | 40 | * for this file. */ | 
|---|
|  | 41 | Tcl_FileProc *proc;         /* Function to call, in the style of | 
|---|
|  | 42 | * Tcl_CreateFileHandler. */ | 
|---|
|  | 43 | ClientData clientData;      /* Argument to pass to proc. */ | 
|---|
|  | 44 | struct FileHandler *nextPtr;/* Next in list of all files we care about. */ | 
|---|
|  | 45 | } FileHandler; | 
|---|
|  | 46 |  | 
|---|
|  | 47 | /* | 
|---|
|  | 48 | * The following structure is what is added to the Tcl event queue when file | 
|---|
|  | 49 | * handlers are ready to fire. | 
|---|
|  | 50 | */ | 
|---|
|  | 51 |  | 
|---|
|  | 52 | typedef struct FileHandlerEvent { | 
|---|
|  | 53 | Tcl_Event header;           /* Information that is standard for all | 
|---|
|  | 54 | * events. */ | 
|---|
|  | 55 | int fd;                     /* File descriptor that is ready. Used to find | 
|---|
|  | 56 | * the FileHandler structure for the file | 
|---|
|  | 57 | * (can't point directly to the FileHandler | 
|---|
|  | 58 | * structure because it could go away while | 
|---|
|  | 59 | * the event is queued). */ | 
|---|
|  | 60 | } FileHandlerEvent; | 
|---|
|  | 61 |  | 
|---|
|  | 62 | /* | 
|---|
|  | 63 | * The following structure contains a set of select() masks to track readable, | 
|---|
|  | 64 | * writable, and exceptional conditions. | 
|---|
|  | 65 | */ | 
|---|
|  | 66 |  | 
|---|
|  | 67 | typedef struct SelectMasks { | 
|---|
|  | 68 | fd_set readable; | 
|---|
|  | 69 | fd_set writable; | 
|---|
|  | 70 | fd_set exceptional; | 
|---|
|  | 71 | } SelectMasks; | 
|---|
|  | 72 |  | 
|---|
|  | 73 | /* | 
|---|
|  | 74 | * The following static structure contains the state information for the | 
|---|
|  | 75 | * select based implementation of the Tcl notifier. One of these structures is | 
|---|
|  | 76 | * created for each thread that is using the notifier. | 
|---|
|  | 77 | */ | 
|---|
|  | 78 |  | 
|---|
|  | 79 | typedef struct ThreadSpecificData { | 
|---|
|  | 80 | FileHandler *firstFileHandlerPtr; | 
|---|
|  | 81 | /* Pointer to head of file handler list. */ | 
|---|
|  | 82 | SelectMasks checkMasks;     /* This structure is used to build up the | 
|---|
|  | 83 | * masks to be used in the next call to | 
|---|
|  | 84 | * select. Bits are set in response to calls | 
|---|
|  | 85 | * to Tcl_CreateFileHandler. */ | 
|---|
|  | 86 | SelectMasks readyMasks;     /* This array reflects the readable/writable | 
|---|
|  | 87 | * conditions that were found to exist by the | 
|---|
|  | 88 | * last call to select. */ | 
|---|
|  | 89 | int numFdBits;              /* Number of valid bits in checkMasks (one | 
|---|
|  | 90 | * more than highest fd for which | 
|---|
|  | 91 | * Tcl_WatchFile has been called). */ | 
|---|
|  | 92 | #ifdef TCL_THREADS | 
|---|
|  | 93 | int onList;                 /* True if it is in this list */ | 
|---|
|  | 94 | unsigned int pollState;     /* pollState is used to implement a polling | 
|---|
|  | 95 | * handshake between each thread and the | 
|---|
|  | 96 | * notifier thread. Bits defined below. */ | 
|---|
|  | 97 | struct ThreadSpecificData *nextPtr, *prevPtr; | 
|---|
|  | 98 | /* All threads that are currently waiting on | 
|---|
|  | 99 | * an event have their ThreadSpecificData | 
|---|
|  | 100 | * structure on a doubly-linked listed formed | 
|---|
|  | 101 | * from these pointers. You must hold the | 
|---|
|  | 102 | * notifierMutex lock before accessing these | 
|---|
|  | 103 | * fields. */ | 
|---|
|  | 104 | Tcl_Condition waitCV;       /* Any other thread alerts a notifier that an | 
|---|
|  | 105 | * event is ready to be processed by signaling | 
|---|
|  | 106 | * this condition variable. */ | 
|---|
|  | 107 | int eventReady;             /* True if an event is ready to be processed. | 
|---|
|  | 108 | * Used as condition flag together with waitCV | 
|---|
|  | 109 | * above. */ | 
|---|
|  | 110 | #endif | 
|---|
|  | 111 | } ThreadSpecificData; | 
|---|
|  | 112 |  | 
|---|
|  | 113 | static Tcl_ThreadDataKey dataKey; | 
|---|
|  | 114 |  | 
|---|
|  | 115 | #ifdef TCL_THREADS | 
|---|
|  | 116 | /* | 
|---|
|  | 117 | * The following static indicates the number of threads that have initialized | 
|---|
|  | 118 | * notifiers. | 
|---|
|  | 119 | * | 
|---|
|  | 120 | * You must hold the notifierMutex lock before accessing this variable. | 
|---|
|  | 121 | */ | 
|---|
|  | 122 |  | 
|---|
|  | 123 | static int notifierCount = 0; | 
|---|
|  | 124 |  | 
|---|
|  | 125 | /* | 
|---|
|  | 126 | * The following variable points to the head of a doubly-linked list of | 
|---|
|  | 127 | * ThreadSpecificData structures for all threads that are currently waiting on | 
|---|
|  | 128 | * an event. | 
|---|
|  | 129 | * | 
|---|
|  | 130 | * You must hold the notifierMutex lock before accessing this list. | 
|---|
|  | 131 | */ | 
|---|
|  | 132 |  | 
|---|
|  | 133 | static ThreadSpecificData *waitingListPtr = NULL; | 
|---|
|  | 134 |  | 
|---|
|  | 135 | /* | 
|---|
|  | 136 | * The notifier thread spends all its time in select() waiting for a file | 
|---|
|  | 137 | * descriptor associated with one of the threads on the waitingListPtr list to | 
|---|
|  | 138 | * do something interesting. But if the contents of the waitingListPtr list | 
|---|
|  | 139 | * ever changes, we need to wake up and restart the select() system call. You | 
|---|
|  | 140 | * can wake up the notifier thread by writing a single byte to the file | 
|---|
|  | 141 | * descriptor defined below. This file descriptor is the input-end of a pipe | 
|---|
|  | 142 | * and the notifier thread is listening for data on the output-end of the same | 
|---|
|  | 143 | * pipe. Hence writing to this file descriptor will cause the select() system | 
|---|
|  | 144 | * call to return and wake up the notifier thread. | 
|---|
|  | 145 | * | 
|---|
|  | 146 | * You must hold the notifierMutex lock before accessing this list. | 
|---|
|  | 147 | */ | 
|---|
|  | 148 |  | 
|---|
|  | 149 | static int triggerPipe = -1; | 
|---|
|  | 150 |  | 
|---|
|  | 151 | /* | 
|---|
|  | 152 | * The notifierMutex locks access to all of the global notifier state. | 
|---|
|  | 153 | */ | 
|---|
|  | 154 |  | 
|---|
|  | 155 | TCL_DECLARE_MUTEX(notifierMutex) | 
|---|
|  | 156 |  | 
|---|
|  | 157 | /* | 
|---|
|  | 158 | * The notifier thread signals the notifierCV when it has finished | 
|---|
|  | 159 | * initializing the triggerPipe and right before the notifier thread | 
|---|
|  | 160 | * terminates. | 
|---|
|  | 161 | */ | 
|---|
|  | 162 |  | 
|---|
|  | 163 | static Tcl_Condition notifierCV; | 
|---|
|  | 164 |  | 
|---|
|  | 165 | /* | 
|---|
|  | 166 | * The pollState bits | 
|---|
|  | 167 | *      POLL_WANT is set by each thread before it waits on its condition | 
|---|
|  | 168 | *              variable. It is checked by the notifier before it does select. | 
|---|
|  | 169 | *      POLL_DONE is set by the notifier if it goes into select after seeing | 
|---|
|  | 170 | *              POLL_WANT. The idea is to ensure it tries a select with the | 
|---|
|  | 171 | *              same bits the initial thread had set. | 
|---|
|  | 172 | */ | 
|---|
|  | 173 |  | 
|---|
|  | 174 | #define POLL_WANT       0x1 | 
|---|
|  | 175 | #define POLL_DONE       0x2 | 
|---|
|  | 176 |  | 
|---|
|  | 177 | /* | 
|---|
|  | 178 | * This is the thread ID of the notifier thread that does select. | 
|---|
|  | 179 | */ | 
|---|
|  | 180 |  | 
|---|
|  | 181 | static Tcl_ThreadId notifierThread; | 
|---|
|  | 182 |  | 
|---|
|  | 183 | #endif | 
|---|
|  | 184 |  | 
|---|
|  | 185 | /* | 
|---|
|  | 186 | * Static routines defined in this file. | 
|---|
|  | 187 | */ | 
|---|
|  | 188 |  | 
|---|
|  | 189 | #ifdef TCL_THREADS | 
|---|
|  | 190 | static void     NotifierThreadProc(ClientData clientData); | 
|---|
|  | 191 | #endif | 
|---|
|  | 192 | static int      FileHandlerEventProc(Tcl_Event *evPtr, int flags); | 
|---|
|  | 193 |  | 
|---|
|  | 194 | /* | 
|---|
|  | 195 | *---------------------------------------------------------------------- | 
|---|
|  | 196 | * | 
|---|
|  | 197 | * Tcl_InitNotifier -- | 
|---|
|  | 198 | * | 
|---|
|  | 199 | *      Initializes the platform specific notifier state. | 
|---|
|  | 200 | * | 
|---|
|  | 201 | * Results: | 
|---|
|  | 202 | *      Returns a handle to the notifier state for this thread.. | 
|---|
|  | 203 | * | 
|---|
|  | 204 | * Side effects: | 
|---|
|  | 205 | *      None. | 
|---|
|  | 206 | * | 
|---|
|  | 207 | *---------------------------------------------------------------------- | 
|---|
|  | 208 | */ | 
|---|
|  | 209 |  | 
|---|
|  | 210 | ClientData | 
|---|
|  | 211 | Tcl_InitNotifier(void) | 
|---|
|  | 212 | { | 
|---|
|  | 213 | ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 214 |  | 
|---|
|  | 215 | #ifdef TCL_THREADS | 
|---|
|  | 216 | tsdPtr->eventReady = 0; | 
|---|
|  | 217 |  | 
|---|
|  | 218 | /* | 
|---|
|  | 219 | * Start the Notifier thread if necessary. | 
|---|
|  | 220 | */ | 
|---|
|  | 221 |  | 
|---|
|  | 222 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 223 | if (notifierCount == 0) { | 
|---|
|  | 224 | if (TclpThreadCreate(¬ifierThread, NotifierThreadProc, NULL, | 
|---|
|  | 225 | TCL_THREAD_STACK_DEFAULT, TCL_THREAD_JOINABLE) != TCL_OK) { | 
|---|
|  | 226 | Tcl_Panic("Tcl_InitNotifier: unable to start notifier thread"); | 
|---|
|  | 227 | } | 
|---|
|  | 228 | } | 
|---|
|  | 229 | notifierCount++; | 
|---|
|  | 230 |  | 
|---|
|  | 231 | /* | 
|---|
|  | 232 | * Wait for the notifier pipe to be created. | 
|---|
|  | 233 | */ | 
|---|
|  | 234 |  | 
|---|
|  | 235 | while (triggerPipe < 0) { | 
|---|
|  | 236 | Tcl_ConditionWait(¬ifierCV, ¬ifierMutex, NULL); | 
|---|
|  | 237 | } | 
|---|
|  | 238 |  | 
|---|
|  | 239 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 240 | #endif | 
|---|
|  | 241 | return (ClientData) tsdPtr; | 
|---|
|  | 242 | } | 
|---|
|  | 243 |  | 
|---|
|  | 244 | /* | 
|---|
|  | 245 | *---------------------------------------------------------------------- | 
|---|
|  | 246 | * | 
|---|
|  | 247 | * Tcl_FinalizeNotifier -- | 
|---|
|  | 248 | * | 
|---|
|  | 249 | *      This function is called to cleanup the notifier state before a thread | 
|---|
|  | 250 | *      is terminated. | 
|---|
|  | 251 | * | 
|---|
|  | 252 | * Results: | 
|---|
|  | 253 | *      None. | 
|---|
|  | 254 | * | 
|---|
|  | 255 | * Side effects: | 
|---|
|  | 256 | *      May terminate the background notifier thread if this is the last | 
|---|
|  | 257 | *      notifier instance. | 
|---|
|  | 258 | * | 
|---|
|  | 259 | *---------------------------------------------------------------------- | 
|---|
|  | 260 | */ | 
|---|
|  | 261 |  | 
|---|
|  | 262 | void | 
|---|
|  | 263 | Tcl_FinalizeNotifier( | 
|---|
|  | 264 | ClientData clientData)              /* Not used. */ | 
|---|
|  | 265 | { | 
|---|
|  | 266 | #ifdef TCL_THREADS | 
|---|
|  | 267 | ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 268 |  | 
|---|
|  | 269 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 270 | notifierCount--; | 
|---|
|  | 271 |  | 
|---|
|  | 272 | /* | 
|---|
|  | 273 | * If this is the last thread to use the notifier, close the notifier pipe | 
|---|
|  | 274 | * and wait for the background thread to terminate. | 
|---|
|  | 275 | */ | 
|---|
|  | 276 |  | 
|---|
|  | 277 | if (notifierCount == 0) { | 
|---|
|  | 278 | int result; | 
|---|
|  | 279 |  | 
|---|
|  | 280 | if (triggerPipe < 0) { | 
|---|
|  | 281 | Tcl_Panic("Tcl_FinalizeNotifier: notifier pipe not initialized"); | 
|---|
|  | 282 | } | 
|---|
|  | 283 |  | 
|---|
|  | 284 | /* | 
|---|
|  | 285 | * Send "q" message to the notifier thread so that it will terminate. | 
|---|
|  | 286 | * The notifier will return from its call to select() and notice that | 
|---|
|  | 287 | * a "q" message has arrived, it will then close its side of the pipe | 
|---|
|  | 288 | * and terminate its thread. Note the we can not just close the pipe | 
|---|
|  | 289 | * and check for EOF in the notifier thread because if a background | 
|---|
|  | 290 | * child process was created with exec, select() would not register | 
|---|
|  | 291 | * the EOF on the pipe until the child processes had terminated. [Bug: | 
|---|
|  | 292 | * 4139] [Bug: 1222872] | 
|---|
|  | 293 | */ | 
|---|
|  | 294 |  | 
|---|
|  | 295 | write(triggerPipe, "q", 1); | 
|---|
|  | 296 | close(triggerPipe); | 
|---|
|  | 297 | while(triggerPipe >= 0) { | 
|---|
|  | 298 | Tcl_ConditionWait(¬ifierCV, ¬ifierMutex, NULL); | 
|---|
|  | 299 | } | 
|---|
|  | 300 |  | 
|---|
|  | 301 | result = Tcl_JoinThread(notifierThread, NULL); | 
|---|
|  | 302 | if (result) { | 
|---|
|  | 303 | Tcl_Panic("Tcl_FinalizeNotifier: unable to join notifier thread"); | 
|---|
|  | 304 | } | 
|---|
|  | 305 | } | 
|---|
|  | 306 |  | 
|---|
|  | 307 | /* | 
|---|
|  | 308 | * Clean up any synchronization objects in the thread local storage. | 
|---|
|  | 309 | */ | 
|---|
|  | 310 |  | 
|---|
|  | 311 | Tcl_ConditionFinalize(&(tsdPtr->waitCV)); | 
|---|
|  | 312 |  | 
|---|
|  | 313 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 314 | #endif | 
|---|
|  | 315 | } | 
|---|
|  | 316 |  | 
|---|
|  | 317 | /* | 
|---|
|  | 318 | *---------------------------------------------------------------------- | 
|---|
|  | 319 | * | 
|---|
|  | 320 | * Tcl_AlertNotifier -- | 
|---|
|  | 321 | * | 
|---|
|  | 322 | *      Wake up the specified notifier from any thread. This routine is called | 
|---|
|  | 323 | *      by the platform independent notifier code whenever the Tcl_ThreadAlert | 
|---|
|  | 324 | *      routine is called. This routine is guaranteed not to be called on a | 
|---|
|  | 325 | *      given notifier after Tcl_FinalizeNotifier is called for that notifier. | 
|---|
|  | 326 | * | 
|---|
|  | 327 | * Results: | 
|---|
|  | 328 | *      None. | 
|---|
|  | 329 | * | 
|---|
|  | 330 | * Side effects: | 
|---|
|  | 331 | *      Signals the notifier condition variable for the specified notifier. | 
|---|
|  | 332 | * | 
|---|
|  | 333 | *---------------------------------------------------------------------- | 
|---|
|  | 334 | */ | 
|---|
|  | 335 |  | 
|---|
|  | 336 | void | 
|---|
|  | 337 | Tcl_AlertNotifier( | 
|---|
|  | 338 | ClientData clientData) | 
|---|
|  | 339 | { | 
|---|
|  | 340 | #ifdef TCL_THREADS | 
|---|
|  | 341 | ThreadSpecificData *tsdPtr = (ThreadSpecificData *) clientData; | 
|---|
|  | 342 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 343 | tsdPtr->eventReady = 1; | 
|---|
|  | 344 | Tcl_ConditionNotify(&tsdPtr->waitCV); | 
|---|
|  | 345 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 346 | #endif | 
|---|
|  | 347 | } | 
|---|
|  | 348 |  | 
|---|
|  | 349 | /* | 
|---|
|  | 350 | *---------------------------------------------------------------------- | 
|---|
|  | 351 | * | 
|---|
|  | 352 | * Tcl_SetTimer -- | 
|---|
|  | 353 | * | 
|---|
|  | 354 | *      This function sets the current notifier timer value. This interface is | 
|---|
|  | 355 | *      not implemented in this notifier because we are always running inside | 
|---|
|  | 356 | *      of Tcl_DoOneEvent. | 
|---|
|  | 357 | * | 
|---|
|  | 358 | * Results: | 
|---|
|  | 359 | *      None. | 
|---|
|  | 360 | * | 
|---|
|  | 361 | * Side effects: | 
|---|
|  | 362 | *      None. | 
|---|
|  | 363 | * | 
|---|
|  | 364 | *---------------------------------------------------------------------- | 
|---|
|  | 365 | */ | 
|---|
|  | 366 |  | 
|---|
|  | 367 | void | 
|---|
|  | 368 | Tcl_SetTimer( | 
|---|
|  | 369 | Tcl_Time *timePtr)          /* Timeout value, may be NULL. */ | 
|---|
|  | 370 | { | 
|---|
|  | 371 | /* | 
|---|
|  | 372 | * The interval timer doesn't do anything in this implementation, because | 
|---|
|  | 373 | * the only event loop is via Tcl_DoOneEvent, which passes timeout values | 
|---|
|  | 374 | * to Tcl_WaitForEvent. | 
|---|
|  | 375 | */ | 
|---|
|  | 376 |  | 
|---|
|  | 377 | if (tclStubs.tcl_SetTimer != tclOriginalNotifier.setTimerProc) { | 
|---|
|  | 378 | tclStubs.tcl_SetTimer(timePtr); | 
|---|
|  | 379 | } | 
|---|
|  | 380 | } | 
|---|
|  | 381 |  | 
|---|
|  | 382 | /* | 
|---|
|  | 383 | *---------------------------------------------------------------------- | 
|---|
|  | 384 | * | 
|---|
|  | 385 | * Tcl_ServiceModeHook -- | 
|---|
|  | 386 | * | 
|---|
|  | 387 | *      This function is invoked whenever the service mode changes. | 
|---|
|  | 388 | * | 
|---|
|  | 389 | * Results: | 
|---|
|  | 390 | *      None. | 
|---|
|  | 391 | * | 
|---|
|  | 392 | * Side effects: | 
|---|
|  | 393 | *      None. | 
|---|
|  | 394 | * | 
|---|
|  | 395 | *---------------------------------------------------------------------- | 
|---|
|  | 396 | */ | 
|---|
|  | 397 |  | 
|---|
|  | 398 | void | 
|---|
|  | 399 | Tcl_ServiceModeHook( | 
|---|
|  | 400 | int mode)                   /* Either TCL_SERVICE_ALL, or | 
|---|
|  | 401 | * TCL_SERVICE_NONE. */ | 
|---|
|  | 402 | { | 
|---|
|  | 403 | } | 
|---|
|  | 404 |  | 
|---|
|  | 405 | /* | 
|---|
|  | 406 | *---------------------------------------------------------------------- | 
|---|
|  | 407 | * | 
|---|
|  | 408 | * Tcl_CreateFileHandler -- | 
|---|
|  | 409 | * | 
|---|
|  | 410 | *      This function registers a file handler with the select notifier. | 
|---|
|  | 411 | * | 
|---|
|  | 412 | * Results: | 
|---|
|  | 413 | *      None. | 
|---|
|  | 414 | * | 
|---|
|  | 415 | * Side effects: | 
|---|
|  | 416 | *      Creates a new file handler structure. | 
|---|
|  | 417 | * | 
|---|
|  | 418 | *---------------------------------------------------------------------- | 
|---|
|  | 419 | */ | 
|---|
|  | 420 |  | 
|---|
|  | 421 | void | 
|---|
|  | 422 | Tcl_CreateFileHandler( | 
|---|
|  | 423 | int fd,                     /* Handle of stream to watch. */ | 
|---|
|  | 424 | int mask,                   /* OR'ed combination of TCL_READABLE, | 
|---|
|  | 425 | * TCL_WRITABLE, and TCL_EXCEPTION: indicates | 
|---|
|  | 426 | * conditions under which proc should be | 
|---|
|  | 427 | * called. */ | 
|---|
|  | 428 | Tcl_FileProc *proc,         /* Function to call for each selected | 
|---|
|  | 429 | * event. */ | 
|---|
|  | 430 | ClientData clientData)      /* Arbitrary data to pass to proc. */ | 
|---|
|  | 431 | { | 
|---|
|  | 432 | ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 433 | FileHandler *filePtr; | 
|---|
|  | 434 |  | 
|---|
|  | 435 | if (tclStubs.tcl_CreateFileHandler != | 
|---|
|  | 436 | tclOriginalNotifier.createFileHandlerProc) { | 
|---|
|  | 437 | tclStubs.tcl_CreateFileHandler(fd, mask, proc, clientData); | 
|---|
|  | 438 | return; | 
|---|
|  | 439 | } | 
|---|
|  | 440 |  | 
|---|
|  | 441 | for (filePtr = tsdPtr->firstFileHandlerPtr; filePtr != NULL; | 
|---|
|  | 442 | filePtr = filePtr->nextPtr) { | 
|---|
|  | 443 | if (filePtr->fd == fd) { | 
|---|
|  | 444 | break; | 
|---|
|  | 445 | } | 
|---|
|  | 446 | } | 
|---|
|  | 447 | if (filePtr == NULL) { | 
|---|
|  | 448 | filePtr = (FileHandler*) ckalloc(sizeof(FileHandler)); | 
|---|
|  | 449 | filePtr->fd = fd; | 
|---|
|  | 450 | filePtr->readyMask = 0; | 
|---|
|  | 451 | filePtr->nextPtr = tsdPtr->firstFileHandlerPtr; | 
|---|
|  | 452 | tsdPtr->firstFileHandlerPtr = filePtr; | 
|---|
|  | 453 | } | 
|---|
|  | 454 | filePtr->proc = proc; | 
|---|
|  | 455 | filePtr->clientData = clientData; | 
|---|
|  | 456 | filePtr->mask = mask; | 
|---|
|  | 457 |  | 
|---|
|  | 458 | /* | 
|---|
|  | 459 | * Update the check masks for this file. | 
|---|
|  | 460 | */ | 
|---|
|  | 461 |  | 
|---|
|  | 462 | if (mask & TCL_READABLE) { | 
|---|
|  | 463 | FD_SET(fd, &(tsdPtr->checkMasks.readable)); | 
|---|
|  | 464 | } else { | 
|---|
|  | 465 | FD_CLR(fd, &(tsdPtr->checkMasks.readable)); | 
|---|
|  | 466 | } | 
|---|
|  | 467 | if (mask & TCL_WRITABLE) { | 
|---|
|  | 468 | FD_SET(fd, &(tsdPtr->checkMasks.writable)); | 
|---|
|  | 469 | } else { | 
|---|
|  | 470 | FD_CLR(fd, &(tsdPtr->checkMasks.writable)); | 
|---|
|  | 471 | } | 
|---|
|  | 472 | if (mask & TCL_EXCEPTION) { | 
|---|
|  | 473 | FD_SET(fd, &(tsdPtr->checkMasks.exceptional)); | 
|---|
|  | 474 | } else { | 
|---|
|  | 475 | FD_CLR(fd, &(tsdPtr->checkMasks.exceptional)); | 
|---|
|  | 476 | } | 
|---|
|  | 477 | if (tsdPtr->numFdBits <= fd) { | 
|---|
|  | 478 | tsdPtr->numFdBits = fd+1; | 
|---|
|  | 479 | } | 
|---|
|  | 480 | } | 
|---|
|  | 481 |  | 
|---|
|  | 482 | /* | 
|---|
|  | 483 | *---------------------------------------------------------------------- | 
|---|
|  | 484 | * | 
|---|
|  | 485 | * Tcl_DeleteFileHandler -- | 
|---|
|  | 486 | * | 
|---|
|  | 487 | *      Cancel a previously-arranged callback arrangement for a file. | 
|---|
|  | 488 | * | 
|---|
|  | 489 | * Results: | 
|---|
|  | 490 | *      None. | 
|---|
|  | 491 | * | 
|---|
|  | 492 | * Side effects: | 
|---|
|  | 493 | *      If a callback was previously registered on file, remove it. | 
|---|
|  | 494 | * | 
|---|
|  | 495 | *---------------------------------------------------------------------- | 
|---|
|  | 496 | */ | 
|---|
|  | 497 |  | 
|---|
|  | 498 | void | 
|---|
|  | 499 | Tcl_DeleteFileHandler( | 
|---|
|  | 500 | int fd)                     /* Stream id for which to remove callback | 
|---|
|  | 501 | * function. */ | 
|---|
|  | 502 | { | 
|---|
|  | 503 | FileHandler *filePtr, *prevPtr; | 
|---|
|  | 504 | int i; | 
|---|
|  | 505 | ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 506 |  | 
|---|
|  | 507 | if (tclStubs.tcl_DeleteFileHandler != | 
|---|
|  | 508 | tclOriginalNotifier.deleteFileHandlerProc) { | 
|---|
|  | 509 | tclStubs.tcl_DeleteFileHandler(fd); | 
|---|
|  | 510 | return; | 
|---|
|  | 511 | } | 
|---|
|  | 512 |  | 
|---|
|  | 513 | /* | 
|---|
|  | 514 | * Find the entry for the given file (and return if there isn't one). | 
|---|
|  | 515 | */ | 
|---|
|  | 516 |  | 
|---|
|  | 517 | for (prevPtr = NULL, filePtr = tsdPtr->firstFileHandlerPtr; ; | 
|---|
|  | 518 | prevPtr = filePtr, filePtr = filePtr->nextPtr) { | 
|---|
|  | 519 | if (filePtr == NULL) { | 
|---|
|  | 520 | return; | 
|---|
|  | 521 | } | 
|---|
|  | 522 | if (filePtr->fd == fd) { | 
|---|
|  | 523 | break; | 
|---|
|  | 524 | } | 
|---|
|  | 525 | } | 
|---|
|  | 526 |  | 
|---|
|  | 527 | /* | 
|---|
|  | 528 | * Update the check masks for this file. | 
|---|
|  | 529 | */ | 
|---|
|  | 530 |  | 
|---|
|  | 531 | if (filePtr->mask & TCL_READABLE) { | 
|---|
|  | 532 | FD_CLR(fd, &(tsdPtr->checkMasks.readable)); | 
|---|
|  | 533 | } | 
|---|
|  | 534 | if (filePtr->mask & TCL_WRITABLE) { | 
|---|
|  | 535 | FD_CLR(fd, &(tsdPtr->checkMasks.writable)); | 
|---|
|  | 536 | } | 
|---|
|  | 537 | if (filePtr->mask & TCL_EXCEPTION) { | 
|---|
|  | 538 | FD_CLR(fd, &(tsdPtr->checkMasks.exceptional)); | 
|---|
|  | 539 | } | 
|---|
|  | 540 |  | 
|---|
|  | 541 | /* | 
|---|
|  | 542 | * Find current max fd. | 
|---|
|  | 543 | */ | 
|---|
|  | 544 |  | 
|---|
|  | 545 | if (fd+1 == tsdPtr->numFdBits) { | 
|---|
|  | 546 | tsdPtr->numFdBits = 0; | 
|---|
|  | 547 | for (i = fd-1; i >= 0; i--) { | 
|---|
|  | 548 | if (FD_ISSET(i, &(tsdPtr->checkMasks.readable)) | 
|---|
|  | 549 | || FD_ISSET(i, &(tsdPtr->checkMasks.writable)) | 
|---|
|  | 550 | || FD_ISSET(i, &(tsdPtr->checkMasks.exceptional))) { | 
|---|
|  | 551 | tsdPtr->numFdBits = i+1; | 
|---|
|  | 552 | break; | 
|---|
|  | 553 | } | 
|---|
|  | 554 | } | 
|---|
|  | 555 | } | 
|---|
|  | 556 |  | 
|---|
|  | 557 | /* | 
|---|
|  | 558 | * Clean up information in the callback record. | 
|---|
|  | 559 | */ | 
|---|
|  | 560 |  | 
|---|
|  | 561 | if (prevPtr == NULL) { | 
|---|
|  | 562 | tsdPtr->firstFileHandlerPtr = filePtr->nextPtr; | 
|---|
|  | 563 | } else { | 
|---|
|  | 564 | prevPtr->nextPtr = filePtr->nextPtr; | 
|---|
|  | 565 | } | 
|---|
|  | 566 | ckfree((char *) filePtr); | 
|---|
|  | 567 | } | 
|---|
|  | 568 |  | 
|---|
|  | 569 | /* | 
|---|
|  | 570 | *---------------------------------------------------------------------- | 
|---|
|  | 571 | * | 
|---|
|  | 572 | * FileHandlerEventProc -- | 
|---|
|  | 573 | * | 
|---|
|  | 574 | *      This function is called by Tcl_ServiceEvent when a file event reaches | 
|---|
|  | 575 | *      the front of the event queue. This function is responsible for | 
|---|
|  | 576 | *      actually handling the event by invoking the callback for the file | 
|---|
|  | 577 | *      handler. | 
|---|
|  | 578 | * | 
|---|
|  | 579 | * Results: | 
|---|
|  | 580 | *      Returns 1 if the event was handled, meaning it should be removed from | 
|---|
|  | 581 | *      the queue. Returns 0 if the event was not handled, meaning it should | 
|---|
|  | 582 | *      stay on the queue. The only time the event isn't handled is if the | 
|---|
|  | 583 | *      TCL_FILE_EVENTS flag bit isn't set. | 
|---|
|  | 584 | * | 
|---|
|  | 585 | * Side effects: | 
|---|
|  | 586 | *      Whatever the file handler's callback function does. | 
|---|
|  | 587 | * | 
|---|
|  | 588 | *---------------------------------------------------------------------- | 
|---|
|  | 589 | */ | 
|---|
|  | 590 |  | 
|---|
|  | 591 | static int | 
|---|
|  | 592 | FileHandlerEventProc( | 
|---|
|  | 593 | Tcl_Event *evPtr,           /* Event to service. */ | 
|---|
|  | 594 | int flags)                  /* Flags that indicate what events to handle, | 
|---|
|  | 595 | * such as TCL_FILE_EVENTS. */ | 
|---|
|  | 596 | { | 
|---|
|  | 597 | int mask; | 
|---|
|  | 598 | FileHandler *filePtr; | 
|---|
|  | 599 | FileHandlerEvent *fileEvPtr = (FileHandlerEvent *) evPtr; | 
|---|
|  | 600 | ThreadSpecificData *tsdPtr; | 
|---|
|  | 601 |  | 
|---|
|  | 602 | if (!(flags & TCL_FILE_EVENTS)) { | 
|---|
|  | 603 | return 0; | 
|---|
|  | 604 | } | 
|---|
|  | 605 |  | 
|---|
|  | 606 | /* | 
|---|
|  | 607 | * Search through the file handlers to find the one whose handle matches | 
|---|
|  | 608 | * the event. We do this rather than keeping a pointer to the file handler | 
|---|
|  | 609 | * directly in the event, so that the handler can be deleted while the | 
|---|
|  | 610 | * event is queued without leaving a dangling pointer. | 
|---|
|  | 611 | */ | 
|---|
|  | 612 |  | 
|---|
|  | 613 | tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 614 | for (filePtr = tsdPtr->firstFileHandlerPtr; filePtr != NULL; | 
|---|
|  | 615 | filePtr = filePtr->nextPtr) { | 
|---|
|  | 616 | if (filePtr->fd != fileEvPtr->fd) { | 
|---|
|  | 617 | continue; | 
|---|
|  | 618 | } | 
|---|
|  | 619 |  | 
|---|
|  | 620 | /* | 
|---|
|  | 621 | * The code is tricky for two reasons: | 
|---|
|  | 622 | * 1. The file handler's desired events could have changed since the | 
|---|
|  | 623 | *    time when the event was queued, so AND the ready mask with the | 
|---|
|  | 624 | *    desired mask. | 
|---|
|  | 625 | * 2. The file could have been closed and re-opened since the time | 
|---|
|  | 626 | *    when the event was queued. This is why the ready mask is stored | 
|---|
|  | 627 | *    in the file handler rather than the queued event: it will be | 
|---|
|  | 628 | *    zeroed when a new file handler is created for the newly opened | 
|---|
|  | 629 | *    file. | 
|---|
|  | 630 | */ | 
|---|
|  | 631 |  | 
|---|
|  | 632 | mask = filePtr->readyMask & filePtr->mask; | 
|---|
|  | 633 | filePtr->readyMask = 0; | 
|---|
|  | 634 | if (mask != 0) { | 
|---|
|  | 635 | (*filePtr->proc)(filePtr->clientData, mask); | 
|---|
|  | 636 | } | 
|---|
|  | 637 | break; | 
|---|
|  | 638 | } | 
|---|
|  | 639 | return 1; | 
|---|
|  | 640 | } | 
|---|
|  | 641 |  | 
|---|
|  | 642 | /* | 
|---|
|  | 643 | *---------------------------------------------------------------------- | 
|---|
|  | 644 | * | 
|---|
|  | 645 | * Tcl_WaitForEvent -- | 
|---|
|  | 646 | * | 
|---|
|  | 647 | *      This function is called by Tcl_DoOneEvent to wait for new events on | 
|---|
|  | 648 | *      the message queue. If the block time is 0, then Tcl_WaitForEvent just | 
|---|
|  | 649 | *      polls without blocking. | 
|---|
|  | 650 | * | 
|---|
|  | 651 | * Results: | 
|---|
|  | 652 | *      Returns -1 if the select would block forever, otherwise returns 0. | 
|---|
|  | 653 | * | 
|---|
|  | 654 | * Side effects: | 
|---|
|  | 655 | *      Queues file events that are detected by the select. | 
|---|
|  | 656 | * | 
|---|
|  | 657 | *---------------------------------------------------------------------- | 
|---|
|  | 658 | */ | 
|---|
|  | 659 |  | 
|---|
|  | 660 | int | 
|---|
|  | 661 | Tcl_WaitForEvent( | 
|---|
|  | 662 | Tcl_Time *timePtr)          /* Maximum block time, or NULL. */ | 
|---|
|  | 663 | { | 
|---|
|  | 664 | FileHandler *filePtr; | 
|---|
|  | 665 | FileHandlerEvent *fileEvPtr; | 
|---|
|  | 666 | int mask; | 
|---|
|  | 667 | Tcl_Time myTime; | 
|---|
|  | 668 | #ifdef TCL_THREADS | 
|---|
|  | 669 | int waitForFiles; | 
|---|
|  | 670 | Tcl_Time *myTimePtr; | 
|---|
|  | 671 | #else | 
|---|
|  | 672 | /* | 
|---|
|  | 673 | * Impl. notes: timeout & timeoutPtr are used if, and only if threads are | 
|---|
|  | 674 | * not enabled. They are the arguments for the regular select() used when | 
|---|
|  | 675 | * the core is not thread-enabled. | 
|---|
|  | 676 | */ | 
|---|
|  | 677 |  | 
|---|
|  | 678 | struct timeval timeout, *timeoutPtr; | 
|---|
|  | 679 | int numFound; | 
|---|
|  | 680 | #endif /* TCL_THREADS */ | 
|---|
|  | 681 | ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); | 
|---|
|  | 682 |  | 
|---|
|  | 683 | if (tclStubs.tcl_WaitForEvent != tclOriginalNotifier.waitForEventProc) { | 
|---|
|  | 684 | return tclStubs.tcl_WaitForEvent(timePtr); | 
|---|
|  | 685 | } | 
|---|
|  | 686 |  | 
|---|
|  | 687 | /* | 
|---|
|  | 688 | * Set up the timeout structure. Note that if there are no events to check | 
|---|
|  | 689 | * for, we return with a negative result rather than blocking forever. | 
|---|
|  | 690 | */ | 
|---|
|  | 691 |  | 
|---|
|  | 692 | if (timePtr != NULL) { | 
|---|
|  | 693 | /* | 
|---|
|  | 694 | * TIP #233 (Virtualized Time). Is virtual time in effect? And do we | 
|---|
|  | 695 | * actually have something to scale? If yes to both then we call the | 
|---|
|  | 696 | * handler to do this scaling. | 
|---|
|  | 697 | */ | 
|---|
|  | 698 |  | 
|---|
|  | 699 | myTime.sec  = timePtr->sec; | 
|---|
|  | 700 | myTime.usec = timePtr->usec; | 
|---|
|  | 701 |  | 
|---|
|  | 702 | if (myTime.sec != 0 || myTime.usec != 0) { | 
|---|
|  | 703 | (*tclScaleTimeProcPtr) (&myTime, tclTimeClientData); | 
|---|
|  | 704 | } | 
|---|
|  | 705 |  | 
|---|
|  | 706 | #ifdef TCL_THREADS | 
|---|
|  | 707 | myTimePtr = &myTime; | 
|---|
|  | 708 | #else | 
|---|
|  | 709 | timeout.tv_sec = myTime.sec; | 
|---|
|  | 710 | timeout.tv_usec = myTime.usec; | 
|---|
|  | 711 | timeoutPtr = &timeout; | 
|---|
|  | 712 | #endif /* TCL_THREADS */ | 
|---|
|  | 713 |  | 
|---|
|  | 714 | #ifndef TCL_THREADS | 
|---|
|  | 715 | } else if (tsdPtr->numFdBits == 0) { | 
|---|
|  | 716 | /* | 
|---|
|  | 717 | * If there are no threads, no timeout, and no fds registered, then | 
|---|
|  | 718 | * there are no events possible and we must avoid deadlock. Note that | 
|---|
|  | 719 | * this is not entirely correct because there might be a signal that | 
|---|
|  | 720 | * could interrupt the select call, but we don't handle that case if | 
|---|
|  | 721 | * we aren't using threads. | 
|---|
|  | 722 | */ | 
|---|
|  | 723 |  | 
|---|
|  | 724 | return -1; | 
|---|
|  | 725 | #endif /* !TCL_THREADS */ | 
|---|
|  | 726 | } else { | 
|---|
|  | 727 | #ifdef TCL_THREADS | 
|---|
|  | 728 | myTimePtr = NULL; | 
|---|
|  | 729 | #else | 
|---|
|  | 730 | timeoutPtr = NULL; | 
|---|
|  | 731 | #endif /* TCL_THREADS */ | 
|---|
|  | 732 | } | 
|---|
|  | 733 |  | 
|---|
|  | 734 | #ifdef TCL_THREADS | 
|---|
|  | 735 | /* | 
|---|
|  | 736 | * Place this thread on the list of interested threads, signal the | 
|---|
|  | 737 | * notifier thread, and wait for a response or a timeout. | 
|---|
|  | 738 | */ | 
|---|
|  | 739 |  | 
|---|
|  | 740 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 741 |  | 
|---|
|  | 742 | waitForFiles = (tsdPtr->numFdBits > 0); | 
|---|
|  | 743 | if (myTimePtr != NULL && myTimePtr->sec == 0 && (myTimePtr->usec == 0 | 
|---|
|  | 744 | #if defined(__APPLE__) && defined(__LP64__) | 
|---|
|  | 745 | /* | 
|---|
|  | 746 | * On 64-bit Darwin, pthread_cond_timedwait() appears to have a bug | 
|---|
|  | 747 | * that causes it to wait forever when passed an absolute time which | 
|---|
|  | 748 | * has already been exceeded by the system time; as a workaround, | 
|---|
|  | 749 | * when given a very brief timeout, just do a poll. [Bug 1457797] | 
|---|
|  | 750 | */ | 
|---|
|  | 751 | || myTimePtr->usec < 10 | 
|---|
|  | 752 | #endif | 
|---|
|  | 753 | )) { | 
|---|
|  | 754 | /* | 
|---|
|  | 755 | * Cannot emulate a polling select with a polling condition variable. | 
|---|
|  | 756 | * Instead, pretend to wait for files and tell the notifier thread | 
|---|
|  | 757 | * what we are doing. The notifier thread makes sure it goes through | 
|---|
|  | 758 | * select with its select mask in the same state as ours currently is. | 
|---|
|  | 759 | * We block until that happens. | 
|---|
|  | 760 | */ | 
|---|
|  | 761 |  | 
|---|
|  | 762 | waitForFiles = 1; | 
|---|
|  | 763 | tsdPtr->pollState = POLL_WANT; | 
|---|
|  | 764 | myTimePtr = NULL; | 
|---|
|  | 765 | } else { | 
|---|
|  | 766 | tsdPtr->pollState = 0; | 
|---|
|  | 767 | } | 
|---|
|  | 768 |  | 
|---|
|  | 769 | if (waitForFiles) { | 
|---|
|  | 770 | /* | 
|---|
|  | 771 | * Add the ThreadSpecificData structure of this thread to the list of | 
|---|
|  | 772 | * ThreadSpecificData structures of all threads that are waiting on | 
|---|
|  | 773 | * file events. | 
|---|
|  | 774 | */ | 
|---|
|  | 775 |  | 
|---|
|  | 776 | tsdPtr->nextPtr = waitingListPtr; | 
|---|
|  | 777 | if (waitingListPtr) { | 
|---|
|  | 778 | waitingListPtr->prevPtr = tsdPtr; | 
|---|
|  | 779 | } | 
|---|
|  | 780 | tsdPtr->prevPtr = 0; | 
|---|
|  | 781 | waitingListPtr = tsdPtr; | 
|---|
|  | 782 | tsdPtr->onList = 1; | 
|---|
|  | 783 |  | 
|---|
|  | 784 | write(triggerPipe, "", 1); | 
|---|
|  | 785 | } | 
|---|
|  | 786 |  | 
|---|
|  | 787 | FD_ZERO(&(tsdPtr->readyMasks.readable)); | 
|---|
|  | 788 | FD_ZERO(&(tsdPtr->readyMasks.writable)); | 
|---|
|  | 789 | FD_ZERO(&(tsdPtr->readyMasks.exceptional)); | 
|---|
|  | 790 |  | 
|---|
|  | 791 | if (!tsdPtr->eventReady) { | 
|---|
|  | 792 | Tcl_ConditionWait(&tsdPtr->waitCV, ¬ifierMutex, myTimePtr); | 
|---|
|  | 793 | } | 
|---|
|  | 794 | tsdPtr->eventReady = 0; | 
|---|
|  | 795 |  | 
|---|
|  | 796 | if (waitForFiles && tsdPtr->onList) { | 
|---|
|  | 797 | /* | 
|---|
|  | 798 | * Remove the ThreadSpecificData structure of this thread from the | 
|---|
|  | 799 | * waiting list. Alert the notifier thread to recompute its select | 
|---|
|  | 800 | * masks - skipping this caused a hang when trying to close a pipe | 
|---|
|  | 801 | * which the notifier thread was still doing a select on. | 
|---|
|  | 802 | */ | 
|---|
|  | 803 |  | 
|---|
|  | 804 | if (tsdPtr->prevPtr) { | 
|---|
|  | 805 | tsdPtr->prevPtr->nextPtr = tsdPtr->nextPtr; | 
|---|
|  | 806 | } else { | 
|---|
|  | 807 | waitingListPtr = tsdPtr->nextPtr; | 
|---|
|  | 808 | } | 
|---|
|  | 809 | if (tsdPtr->nextPtr) { | 
|---|
|  | 810 | tsdPtr->nextPtr->prevPtr = tsdPtr->prevPtr; | 
|---|
|  | 811 | } | 
|---|
|  | 812 | tsdPtr->nextPtr = tsdPtr->prevPtr = NULL; | 
|---|
|  | 813 | tsdPtr->onList = 0; | 
|---|
|  | 814 | write(triggerPipe, "", 1); | 
|---|
|  | 815 | } | 
|---|
|  | 816 |  | 
|---|
|  | 817 | #else | 
|---|
|  | 818 | tsdPtr->readyMasks = tsdPtr->checkMasks; | 
|---|
|  | 819 | numFound = select(tsdPtr->numFdBits, &(tsdPtr->readyMasks.readable), | 
|---|
|  | 820 | &(tsdPtr->readyMasks.writable), &(tsdPtr->readyMasks.exceptional), | 
|---|
|  | 821 | timeoutPtr); | 
|---|
|  | 822 |  | 
|---|
|  | 823 | /* | 
|---|
|  | 824 | * Some systems don't clear the masks after an error, so we have to do it | 
|---|
|  | 825 | * here. | 
|---|
|  | 826 | */ | 
|---|
|  | 827 |  | 
|---|
|  | 828 | if (numFound == -1) { | 
|---|
|  | 829 | FD_ZERO(&(tsdPtr->readyMasks.readable)); | 
|---|
|  | 830 | FD_ZERO(&(tsdPtr->readyMasks.writable)); | 
|---|
|  | 831 | FD_ZERO(&(tsdPtr->readyMasks.exceptional)); | 
|---|
|  | 832 | } | 
|---|
|  | 833 | #endif /* TCL_THREADS */ | 
|---|
|  | 834 |  | 
|---|
|  | 835 | /* | 
|---|
|  | 836 | * Queue all detected file events before returning. | 
|---|
|  | 837 | */ | 
|---|
|  | 838 |  | 
|---|
|  | 839 | for (filePtr = tsdPtr->firstFileHandlerPtr; (filePtr != NULL); | 
|---|
|  | 840 | filePtr = filePtr->nextPtr) { | 
|---|
|  | 841 |  | 
|---|
|  | 842 | mask = 0; | 
|---|
|  | 843 | if (FD_ISSET(filePtr->fd, &(tsdPtr->readyMasks.readable))) { | 
|---|
|  | 844 | mask |= TCL_READABLE; | 
|---|
|  | 845 | } | 
|---|
|  | 846 | if (FD_ISSET(filePtr->fd, &(tsdPtr->readyMasks.writable))) { | 
|---|
|  | 847 | mask |= TCL_WRITABLE; | 
|---|
|  | 848 | } | 
|---|
|  | 849 | if (FD_ISSET(filePtr->fd, &(tsdPtr->readyMasks.exceptional))) { | 
|---|
|  | 850 | mask |= TCL_EXCEPTION; | 
|---|
|  | 851 | } | 
|---|
|  | 852 |  | 
|---|
|  | 853 | if (!mask) { | 
|---|
|  | 854 | continue; | 
|---|
|  | 855 | } | 
|---|
|  | 856 |  | 
|---|
|  | 857 | /* | 
|---|
|  | 858 | * Don't bother to queue an event if the mask was previously non-zero | 
|---|
|  | 859 | * since an event must still be on the queue. | 
|---|
|  | 860 | */ | 
|---|
|  | 861 |  | 
|---|
|  | 862 | if (filePtr->readyMask == 0) { | 
|---|
|  | 863 | fileEvPtr = (FileHandlerEvent *) ckalloc(sizeof(FileHandlerEvent)); | 
|---|
|  | 864 | fileEvPtr->header.proc = FileHandlerEventProc; | 
|---|
|  | 865 | fileEvPtr->fd = filePtr->fd; | 
|---|
|  | 866 | Tcl_QueueEvent((Tcl_Event *) fileEvPtr, TCL_QUEUE_TAIL); | 
|---|
|  | 867 | } | 
|---|
|  | 868 | filePtr->readyMask = mask; | 
|---|
|  | 869 | } | 
|---|
|  | 870 | #ifdef TCL_THREADS | 
|---|
|  | 871 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 872 | #endif /* TCL_THREADS */ | 
|---|
|  | 873 | return 0; | 
|---|
|  | 874 | } | 
|---|
|  | 875 |  | 
|---|
|  | 876 | #ifdef TCL_THREADS | 
|---|
|  | 877 | /* | 
|---|
|  | 878 | *---------------------------------------------------------------------- | 
|---|
|  | 879 | * | 
|---|
|  | 880 | * NotifierThreadProc -- | 
|---|
|  | 881 | * | 
|---|
|  | 882 | *      This routine is the initial (and only) function executed by the | 
|---|
|  | 883 | *      special notifier thread. Its job is to wait for file descriptors to | 
|---|
|  | 884 | *      become readable or writable or to have an exception condition and then | 
|---|
|  | 885 | *      to notify other threads who are interested in this information by | 
|---|
|  | 886 | *      signalling a condition variable. Other threads can signal this | 
|---|
|  | 887 | *      notifier thread of a change in their interests by writing a single | 
|---|
|  | 888 | *      byte to a special pipe that the notifier thread is monitoring. | 
|---|
|  | 889 | * | 
|---|
|  | 890 | * Result: | 
|---|
|  | 891 | *      None. Once started, this routine never exits. It dies with the overall | 
|---|
|  | 892 | *      process. | 
|---|
|  | 893 | * | 
|---|
|  | 894 | * Side effects: | 
|---|
|  | 895 | *      The trigger pipe used to signal the notifier thread is created when | 
|---|
|  | 896 | *      the notifier thread first starts. | 
|---|
|  | 897 | * | 
|---|
|  | 898 | *---------------------------------------------------------------------- | 
|---|
|  | 899 | */ | 
|---|
|  | 900 |  | 
|---|
|  | 901 | static void | 
|---|
|  | 902 | NotifierThreadProc( | 
|---|
|  | 903 | ClientData clientData)      /* Not used. */ | 
|---|
|  | 904 | { | 
|---|
|  | 905 | ThreadSpecificData *tsdPtr; | 
|---|
|  | 906 | fd_set readableMask; | 
|---|
|  | 907 | fd_set writableMask; | 
|---|
|  | 908 | fd_set exceptionalMask; | 
|---|
|  | 909 | int fds[2]; | 
|---|
|  | 910 | int i, numFdBits = 0, receivePipe; | 
|---|
|  | 911 | long found; | 
|---|
|  | 912 | struct timeval poll = {0., 0.}, *timePtr; | 
|---|
|  | 913 | char buf[2]; | 
|---|
|  | 914 |  | 
|---|
|  | 915 | if (pipe(fds) != 0) { | 
|---|
|  | 916 | Tcl_Panic("NotifierThreadProc: could not create trigger pipe"); | 
|---|
|  | 917 | } | 
|---|
|  | 918 |  | 
|---|
|  | 919 | receivePipe = fds[0]; | 
|---|
|  | 920 |  | 
|---|
|  | 921 | if (TclUnixSetBlockingMode(receivePipe, TCL_MODE_NONBLOCKING) < 0) { | 
|---|
|  | 922 | Tcl_Panic("NotifierThreadProc: could not make receive pipe non blocking"); | 
|---|
|  | 923 | } | 
|---|
|  | 924 | if (TclUnixSetBlockingMode(fds[1], TCL_MODE_NONBLOCKING) < 0) { | 
|---|
|  | 925 | Tcl_Panic("NotifierThreadProc: could not make trigger pipe non blocking"); | 
|---|
|  | 926 | } | 
|---|
|  | 927 |  | 
|---|
|  | 928 | /* | 
|---|
|  | 929 | * Install the write end of the pipe into the global variable. | 
|---|
|  | 930 | */ | 
|---|
|  | 931 |  | 
|---|
|  | 932 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 933 | triggerPipe = fds[1]; | 
|---|
|  | 934 |  | 
|---|
|  | 935 | /* | 
|---|
|  | 936 | * Signal any threads that are waiting. | 
|---|
|  | 937 | */ | 
|---|
|  | 938 |  | 
|---|
|  | 939 | Tcl_ConditionNotify(¬ifierCV); | 
|---|
|  | 940 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 941 |  | 
|---|
|  | 942 | /* | 
|---|
|  | 943 | * Look for file events and report them to interested threads. | 
|---|
|  | 944 | */ | 
|---|
|  | 945 |  | 
|---|
|  | 946 | while (1) { | 
|---|
|  | 947 | FD_ZERO(&readableMask); | 
|---|
|  | 948 | FD_ZERO(&writableMask); | 
|---|
|  | 949 | FD_ZERO(&exceptionalMask); | 
|---|
|  | 950 |  | 
|---|
|  | 951 | /* | 
|---|
|  | 952 | * Compute the logical OR of the select masks from all the waiting | 
|---|
|  | 953 | * notifiers. | 
|---|
|  | 954 | */ | 
|---|
|  | 955 |  | 
|---|
|  | 956 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 957 | timePtr = NULL; | 
|---|
|  | 958 | for (tsdPtr = waitingListPtr; tsdPtr; tsdPtr = tsdPtr->nextPtr) { | 
|---|
|  | 959 | for (i = tsdPtr->numFdBits-1; i >= 0; --i) { | 
|---|
|  | 960 | if (FD_ISSET(i, &(tsdPtr->checkMasks.readable))) { | 
|---|
|  | 961 | FD_SET(i, &readableMask); | 
|---|
|  | 962 | } | 
|---|
|  | 963 | if (FD_ISSET(i, &(tsdPtr->checkMasks.writable))) { | 
|---|
|  | 964 | FD_SET(i, &writableMask); | 
|---|
|  | 965 | } | 
|---|
|  | 966 | if (FD_ISSET(i, &(tsdPtr->checkMasks.exceptional))) { | 
|---|
|  | 967 | FD_SET(i, &exceptionalMask); | 
|---|
|  | 968 | } | 
|---|
|  | 969 | } | 
|---|
|  | 970 | if (tsdPtr->numFdBits > numFdBits) { | 
|---|
|  | 971 | numFdBits = tsdPtr->numFdBits; | 
|---|
|  | 972 | } | 
|---|
|  | 973 | if (tsdPtr->pollState & POLL_WANT) { | 
|---|
|  | 974 | /* | 
|---|
|  | 975 | * Here we make sure we go through select() with the same mask | 
|---|
|  | 976 | * bits that were present when the thread tried to poll. | 
|---|
|  | 977 | */ | 
|---|
|  | 978 |  | 
|---|
|  | 979 | tsdPtr->pollState |= POLL_DONE; | 
|---|
|  | 980 | timePtr = &poll; | 
|---|
|  | 981 | } | 
|---|
|  | 982 | } | 
|---|
|  | 983 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 984 |  | 
|---|
|  | 985 | /* | 
|---|
|  | 986 | * Set up the select mask to include the receive pipe. | 
|---|
|  | 987 | */ | 
|---|
|  | 988 |  | 
|---|
|  | 989 | if (receivePipe >= numFdBits) { | 
|---|
|  | 990 | numFdBits = receivePipe + 1; | 
|---|
|  | 991 | } | 
|---|
|  | 992 | FD_SET(receivePipe, &readableMask); | 
|---|
|  | 993 |  | 
|---|
|  | 994 | if (select(numFdBits, &readableMask, &writableMask, &exceptionalMask, | 
|---|
|  | 995 | timePtr) == -1) { | 
|---|
|  | 996 | /* | 
|---|
|  | 997 | * Try again immediately on an error. | 
|---|
|  | 998 | */ | 
|---|
|  | 999 |  | 
|---|
|  | 1000 | continue; | 
|---|
|  | 1001 | } | 
|---|
|  | 1002 |  | 
|---|
|  | 1003 | /* | 
|---|
|  | 1004 | * Alert any threads that are waiting on a ready file descriptor. | 
|---|
|  | 1005 | */ | 
|---|
|  | 1006 |  | 
|---|
|  | 1007 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 1008 | for (tsdPtr = waitingListPtr; tsdPtr; tsdPtr = tsdPtr->nextPtr) { | 
|---|
|  | 1009 | found = 0; | 
|---|
|  | 1010 |  | 
|---|
|  | 1011 | for (i = tsdPtr->numFdBits-1; i >= 0; --i) { | 
|---|
|  | 1012 | if (FD_ISSET(i, &(tsdPtr->checkMasks.readable)) | 
|---|
|  | 1013 | && FD_ISSET(i, &readableMask)) { | 
|---|
|  | 1014 | FD_SET(i, &(tsdPtr->readyMasks.readable)); | 
|---|
|  | 1015 | found = 1; | 
|---|
|  | 1016 | } | 
|---|
|  | 1017 | if (FD_ISSET(i, &(tsdPtr->checkMasks.writable)) | 
|---|
|  | 1018 | && FD_ISSET(i, &writableMask)) { | 
|---|
|  | 1019 | FD_SET(i, &(tsdPtr->readyMasks.writable)); | 
|---|
|  | 1020 | found = 1; | 
|---|
|  | 1021 | } | 
|---|
|  | 1022 | if (FD_ISSET(i, &(tsdPtr->checkMasks.exceptional)) | 
|---|
|  | 1023 | && FD_ISSET(i, &exceptionalMask)) { | 
|---|
|  | 1024 | FD_SET(i, &(tsdPtr->readyMasks.exceptional)); | 
|---|
|  | 1025 | found = 1; | 
|---|
|  | 1026 | } | 
|---|
|  | 1027 | } | 
|---|
|  | 1028 |  | 
|---|
|  | 1029 | if (found || (tsdPtr->pollState & POLL_DONE)) { | 
|---|
|  | 1030 | tsdPtr->eventReady = 1; | 
|---|
|  | 1031 | if (tsdPtr->onList) { | 
|---|
|  | 1032 | /* | 
|---|
|  | 1033 | * Remove the ThreadSpecificData structure of this thread | 
|---|
|  | 1034 | * from the waiting list. This prevents us from | 
|---|
|  | 1035 | * continuously spining on select until the other threads | 
|---|
|  | 1036 | * runs and services the file event. | 
|---|
|  | 1037 | */ | 
|---|
|  | 1038 |  | 
|---|
|  | 1039 | if (tsdPtr->prevPtr) { | 
|---|
|  | 1040 | tsdPtr->prevPtr->nextPtr = tsdPtr->nextPtr; | 
|---|
|  | 1041 | } else { | 
|---|
|  | 1042 | waitingListPtr = tsdPtr->nextPtr; | 
|---|
|  | 1043 | } | 
|---|
|  | 1044 | if (tsdPtr->nextPtr) { | 
|---|
|  | 1045 | tsdPtr->nextPtr->prevPtr = tsdPtr->prevPtr; | 
|---|
|  | 1046 | } | 
|---|
|  | 1047 | tsdPtr->nextPtr = tsdPtr->prevPtr = NULL; | 
|---|
|  | 1048 | tsdPtr->onList = 0; | 
|---|
|  | 1049 | tsdPtr->pollState = 0; | 
|---|
|  | 1050 | } | 
|---|
|  | 1051 | Tcl_ConditionNotify(&tsdPtr->waitCV); | 
|---|
|  | 1052 | } | 
|---|
|  | 1053 | } | 
|---|
|  | 1054 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 1055 |  | 
|---|
|  | 1056 | /* | 
|---|
|  | 1057 | * Consume the next byte from the notifier pipe if the pipe was | 
|---|
|  | 1058 | * readable. Note that there may be multiple bytes pending, but to | 
|---|
|  | 1059 | * avoid a race condition we only read one at a time. | 
|---|
|  | 1060 | */ | 
|---|
|  | 1061 |  | 
|---|
|  | 1062 | if (FD_ISSET(receivePipe, &readableMask)) { | 
|---|
|  | 1063 | i = read(receivePipe, buf, 1); | 
|---|
|  | 1064 |  | 
|---|
|  | 1065 | if ((i == 0) || ((i == 1) && (buf[0] == 'q'))) { | 
|---|
|  | 1066 | /* | 
|---|
|  | 1067 | * Someone closed the write end of the pipe or sent us a Quit | 
|---|
|  | 1068 | * message [Bug: 4139] and then closed the write end of the | 
|---|
|  | 1069 | * pipe so we need to shut down the notifier thread. | 
|---|
|  | 1070 | */ | 
|---|
|  | 1071 |  | 
|---|
|  | 1072 | break; | 
|---|
|  | 1073 | } | 
|---|
|  | 1074 | } | 
|---|
|  | 1075 | } | 
|---|
|  | 1076 |  | 
|---|
|  | 1077 | /* | 
|---|
|  | 1078 | * Clean up the read end of the pipe and signal any threads waiting on | 
|---|
|  | 1079 | * termination of the notifier thread. | 
|---|
|  | 1080 | */ | 
|---|
|  | 1081 |  | 
|---|
|  | 1082 | close(receivePipe); | 
|---|
|  | 1083 | Tcl_MutexLock(¬ifierMutex); | 
|---|
|  | 1084 | triggerPipe = -1; | 
|---|
|  | 1085 | Tcl_ConditionNotify(¬ifierCV); | 
|---|
|  | 1086 | Tcl_MutexUnlock(¬ifierMutex); | 
|---|
|  | 1087 |  | 
|---|
|  | 1088 | TclpThreadExit (0); | 
|---|
|  | 1089 | } | 
|---|
|  | 1090 | #endif /* TCL_THREADS */ | 
|---|
|  | 1091 |  | 
|---|
|  | 1092 | #endif /* HAVE_COREFOUNDATION */ | 
|---|
|  | 1093 |  | 
|---|
|  | 1094 | /* | 
|---|
|  | 1095 | * Local Variables: | 
|---|
|  | 1096 | * mode: c | 
|---|
|  | 1097 | * c-basic-offset: 4 | 
|---|
|  | 1098 | * fill-column: 78 | 
|---|
|  | 1099 | * End: | 
|---|
|  | 1100 | */ | 
|---|