diff options
Diffstat (limited to 'generic/tclIORChan.c')
| -rw-r--r-- | generic/tclIORChan.c | 860 | 
1 files changed, 487 insertions, 373 deletions
| diff --git a/generic/tclIORChan.c b/generic/tclIORChan.c index 50374a5..94428bb 100644 --- a/generic/tclIORChan.c +++ b/generic/tclIORChan.c @@ -14,12 +14,10 @@   *   * See the file "license.terms" for information on usage and redistribution of   * this file, and for a DISCLAIMER OF ALL WARRANTIES. - * - * RCS: @(#) $Id: tclIORChan.c,v 1.46 2010/03/09 21:15:19 andreas_kupries Exp $   */ -#include <tclInt.h> -#include <tclIO.h> +#include "tclInt.h" +#include "tclIO.h"  #include <assert.h>  #ifndef EINVAL @@ -41,6 +39,9 @@ static int		ReflectOutput(ClientData clientData, const char *buf,  			    int toWrite, int *errorCodePtr);  static void		ReflectWatch(ClientData clientData, int mask);  static int		ReflectBlock(ClientData clientData, int mode); +#ifdef TCL_THREADS +static void		ReflectThread(ClientData clientData, int action); +#endif  static Tcl_WideInt	ReflectSeekWide(ClientData clientData,  			    Tcl_WideInt offset, int mode, int *errorCodePtr);  static int		ReflectSeek(ClientData clientData, long offset, @@ -73,7 +74,11 @@ static const Tcl_ChannelType tclRChannelType = {      NULL,		   /* Flush channel. Not used by core.	NULL'able */      NULL,		   /* Handle events.			NULL'able */      ReflectSeekWide,	   /* Move access point (64 bit).	NULL'able */ +#ifdef TCL_THREADS +    ReflectThread,         /* thread action, tracking owner */ +#else      NULL,		   /* thread action */ +#endif      NULL		   /* truncate */  }; @@ -91,38 +96,20 @@ typedef struct {  				 * command is gone.  				 */  #ifdef TCL_THREADS -    Tcl_ThreadId thread;	/* Thread the 'interp' belongs to. */ +    Tcl_ThreadId thread;	/* Thread the 'interp' belongs to. == Handler thread */ +    Tcl_ThreadId owner;         /* Thread owning the structure.    == Channel thread */  #endif - -    /* See [==] as well. -     * Storage for the command prefix and the additional words required for -     * the invocation of methods in the command handler. -     * -     * argv [0] ... [.] | [argc-2] [argc-1] | [argc]  [argc+2] -     *      cmd ... pfx | method   chan     | detail1 detail2 -     *      ~~~~ CT ~~~            ~~ CT ~~ -     * -     * CT = Belongs to the 'Command handler Thread'. -     */ - -    int argc;			/* Number of preallocated words - 2 */ -    Tcl_Obj **argv;		/* Preallocated array for calling the handler. -				 * args[0] is placeholder for cmd word. -				 * Followed by the arguments in the prefix, -				 * plus 4 placeholders for method, channel, -				 * and at most two varying (method specific) -				 * words. */ -    int methods;		/* Bitmask of supported methods */ - -    /* -     * NOTE (9): Should we have predefined shared literals for the method -     * names? -     */ +    Tcl_Obj *cmd;		/* Callback command prefix */ +    Tcl_Obj *methods;		/* Methods to append to command prefix */ +    Tcl_Obj *name;		/* Name of the channel as created */      int mode;			/* Mask of R/W mode */      int interest;		/* Mask of events the channel is interested  				 * in. */ +    int dead;			/* Boolean signal that some operations +				 * should no longer be attempted. */ +      /*       * Note regarding the usage of timers.       * @@ -389,31 +376,31 @@ TCL_DECLARE_MUTEX(rcForwardMutex)   * leak resources when threads go away.   */ -static void		ForwardOpToOwnerThread(ReflectedChannel *rcPtr, +static void		ForwardOpToHandlerThread(ReflectedChannel *rcPtr,  			    ForwardedOperation op, const void *param);  static int		ForwardProc(Tcl_Event *evPtr, int mask);  static void		SrcExitProc(ClientData clientData);  #define FreeReceivedError(p) \ -	if ((p)->base.mustFree) { \ -	    ckfree((p)->base.msgStr); \ +	if ((p)->base.mustFree) {                               \ +	    ckfree((p)->base.msgStr);                           \  	}  #define PassReceivedErrorInterp(i,p) \ -	if ((i) != NULL) { \ -	    Tcl_SetChannelErrorInterp((i), \ -		    Tcl_NewStringObj((p)->base.msgStr, -1)); \ -	} \ +	if ((i) != NULL) {                                      \ +	    Tcl_SetChannelErrorInterp((i),                      \ +		    Tcl_NewStringObj((p)->base.msgStr, -1));    \ +	}                                                       \  	FreeReceivedError(p)  #define PassReceivedError(c,p) \  	Tcl_SetChannelError((c), Tcl_NewStringObj((p)->base.msgStr, -1)); \  	FreeReceivedError(p)  #define ForwardSetStaticError(p,emsg) \ -	(p)->base.code = TCL_ERROR; \ -	(p)->base.mustFree = 0; \ +	(p)->base.code = TCL_ERROR;                             \ +	(p)->base.mustFree = 0;                                 \  	(p)->base.msgStr = (char *) (emsg)  #define ForwardSetDynamicError(p,emsg) \ -	(p)->base.code = TCL_ERROR; \ -	(p)->base.mustFree = 1; \ +	(p)->base.code = TCL_ERROR;                             \ +	(p)->base.mustFree = 1;                                 \  	(p)->base.msgStr = (char *) (emsg)  static void		ForwardSetObjError(ForwardParam *p, Tcl_Obj *objPtr); @@ -442,7 +429,7 @@ static ReflectedChannel * NewReflectedChannel(Tcl_Interp *interp,  static Tcl_Obj *	NextHandle(void);  static void		FreeReflectedChannel(ReflectedChannel *rcPtr);  static int		InvokeTclMethod(ReflectedChannel *rcPtr, -			    const char *method, Tcl_Obj *argOneObj, +			    MethodName method, Tcl_Obj *argOneObj,  			    Tcl_Obj *argTwoObj, Tcl_Obj **resultObjPtr);  static ReflectedChannelMap *	GetReflectedChannelMap(Tcl_Interp *interp); @@ -457,9 +444,7 @@ static int		ErrnoReturn(ReflectedChannel *rcPtr, Tcl_Obj *resObj);   * list-quoting to keep the words of the message together. See also [x].   */ -static const char *msg_read_unsup = "{read not supported by Tcl driver}";  static const char *msg_read_toomuch = "{read delivered more than requested}"; -static const char *msg_write_unsup = "{write not supported by Tcl driver}";  static const char *msg_write_toomuch = "{write wrote more than requested}";  static const char *msg_write_nothing = "{write wrote nothing}";  static const char *msg_seek_beforestart = "{Tried to seek before origin}"; @@ -573,10 +558,6 @@ TclChanCreateObjCmd(      rcId = NextHandle();      rcPtr = NewReflectedChannel(interp, cmdObj, mode, rcId); -    chan = Tcl_CreateChannel(&tclRChannelType, TclGetString(rcId), rcPtr, -	    mode); -    rcPtr->chan = chan; -    chanPtr = (Channel *) chan;      /*       * Invoke 'initialize' and validate that the handler is present and ok. @@ -589,8 +570,10 @@ TclChanCreateObjCmd(       */      modeObj = DecodeEventMask(mode); -    result = InvokeTclMethod(rcPtr, "initialize", modeObj, NULL, &resObj); +    /* assert modeObj.refCount == 1 */ +    result = InvokeTclMethod(rcPtr, METH_INIT, modeObj, NULL, &resObj);      Tcl_DecrRefCount(modeObj); +      if (result != TCL_OK) {  	UnmarshallErrorResult(interp, resObj);  	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ @@ -605,11 +588,9 @@ TclChanCreateObjCmd(       */      if (Tcl_ListObjGetElements(NULL, resObj, &listc, &listv) != TCL_OK) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, " initialize\" returned non-list: ", -1); -	Tcl_AppendObjToObj(err, resObj); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s initialize\" returned non-list: %s", +                Tcl_GetString(cmdObj), Tcl_GetString(resObj)));  	Tcl_DecrRefCount(resObj);  	goto error;      } @@ -633,42 +614,37 @@ TclChanCreateObjCmd(      Tcl_DecrRefCount(resObj);      if ((REQUIRED_METHODS & methods) != REQUIRED_METHODS) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, "\" does not support all required methods", -1); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" does not support all required methods", +                Tcl_GetString(cmdObj)));  	goto error;      }      if ((mode & TCL_READABLE) && !HAS(methods, METH_READ)) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, "\" lacks a \"read\" method", -1); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" lacks a \"read\" method", +                Tcl_GetString(cmdObj)));  	goto error;      }      if ((mode & TCL_WRITABLE) && !HAS(methods, METH_WRITE)) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, "\" lacks a \"write\" method", -1); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" lacks a \"write\" method", +                Tcl_GetString(cmdObj)));  	goto error;      }      if (!IMPLIES(HAS(methods, METH_CGET), HAS(methods, METH_CGETALL))) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, "\" supports \"cget\" but not \"cgetall\"", -1); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" supports \"cget\" but not \"cgetall\"", +                Tcl_GetString(cmdObj)));  	goto error;      }      if (!IMPLIES(HAS(methods, METH_CGETALL), HAS(methods, METH_CGET))) { -	TclNewLiteralStringObj(err, "chan handler \""); -	Tcl_AppendObjToObj(err, cmdObj); -	Tcl_AppendToObj(err, "\" supports \"cgetall\" but not \"cget\"", -1); -	Tcl_SetObjResult(interp, err); +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" supports \"cgetall\" but not \"cget\"", +                Tcl_GetString(cmdObj)));  	goto error;      } @@ -678,7 +654,11 @@ TclChanCreateObjCmd(       * Everything is fine now.       */ -    rcPtr->methods = methods; +    chan = Tcl_CreateChannel(&tclRChannelType, TclGetString(rcId), rcPtr, +	    mode); +    rcPtr->chan = chan; +    Tcl_Preserve(chan); +    chanPtr = (Channel *) chan;      if ((methods & NULLABLE_METHODS) != NULLABLE_METHODS) {  	/* @@ -687,8 +667,7 @@ TclChanCreateObjCmd(  	 * as the actual channel type.  	 */ -	Tcl_ChannelType *clonePtr = (Tcl_ChannelType *) -		ckalloc(sizeof(Tcl_ChannelType)); +	Tcl_ChannelType *clonePtr = ckalloc(sizeof(Tcl_ChannelType));  	memcpy(clonePtr, &tclRChannelType, sizeof(Tcl_ChannelType)); @@ -735,16 +714,15 @@ TclChanCreateObjCmd(       * Return handle as result of command.       */ -    Tcl_SetResult(interp, (char *)chanPtr->state->channelName, TCL_VOLATILE); +    Tcl_SetObjResult(interp, +            Tcl_NewStringObj(chanPtr->state->channelName, -1));      return TCL_OK;    error: -    /* -     * Signal to ReflectClose to not call 'finalize'. -     */ - -    rcPtr->methods = 0; -    Tcl_Close(interp, chan); +    Tcl_DecrRefCount(rcPtr->name); +    Tcl_DecrRefCount(rcPtr->methods); +    Tcl_DecrRefCount(rcPtr->cmd); +    ckfree((char*) rcPtr);      return TCL_ERROR;  #undef MODE @@ -769,6 +747,50 @@ TclChanCreateObjCmd(   *----------------------------------------------------------------------   */ +typedef struct ReflectEvent { +    Tcl_Event header; +    ReflectedChannel *rcPtr; +    int events; +} ReflectEvent; + +static int +ReflectEventRun( +    Tcl_Event *ev, +    int flags) +{ +    /* OWNER thread +     * +     * Note: When the channel is closed any pending events of this type are +     * deleted. See ReflectClose() for the Tcl_DeleteEvents() calls +     * accomplishing that. +     */ + +    ReflectEvent *e = (ReflectEvent *) ev; + +    Tcl_NotifyChannel(e->rcPtr->chan, e->events); +    return 1; +} + +static int +ReflectEventDelete( +    Tcl_Event *ev, +    ClientData cd) +{ +    /* OWNER thread +     * +     * Invoked by DeleteThreadReflectedChannelMap() and ReflectClose(). The +     * latter ensures that no pending events of this type are run on an +     * invalid channel. +     */ + +    ReflectEvent *e = (ReflectEvent *) ev; + +    if ((ev->proc != ReflectEventRun) || ((cd != NULL) && (cd != e->rcPtr))) { +        return 0; +    } +    return 1; +} +  int  TclChanPostEventObjCmd(      ClientData clientData, @@ -777,6 +799,8 @@ TclChanPostEventObjCmd(      Tcl_Obj *const *objv)  {      /* +     * Ensure -> HANDLER thread +     *       * Syntax:   chan postevent CHANNEL EVENTSPEC       *           [0]  [1]       [2]     [3]       * @@ -819,8 +843,8 @@ TclChanPostEventObjCmd(      hPtr = Tcl_FindHashEntry(&rcmPtr->map, chanId);      if (hPtr == NULL) { -	Tcl_AppendResult(interp, "can not find reflected channel named \"", -		chanId, "\"", NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "can not find reflected channel named \"%s\"", chanId));  	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "CHANNEL", chanId, NULL);  	return TCL_ERROR;      } @@ -877,8 +901,9 @@ TclChanPostEventObjCmd(       */      if (events & ~rcPtr->interest) { -	Tcl_AppendResult(interp, "tried to post events channel \"", chanId, -		"\" is not interested in", NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "tried to post events channel \"%s\" is not interested in", +                chanId));  	return TCL_ERROR;      } @@ -886,7 +911,44 @@ TclChanPostEventObjCmd(       * We have the channel and the events to post.       */ -    Tcl_NotifyChannel(chan, events); +#ifdef TCL_THREADS +    if (rcPtr->owner == rcPtr->thread) { +#endif +        Tcl_NotifyChannel(chan, events); +#ifdef TCL_THREADS +    } else { +        ReflectEvent *ev = ckalloc(sizeof(ReflectEvent)); + +        ev->header.proc = ReflectEventRun; +        ev->events = events; +        ev->rcPtr = rcPtr; + +        /* +         * We are not preserving the structure here. When the channel is +         * closed any pending events are deleted, see ReflectClose(), and +         * ReflectEventDelete(). Trying to preserve and later release when the +         * event is run may generate a situation where the channel structure +         * is deleted but not our structure, crashing in +         * FreeReflectedChannel(). +         * +         * Force creation of the RCM, for proper cleanup on thread teardown. +         * The teardown of unprocessed events is currently coupled to the +         * thread reflected channel map +         */ + +        (void) GetThreadReflectedChannelMap(); + +        /* XXX Race condition !! +         * XXX The destination thread may not exist anymore already. +         * XXX (Delayed postevent executed after channel got removed). +         * XXX Can we detect this ? (check the validity of the owner threadid ?) +         * XXX Actually, in that case the channel should be dead also ! +         */ + +        Tcl_ThreadQueueEvent(rcPtr->owner, (Tcl_Event *) ev, TCL_QUEUE_TAIL); +        Tcl_ThreadAlert(rcPtr->owner); +    } +#endif      /*       * Squash interp results left by the event script. @@ -1071,13 +1133,14 @@ ReflectClose(  	if (rcPtr->thread != Tcl_GetCurrentThread()) {  	    ForwardParam p; -	    ForwardOpToOwnerThread(rcPtr, ForwardedClose, &p); +	    ForwardOpToHandlerThread(rcPtr, ForwardedClose, &p);  	    result = p.base.code; -	    /* -	     * FreeReflectedChannel is done in the forwarded operation!, in -	     * the other thread. rcPtr here is gone! -	     */ +            /* +             * Now squash the pending reflection events for this channel. +             */ + +            Tcl_DeleteEvents(ReflectEventDelete, rcPtr);  	    if (result != TCL_OK) {  		FreeReceivedError(&p); @@ -1086,19 +1149,7 @@ ReflectClose(  	}  #endif -	FreeReflectedChannel(rcPtr); -	return EOK; -    } - -    /* -     * -- No -- ASSERT rcPtr->methods & FLAG(METH_FINAL) -     * -     * A cleaned method mask here implies that the channel creation was -     * aborted, and "finalize" must not be called. -     */ - -    if (rcPtr->methods == 0) { -	FreeReflectedChannel(rcPtr); +        Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel);  	return EOK;      } @@ -1110,20 +1161,21 @@ ReflectClose(      if (rcPtr->thread != Tcl_GetCurrentThread()) {  	ForwardParam p; -	ForwardOpToOwnerThread(rcPtr, ForwardedClose, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedClose, &p);  	result = p.base.code; -	/* -	 * FreeReflectedChannel is done in the forwarded operation!, in the -	 * other thread. rcPtr here is gone! -	 */ +        /* +         * Now squash the pending reflection events for this channel. +         */ + +        Tcl_DeleteEvents(ReflectEventDelete, rcPtr);  	if (result != TCL_OK) {  	    PassReceivedErrorInterp(interp, &p);  	}      } else {  #endif -	result = InvokeTclMethod(rcPtr, "finalize", NULL, NULL, &resObj); +	result = InvokeTclMethod(rcPtr, METH_FINAL, NULL, NULL, &resObj);  	if ((result != TCL_OK) && (interp != NULL)) {  	    Tcl_SetChannelErrorInterp(interp, resObj);  	} @@ -1144,7 +1196,7 @@ ReflectClose(  	 * the per-interp DeleteReflectedChannelMap exit-handler.  	 */ -	if (rcPtr->interp) { +	if (!rcPtr->dead) {  	    rcmPtr = GetReflectedChannelMap(rcPtr->interp);  	    hPtr = Tcl_FindHashEntry(&rcmPtr->map,  		    Tcl_GetChannelName(rcPtr->chan)); @@ -1161,7 +1213,7 @@ ReflectClose(  	}  #endif -	FreeReflectedChannel(rcPtr); +        Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel);  #ifdef TCL_THREADS      }  #endif @@ -1198,18 +1250,6 @@ ReflectInput(      Tcl_Obj *resObj;		/* Result data for 'read' */      /* -     * The following check can be done before thread redirection, because we -     * are reading from an item which is readonly, i.e. will never change -     * during the lifetime of the channel. -     */ - -    if (!(rcPtr->methods & FLAG(METH_READ))) { -	SetChannelErrorStr(rcPtr->chan, msg_read_unsup); -	*errorCodePtr = EINVAL; -	return -1; -    } - -    /*       * Are we in the correct thread?       */ @@ -1220,7 +1260,7 @@ ReflectInput(  	p.input.buf = buf;  	p.input.toRead = toRead; -	ForwardOpToOwnerThread(rcPtr, ForwardedInput, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedInput, &p);  	if (p.base.code != TCL_OK) {  	    if (p.base.code < 0) { @@ -1242,39 +1282,46 @@ ReflectInput(      /* ASSERT: rcPtr->method & FLAG(METH_READ) */      /* ASSERT: rcPtr->mode & TCL_READABLE */ +    Tcl_Preserve(rcPtr); +      toReadObj = Tcl_NewIntObj(toRead); -    if (InvokeTclMethod(rcPtr, "read", toReadObj, NULL, &resObj)!=TCL_OK) { +    Tcl_IncrRefCount(toReadObj); + +    if (InvokeTclMethod(rcPtr, METH_READ, toReadObj, NULL, &resObj)!=TCL_OK) {  	int code = ErrnoReturn(rcPtr, resObj);  	if (code < 0) { -	    Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	    *errorCodePtr = -code; -	    return -1; +            goto error;  	}  	Tcl_SetChannelError(rcPtr->chan, resObj); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      bytev = Tcl_GetByteArrayFromObj(resObj, &bytec);      if (toRead < bytec) { -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	SetChannelErrorStr(rcPtr->chan, msg_read_toomuch); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      *errorCodePtr = EOK;      if (bytec > 0) { -	memcpy(buf, bytev, (size_t)bytec); +	memcpy(buf, bytev, (size_t) bytec);      } + stop: +    Tcl_DecrRefCount(toReadObj);      Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ +    Tcl_Release(rcPtr);      return bytec; + invalid: +    *errorCodePtr = EINVAL; + error: +    bytec = -1; +    goto stop;  }  /* @@ -1306,18 +1353,6 @@ ReflectOutput(      int written;      /* -     * The following check can be done before thread redirection, because we -     * are reading from an item which is readonly, i.e. will never change -     * during the lifetime of the channel. -     */ - -    if (!(rcPtr->methods & FLAG(METH_WRITE))) { -	SetChannelErrorStr(rcPtr->chan, msg_write_unsup); -	*errorCodePtr = EINVAL; -	return -1; -    } - -    /*       * Are we in the correct thread?       */ @@ -1328,7 +1363,7 @@ ReflectOutput(  	p.output.buf = buf;  	p.output.toWrite = toWrite; -	ForwardOpToOwnerThread(rcPtr, ForwardedOutput, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedOutput, &p);  	if (p.base.code != TCL_OK) {  	    if (p.base.code < 0) { @@ -1350,31 +1385,28 @@ ReflectOutput(      /* ASSERT: rcPtr->method & FLAG(METH_WRITE) */      /* ASSERT: rcPtr->mode & TCL_WRITABLE */ +    Tcl_Preserve(rcPtr); +      bufObj = Tcl_NewByteArrayObj((unsigned char *) buf, toWrite); -    if (InvokeTclMethod(rcPtr, "write", bufObj, NULL, &resObj) != TCL_OK) { +    Tcl_IncrRefCount(bufObj); + +    if (InvokeTclMethod(rcPtr, METH_WRITE, bufObj, NULL, &resObj) != TCL_OK) {  	int code = ErrnoReturn(rcPtr, resObj);  	if (code < 0) { -	    Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	    *errorCodePtr = -code; -	    return -1; +            goto error;  	}  	Tcl_SetChannelError(rcPtr->chan, resObj); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      if (Tcl_GetIntFromObj(rcPtr->interp, resObj, &written) != TCL_OK) { -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	Tcl_SetChannelError(rcPtr->chan, MarshallError(rcPtr->interp)); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      } -    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ -      if ((written == 0) && (toWrite > 0)) {  	/*  	 * The handler claims to have written nothing of what it was @@ -1382,8 +1414,7 @@ ReflectOutput(  	 */  	SetChannelErrorStr(rcPtr->chan, msg_write_nothing); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      if (toWrite < written) {  	/* @@ -1393,12 +1424,20 @@ ReflectOutput(  	 */  	SetChannelErrorStr(rcPtr->chan, msg_write_toomuch); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      *errorCodePtr = EOK; + stop: +    Tcl_DecrRefCount(bufObj); +    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ +    Tcl_Release(rcPtr);      return written; + invalid: +    *errorCodePtr = EINVAL; + error: +    written = -1; +    goto stop;  }  /* @@ -1440,7 +1479,7 @@ ReflectSeekWide(  	p.seek.seekMode = seekMode;  	p.seek.offset = offset; -	ForwardOpToOwnerThread(rcPtr, ForwardedSeek, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedSeek, &p);  	if (p.base.code != TCL_OK) {  	    PassReceivedError(rcPtr->chan, &p); @@ -1456,33 +1495,41 @@ ReflectSeekWide(      /* ASSERT: rcPtr->method & FLAG(METH_SEEK) */ -    offObj = Tcl_NewWideIntObj(offset); -    baseObj = Tcl_NewStringObj((seekMode == SEEK_SET) ? "start" : -	    ((seekMode == SEEK_CUR) ? "current" : "end"), -1); -    if (InvokeTclMethod(rcPtr, "seek", offObj, baseObj, &resObj)!=TCL_OK) { +    Tcl_Preserve(rcPtr); + +    offObj  = Tcl_NewWideIntObj(offset); +    baseObj = Tcl_NewStringObj( +            (seekMode == SEEK_SET) ? "start" : +            (seekMode == SEEK_CUR) ? "current" : "end", -1); +    Tcl_IncrRefCount(offObj); +    Tcl_IncrRefCount(baseObj); + +    if (InvokeTclMethod(rcPtr, METH_SEEK, offObj, baseObj, &resObj)!=TCL_OK) {  	Tcl_SetChannelError(rcPtr->chan, resObj); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      if (Tcl_GetWideIntFromObj(rcPtr->interp, resObj, &newLoc) != TCL_OK) { -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	Tcl_SetChannelError(rcPtr->chan, MarshallError(rcPtr->interp)); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      } -    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ -      if (newLoc < Tcl_LongAsWide(0)) {  	SetChannelErrorStr(rcPtr->chan, msg_seek_beforestart); -	*errorCodePtr = EINVAL; -	return -1; +        goto invalid;      }      *errorCodePtr = EOK; + stop: +    Tcl_DecrRefCount(offObj); +    Tcl_DecrRefCount(baseObj); +    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ +    Tcl_Release(rcPtr);      return newLoc; + invalid: +    *errorCodePtr = EINVAL; +    newLoc = -1; +    goto stop;  }  static int @@ -1528,8 +1575,6 @@ ReflectWatch(      ReflectedChannel *rcPtr = clientData;      Tcl_Obj *maskObj; -    /* ASSERT rcPtr->methods & FLAG(METH_WATCH) */ -      /*       * We restrict the interest to what the channel can support. IOW there       * will never be write events for a channel which is not writable. @@ -1557,7 +1602,7 @@ ReflectWatch(  	ForwardParam p;  	p.watch.mask = mask; -	ForwardOpToOwnerThread(rcPtr, ForwardedWatch, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedWatch, &p);  	/*  	 * Any failure from the forward is ignored. We have no place to put @@ -1568,9 +1613,14 @@ ReflectWatch(      }  #endif +    Tcl_Preserve(rcPtr); +      maskObj = DecodeEventMask(mask); -    (void) InvokeTclMethod(rcPtr, "watch", maskObj, NULL, NULL); +    /* assert maskObj.refCount == 1 */ +    (void) InvokeTclMethod(rcPtr, METH_WATCH, maskObj, NULL, NULL);      Tcl_DecrRefCount(maskObj); + +    Tcl_Release(rcPtr);  }  /* @@ -1610,7 +1660,7 @@ ReflectBlock(  	p.block.nonblocking = nonblocking; -	ForwardOpToOwnerThread(rcPtr, ForwardedBlock, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedBlock, &p);  	if (p.base.code != TCL_OK) {  	    PassReceivedError(rcPtr->chan, &p); @@ -1622,18 +1672,62 @@ ReflectBlock(  #endif      blockObj = Tcl_NewBooleanObj(!nonblocking); +    Tcl_IncrRefCount(blockObj); -    if (InvokeTclMethod(rcPtr, "blocking", blockObj, NULL, &resObj)!=TCL_OK) { +    Tcl_Preserve(rcPtr); + +    if (InvokeTclMethod(rcPtr,METH_BLOCKING,blockObj,NULL,&resObj)!=TCL_OK) {  	Tcl_SetChannelError(rcPtr->chan, resObj);  	errorNum = EINVAL;      } else {  	errorNum = EOK;      } +    Tcl_DecrRefCount(blockObj);      Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ + +    Tcl_Release(rcPtr);      return errorNum;  } +#ifdef TCL_THREADS +/* + *---------------------------------------------------------------------- + * + * ReflectThread -- + * + *	This function is invoked to tell the channel about thread movements. + * + * Results: + *	None. + * + * Side effects: + *	Allocates memory. Arbitrary, as it calls upon a script. + * + *---------------------------------------------------------------------- + */ + +static void +ReflectThread( +    ClientData clientData, +    int action) +{ +    ReflectedChannel *rcPtr = clientData; + +    switch (action) { +    case TCL_CHANNEL_THREAD_INSERT: +        rcPtr->owner = Tcl_GetCurrentThread(); +        break; +    case TCL_CHANNEL_THREAD_REMOVE: +        rcPtr->owner = NULL; +        break; +    default: +        Tcl_Panic("Unknown thread action code."); +        break; +    } +} + +#endif  /*   *----------------------------------------------------------------------   * @@ -1673,7 +1767,7 @@ ReflectSetOption(  	p.setOpt.name = optionName;  	p.setOpt.value = newValue; -	ForwardOpToOwnerThread(rcPtr, ForwardedSetOpt, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedSetOpt, &p);  	if (p.base.code != TCL_OK) {  	    Tcl_Obj *err = Tcl_NewStringObj(p.base.msgStr, -1); @@ -1686,15 +1780,23 @@ ReflectSetOption(  	return p.base.code;      }  #endif +    Tcl_Preserve(rcPtr);      optionObj = Tcl_NewStringObj(optionName, -1);      valueObj = Tcl_NewStringObj(newValue, -1); -    result = InvokeTclMethod(rcPtr, "configure",optionObj,valueObj, &resObj); + +    Tcl_IncrRefCount(optionObj); +    Tcl_IncrRefCount(valueObj); + +    result = InvokeTclMethod(rcPtr, METH_CONFIGURE,optionObj,valueObj, &resObj);      if (result != TCL_OK) {  	UnmarshallErrorResult(interp, resObj);      } +    Tcl_DecrRefCount(optionObj); +    Tcl_DecrRefCount(valueObj);      Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ +    Tcl_Release(rcPtr);      return result;  } @@ -1729,9 +1831,9 @@ ReflectGetOption(      ReflectedChannel *rcPtr = clientData;      Tcl_Obj *optionObj;      Tcl_Obj *resObj;		/* Result data for 'configure' */ -    int listc; +    int listc, result = TCL_OK;      Tcl_Obj **listv; -    const char *method; +    MethodName method;      /*       * Are we in the correct thread? @@ -1751,7 +1853,7 @@ ReflectGetOption(  	    opcode = ForwardedGetOpt;  	} -	ForwardOpToOwnerThread(rcPtr, opcode, &p); +	ForwardOpToHandlerThread(rcPtr, opcode, &p);  	if (p.base.code != TCL_OK) {  	    Tcl_Obj *err = Tcl_NewStringObj(p.base.msgStr, -1); @@ -1770,21 +1872,23 @@ ReflectGetOption(  	 * Retrieve all options.  	 */ -	method = "cgetall"; +	method = METH_CGETALL;  	optionObj = NULL;      } else {  	/*  	 * Retrieve the value of one option.  	 */ -	method = "cget"; +	method = METH_CGET;  	optionObj = Tcl_NewStringObj(optionName, -1); +        Tcl_IncrRefCount(optionObj);      } +    Tcl_Preserve(rcPtr); +      if (InvokeTclMethod(rcPtr, method, optionObj, NULL, &resObj)!=TCL_OK) {  	UnmarshallErrorResult(interp, resObj); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	return TCL_ERROR; +        goto error;      }      /* @@ -1793,9 +1897,8 @@ ReflectGetOption(       */      if (optionObj != NULL) { -	Tcl_DStringAppend(dsPtr, TclGetString(resObj), -1); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	return TCL_OK; +	TclDStringAppendObj(dsPtr, resObj); +        goto ok;      }      /* @@ -1810,8 +1913,7 @@ ReflectGetOption(       */      if (Tcl_ListObjGetElements(interp, resObj, &listc, &listv) != TCL_OK) { -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	return TCL_ERROR; +        goto error;      }      if ((listc % 2) == 1) { @@ -1824,19 +1926,30 @@ ReflectGetOption(  		"Expected list with even number of "  		"elements, got %d element%s instead", listc,  		(listc == 1 ? "" : "s"))); -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	return TCL_ERROR; +        goto error;      } else {  	int len;  	const char *str = Tcl_GetStringFromObj(resObj, &len);  	if (len) { -	    Tcl_DStringAppend(dsPtr, " ", 1); +	    TclDStringAppendLiteral(dsPtr, " ");  	    Tcl_DStringAppend(dsPtr, str, len);  	} -	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ -	return TCL_OK; +        goto ok; +    } + + ok: +    result = TCL_OK; + stop: +    if (optionObj) { +        Tcl_DecrRefCount(optionObj);      } +    Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ +    Tcl_Release(rcPtr); +    return result; + error: +    result = TCL_ERROR; +    goto stop;  }  /* @@ -1883,7 +1996,8 @@ EncodeEventMask(      }      if (listc < 1) { -	Tcl_AppendResult(interp, "bad ", objName, " list: is empty", NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "bad %s list: is empty", objName));  	return TCL_ERROR;      } @@ -1916,7 +2030,7 @@ EncodeEventMask(   *	This function takes an internal bitmask of events and constructs the   *	equivalent list of event items.   * - * Results: + * Results, Contract:   *	A Tcl_Obj reference. The object will have a refCount of one. The user   *	has to decrement it to release the object.   * @@ -1950,6 +2064,7 @@ DecodeEventMask(      evObj = Tcl_NewStringObj(eventStr, -1);      Tcl_IncrRefCount(evObj); +    /* assert evObj.refCount == 1 */      return evObj;  } @@ -1978,71 +2093,32 @@ NewReflectedChannel(      Tcl_Obj *handleObj)  {      ReflectedChannel *rcPtr; -    int i, listc; -    Tcl_Obj **listv; +    MethodName mn = METH_BLOCKING; -    rcPtr = (ReflectedChannel *) ckalloc(sizeof(ReflectedChannel)); +    rcPtr = ckalloc(sizeof(ReflectedChannel));      /* rcPtr->chan: Assigned by caller. Dummy data here. */ -    /* rcPtr->methods: Assigned by caller. Dummy data here. */      rcPtr->chan = NULL; -    rcPtr->methods = 0;      rcPtr->interp = interp; +    rcPtr->dead = 0;  #ifdef TCL_THREADS      rcPtr->thread = Tcl_GetCurrentThread();  #endif      rcPtr->mode = mode;      rcPtr->interest = 0;		/* Initially no interest registered */ -    /* -     * Method placeholder. -     */ -      /* ASSERT: cmdpfxObj is a Tcl List */ - -    Tcl_ListObjGetElements(interp, cmdpfxObj, &listc, &listv); - -    /* -     * See [==] as well. -     * Storage for the command prefix and the additional words required for -     * the invocation of methods in the command handler. -     * -     * listv [0] [listc-1] | [listc]  [listc+1] | -     * argv  [0]   ... [.] | [argc-2] [argc-1]  | [argc]  [argc+2] -     *       cmd   ... pfx | method   chan      | detail1 detail2 -     */ - -    rcPtr->argc = listc + 2; -    rcPtr->argv = (Tcl_Obj **) ckalloc(sizeof(Tcl_Obj *) * (listc+4)); - -    /* -     * Duplicate object references. -     */ - -    for (i=0; i<listc ; i++) { -	Tcl_Obj *word = rcPtr->argv[i] = listv[i]; - -	Tcl_IncrRefCount(word); -    } - -    i++;				/* Skip placeholder for method */ - -    /* -     * [Bug 1667990]: See [x] in FreeReflectedChannel for release -     */ - -    rcPtr->argv[i] = handleObj; -    Tcl_IncrRefCount(handleObj); - -    /* -     * The next two objects are kept empty, varying arguments. -     */ - -    /* -     * Initialization complete. -     */ - +    rcPtr->cmd = TclListObjCopy(NULL, cmdpfxObj); +    Tcl_IncrRefCount(rcPtr->cmd); +    rcPtr->methods = Tcl_NewListObj(METH_WRITE + 1, NULL); +    while (mn <= METH_WRITE) { +	Tcl_ListObjAppendElement(NULL, rcPtr->methods, +		Tcl_NewStringObj(methodNames[mn++], -1)); +    } +    Tcl_IncrRefCount(rcPtr->methods); +    rcPtr->name = handleObj; +    Tcl_IncrRefCount(rcPtr->name);      return rcPtr;  } @@ -2093,29 +2169,20 @@ FreeReflectedChannel(      ReflectedChannel *rcPtr)  {      Channel *chanPtr = (Channel *) rcPtr->chan; -    int i, n;      if (chanPtr->typePtr != &tclRChannelType) {  	/*  	 * Delete a cloned ChannelType structure.  	 */ -	ckfree((char *) chanPtr->typePtr); +	ckfree(chanPtr->typePtr); +	chanPtr->typePtr = NULL;      } - -    n = rcPtr->argc - 2; -    for (i=0; i<n; i++) { -	Tcl_DecrRefCount(rcPtr->argv[i]); -    } - -    /* -     * [Bug 1667990]: See [x] in NewReflectedChannel for lock. n+1 = argc-1. -     */ - -    Tcl_DecrRefCount(rcPtr->argv[n+1]); - -    ckfree((char *) rcPtr->argv); -    ckfree((char *) rcPtr); +    Tcl_Release(chanPtr); +    Tcl_DecrRefCount(rcPtr->name); +    Tcl_DecrRefCount(rcPtr->methods); +    Tcl_DecrRefCount(rcPtr->cmd); +    ckfree(rcPtr);  }  /* @@ -2134,24 +2201,29 @@ FreeReflectedChannel(   * Side effects:   *	Arbitrary, as it calls upon a Tcl script.   * + * Contract: + *	argOneObj.refCount >= 1 on entry and exit, if argOneObj != NULL + *	argTwoObj.refCount >= 1 on entry and exit, if argTwoObj != NULL + *	resObj.refCount in {0, 1, ...} + *   *----------------------------------------------------------------------   */  static int  InvokeTclMethod(      ReflectedChannel *rcPtr, -    const char *method, +    MethodName method,      Tcl_Obj *argOneObj,		/* NULL'able */      Tcl_Obj *argTwoObj,		/* NULL'able */      Tcl_Obj **resultObjPtr)	/* NULL'able */  { -    int cmdc;			/* #words in constructed command */      Tcl_Obj *methObj = NULL;	/* Method name in object form */      Tcl_InterpState sr;		/* State of handler interp */      int result;			/* Result code of method invokation */      Tcl_Obj *resObj = NULL;	/* Result of method invokation. */ +    Tcl_Obj *cmd; -    if (!rcPtr->interp) { +    if (rcPtr->dead) {  	/*  	 * The channel is marked as dead. Bail out immediately, with an  	 * appropriate error. @@ -2163,49 +2235,37 @@ InvokeTclMethod(  	    Tcl_IncrRefCount(resObj);  	} -	/* -	 * Cleanup of the dynamic parts of the command. -	 */ - -	if (argOneObj) { -	    Tcl_DecrRefCount(argOneObj); -	    if (argTwoObj) { -		Tcl_DecrRefCount(argTwoObj); -	    } -	} +        /* +         * Not touching argOneObj, argTwoObj, they have not been used. +         * See the contract as well. +         */  	return TCL_ERROR;      }      /* -     * NOTE (5): Decide impl. issue: Cache objects with method names? Needs -     * TSD data as reflections can be created in many different threads. -     * NO: Caching of command resolutions means storage per channel. -     */ - -    /* -     * Insert method into the pre-allocated area, after the command prefix, +     * Insert method into the callback command, after the command prefix,       * before the channel id.       */ -    methObj = Tcl_NewStringObj(method, -1); -    Tcl_IncrRefCount(methObj); -    rcPtr->argv[rcPtr->argc - 2] = methObj; +    cmd = TclListObjCopy(NULL, rcPtr->cmd); + +    Tcl_ListObjIndex(NULL, rcPtr->methods, method, &methObj); +    Tcl_ListObjAppendElement(NULL, cmd, methObj); +    Tcl_ListObjAppendElement(NULL, cmd, rcPtr->name);      /*       * Append the additional argument containing method specific details       * behind the channel id. If specified. +     * +     * Because of the contract there is no need to increment the refcounts. +     * The objects will survive the Tcl_EvalObjv without change.       */ -    cmdc = rcPtr->argc;      if (argOneObj) { -	Tcl_IncrRefCount(argOneObj); -	rcPtr->argv[cmdc] = argOneObj; -	cmdc++; +	Tcl_ListObjAppendElement(NULL, cmd, argOneObj);  	if (argTwoObj) { -	    Tcl_IncrRefCount(argTwoObj); -	    rcPtr->argv[cmdc] = argTwoObj; -	    cmdc++; +	    Tcl_ListObjAppendElement(NULL, cmd, argTwoObj);  	}      } @@ -2214,9 +2274,10 @@ InvokeTclMethod(       * existing state intact.       */ +    Tcl_IncrRefCount(cmd);      sr = Tcl_SaveInterpState(rcPtr->interp, 0 /* Dummy */);      Tcl_Preserve(rcPtr->interp); -    result = Tcl_EvalObjv(rcPtr->interp, cmdc, rcPtr->argv, TCL_EVAL_GLOBAL); +    result = Tcl_EvalObjEx(rcPtr->interp, cmd, TCL_EVAL_GLOBAL);      /*       * We do not try to extract the result information if the caller has no @@ -2242,7 +2303,6 @@ InvokeTclMethod(  	     */  	    if (result != TCL_ERROR) { -		Tcl_Obj *cmd = Tcl_NewListObj(cmdc, rcPtr->argv);  		int cmdLen;  		const char *cmdString = Tcl_GetStringFromObj(cmd, &cmdLen); @@ -2256,27 +2316,17 @@ InvokeTclMethod(  		result = TCL_ERROR;  	    }  	    Tcl_AppendObjToErrorInfo(rcPtr->interp, Tcl_ObjPrintf( -		    "\n    (chan handler subcommand \"%s\")", method)); +		    "\n    (chan handler subcommand \"%s\")", +		    methodNames[method]));  	    resObj = MarshallError(rcPtr->interp);  	}  	Tcl_IncrRefCount(resObj);      } +    Tcl_DecrRefCount(cmd);      Tcl_RestoreInterpState(rcPtr->interp, sr);      Tcl_Release(rcPtr->interp);      /* -     * Cleanup of the dynamic parts of the command. -     */ - -    Tcl_DecrRefCount(methObj); -    if (argOneObj) { -	Tcl_DecrRefCount(argOneObj); -	if (argTwoObj) { -	    Tcl_DecrRefCount(argTwoObj); -	} -    } - -    /*       * The resObj has a ref count of 1 at this location. This means that the       * caller of InvokeTclMethod has to dispose of it (but only if it was       * returned to it). @@ -2322,7 +2372,7 @@ ErrnoReturn(      int code;      Tcl_InterpState sr;		/* State of handler interp */ -    if (!rcPtr->interp) { +    if (rcPtr->dead) {  	return 0;      } @@ -2334,7 +2384,7 @@ ErrnoReturn(      if (((Tcl_GetIntFromObj(rcPtr->interp, resObj, &code) != TCL_OK)  	    || (code >= 0))) {  	if (strcmp("EAGAIN", Tcl_GetString(resObj)) == 0) { -	    code = - EAGAIN; +	    code = -EAGAIN;  	} else {  	    code = 0;  	} @@ -2368,7 +2418,7 @@ GetReflectedChannelMap(      ReflectedChannelMap *rcmPtr = Tcl_GetAssocData(interp, RCMKEY, NULL);      if (rcmPtr == NULL) { -	rcmPtr = (ReflectedChannelMap *) ckalloc(sizeof(ReflectedChannelMap)); +	rcmPtr = ckalloc(sizeof(ReflectedChannelMap));  	Tcl_InitHashTable(&rcmPtr->map, TCL_STRING_KEYS);  	Tcl_SetAssocData(interp, RCMKEY,  		(Tcl_InterpDeleteProc *) DeleteReflectedChannelMap, rcmPtr); @@ -2431,11 +2481,11 @@ DeleteReflectedChannelMap(  	chan = Tcl_GetHashValue(hPtr);  	rcPtr = Tcl_GetChannelInstanceData(chan); -	rcPtr->interp = NULL; +	rcPtr->dead = 1;  	Tcl_DeleteHashEntry(hPtr);      }      Tcl_DeleteHashTable(&rcmPtr->map); -    ckfree((char *) &rcmPtr->map); +    ckfree(&rcmPtr->map);  #ifdef TCL_THREADS      /* @@ -2467,6 +2517,11 @@ DeleteReflectedChannelMap(  	 */  	evPtr = resultPtr->evPtr; + +	/* Basic crash safety until this routine can get revised [3411310] */ +	if (evPtr == NULL) { +	    continue; +	}  	paramPtr = evPtr->param;  	evPtr->resultPtr = NULL; @@ -2477,6 +2532,7 @@ DeleteReflectedChannelMap(  	Tcl_ConditionNotify(&resultPtr->done);      } +    Tcl_MutexUnlock(&rcForwardMutex);      /*       * Get the map of all channels handled by the current thread. This is a @@ -2500,10 +2556,9 @@ DeleteReflectedChannelMap(  	    continue;  	} +	rcPtr->dead = 1;  	Tcl_DeleteHashEntry(hPtr);      } - -    Tcl_MutexUnlock(&rcForwardMutex);  #endif  } @@ -2531,8 +2586,7 @@ GetThreadReflectedChannelMap(void)      ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey);      if (!tsdPtr->rcmPtr) { -	tsdPtr->rcmPtr = (ReflectedChannelMap *) -		ckalloc(sizeof(ReflectedChannelMap)); +	tsdPtr->rcmPtr = ckalloc(sizeof(ReflectedChannelMap));  	Tcl_InitHashTable(&tsdPtr->rcmPtr->map, TCL_STRING_KEYS);  	Tcl_CreateThreadExitHandler(DeleteThreadReflectedChannelMap, NULL);      } @@ -2602,6 +2656,11 @@ DeleteThreadReflectedChannelMap(  	 */  	evPtr = resultPtr->evPtr; + +	/* Basic crash safety until this routine can get revised [3411310] */ +	if (evPtr == NULL ) { +	    continue; +	}  	paramPtr = evPtr->param;  	evPtr->resultPtr = NULL; @@ -2612,6 +2671,16 @@ DeleteThreadReflectedChannelMap(  	Tcl_ConditionNotify(&resultPtr->done);      } +    Tcl_MutexUnlock(&rcForwardMutex); + +    /* +     * Run over the event queue of this thread and remove all ReflectEvent's +     * still pending. These are inbound events for reflected channels this +     * thread owns but doesn't handle. The inverse of the channel map +     * actually. +     */ + +    Tcl_DeleteEvents(ReflectEventDelete, NULL);      /*       * Get the map of all channels handled by the current thread. This is a @@ -2626,23 +2695,26 @@ DeleteThreadReflectedChannelMap(  	Tcl_Channel chan = Tcl_GetHashValue(hPtr);  	ReflectedChannel *rcPtr = Tcl_GetChannelInstanceData(chan); -	rcPtr->interp = NULL; +	rcPtr->dead = 1;  	Tcl_DeleteHashEntry(hPtr);      } - -    Tcl_MutexUnlock(&rcForwardMutex); +    ckfree(rcmPtr);  }  static void -ForwardOpToOwnerThread( +ForwardOpToHandlerThread(      ReflectedChannel *rcPtr,	/* Channel instance */      ForwardedOperation op,	/* Forwarded driver operation */      const void *param)		/* Arguments */  { +    /* +     * Core of the communication from OWNER to HANDLER thread. +     * The receiver is ForwardProc() below. +     */ +      Tcl_ThreadId dst = rcPtr->thread;      ForwardingEvent *evPtr;      ForwardingResult *resultPtr; -    int result;      /*       * We gather the lock early. This allows us to check the liveness of the @@ -2651,7 +2723,7 @@ ForwardOpToOwnerThread(      Tcl_MutexLock(&rcForwardMutex); -    if (rcPtr->interp == NULL) { +    if (rcPtr->dead) {  	/*  	 * The channel is marked as dead. Bail out immediately, with an  	 * appropriate error. Do not forget to unlock the mutex on this path. @@ -2666,8 +2738,8 @@ ForwardOpToOwnerThread(       * Create and initialize the event and data structures.       */ -    evPtr = (ForwardingEvent *) ckalloc(sizeof(ForwardingEvent)); -    resultPtr = (ForwardingResult *) ckalloc(sizeof(ForwardingResult)); +    evPtr = ckalloc(sizeof(ForwardingEvent)); +    resultPtr = ckalloc(sizeof(ForwardingResult));      evPtr->event.proc = ForwardProc;      evPtr->resultPtr = resultPtr; @@ -2692,7 +2764,7 @@ ForwardOpToOwnerThread(      /*       * Ensure cleanup of the event if the origin thread exits while this event       * is pending or in progress. Exit of the destination thread is handled by -     * DeleteThreadReflectionChannelMap(), this is set up by +     * DeleteThreadReflectedChannelMap(), this is set up by       * GetThreadReflectedChannelMap(). This is what we use the 'forwardList'       * (see above) for.       */ @@ -2707,7 +2779,7 @@ ForwardOpToOwnerThread(      Tcl_ThreadAlert(dst);      /* -     * (*) Block until the other thread has either processed the transfer or +     * (*) Block until the handler thread has either processed the transfer or       * rejected it.       */ @@ -2746,8 +2818,7 @@ ForwardOpToOwnerThread(      Tcl_DeleteThreadExitHandler(SrcExitProc, evPtr); -    result = resultPtr->result; -    ckfree((char *) resultPtr); +    ckfree(resultPtr);  }  static int @@ -2756,6 +2827,11 @@ ForwardProc(      int mask)  {      /* +     * HANDLER thread. + +     * The receiver part for the operations coming from the OWNER thread. +     * See ForwardOpToHandlerThread() for the transmitter. +     *       * Notes regarding access to the referenced data.       *       * In principle the data belongs to the originating thread (see @@ -2774,9 +2850,8 @@ ForwardProc(      Tcl_Interp *interp = rcPtr->interp;      ForwardParam *paramPtr = evPtr->param;      Tcl_Obj *resObj = NULL;	/* Interp result of InvokeTclMethod */ -    ReflectedChannelMap *rcmPtr; -				/* Map of reflected channels with handlers in -				 * this interp. */ +    ReflectedChannelMap *rcmPtr;/* Map of reflected channels with handlers in +                                 * this interp. */      Tcl_HashEntry *hPtr;	/* Entry in the above map */      /* @@ -2803,12 +2878,12 @@ ForwardProc(  	 * No parameters/results.  	 */ -	if (InvokeTclMethod(rcPtr, "finalize", NULL, NULL, &resObj)!=TCL_OK) { +	if (InvokeTclMethod(rcPtr, METH_FINAL, NULL, NULL, &resObj)!=TCL_OK) {  	    ForwardSetObjError(paramPtr, resObj);  	}  	/* -	 * Freeing is done here, in the origin thread, because the argv[] +	 * Freeing is done here, in the origin thread, callback command  	 * objects belong to this thread. Deallocating them in a different  	 * thread is not allowed  	 * @@ -2819,21 +2894,23 @@ ForwardProc(  	rcmPtr = GetReflectedChannelMap(interp);  	hPtr = Tcl_FindHashEntry(&rcmPtr->map, -		Tcl_GetChannelName(rcPtr->chan)); +                Tcl_GetChannelName(rcPtr->chan));  	Tcl_DeleteHashEntry(hPtr);  	rcmPtr = GetThreadReflectedChannelMap();  	hPtr = Tcl_FindHashEntry(&rcmPtr->map, -		Tcl_GetChannelName(rcPtr->chan)); +                Tcl_GetChannelName(rcPtr->chan));  	Tcl_DeleteHashEntry(hPtr); -	FreeReflectedChannel(rcPtr); +	Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel);  	break;      case ForwardedInput: {  	Tcl_Obj *toReadObj = Tcl_NewIntObj(paramPtr->input.toRead); +        Tcl_IncrRefCount(toReadObj); -	if (InvokeTclMethod(rcPtr, "read", toReadObj, NULL, &resObj)!=TCL_OK){ +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_READ, toReadObj, NULL, &resObj)!=TCL_OK){  	    int code = ErrnoReturn(rcPtr, resObj);  	    if (code < 0) { @@ -2857,19 +2934,23 @@ ForwardProc(  		paramPtr->input.toRead = -1;  	    } else {  		if (bytec > 0) { -		    memcpy(paramPtr->input.buf, bytev, (size_t)bytec); +		    memcpy(paramPtr->input.buf, bytev, (size_t) bytec);  		}  		paramPtr->input.toRead = bytec;  	    }  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(toReadObj);  	break;      }      case ForwardedOutput: {  	Tcl_Obj *bufObj = Tcl_NewByteArrayObj((unsigned char *) -		paramPtr->output.buf, paramPtr->output.toWrite); +                paramPtr->output.buf, paramPtr->output.toWrite); +        Tcl_IncrRefCount(bufObj); -	if (InvokeTclMethod(rcPtr, "write", bufObj, NULL, &resObj) != TCL_OK) { +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_WRITE, bufObj, NULL, &resObj) != TCL_OK) {  	    int code = ErrnoReturn(rcPtr, resObj);  	    if (code < 0) { @@ -2886,7 +2967,9 @@ ForwardProc(  	    int written;  	    if (Tcl_GetIntFromObj(interp, resObj, &written) != TCL_OK) { -		ForwardSetObjError(paramPtr, MarshallError(interp)); +		Tcl_DecrRefCount(resObj); +		resObj = MarshallError(interp); +		ForwardSetObjError(paramPtr, resObj);  		paramPtr->output.toWrite = -1;  	    } else if (written==0 || paramPtr->output.toWrite<written) {  		ForwardSetStaticError(paramPtr, msg_write_toomuch); @@ -2895,16 +2978,22 @@ ForwardProc(  		paramPtr->output.toWrite = written;  	    }  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(bufObj);  	break;      }      case ForwardedSeek: {  	Tcl_Obj *offObj = Tcl_NewWideIntObj(paramPtr->seek.offset);  	Tcl_Obj *baseObj = Tcl_NewStringObj( -		(paramPtr->seek.seekMode==SEEK_SET) ? "start" : -		(paramPtr->seek.seekMode==SEEK_CUR) ? "current" : "end", -1); +                (paramPtr->seek.seekMode==SEEK_SET) ? "start" : +                (paramPtr->seek.seekMode==SEEK_CUR) ? "current" : "end", -1); -	if (InvokeTclMethod(rcPtr, "seek", offObj, baseObj, &resObj)!=TCL_OK){ +        Tcl_IncrRefCount(offObj); +        Tcl_IncrRefCount(baseObj); + +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_SEEK, offObj, baseObj, &resObj)!=TCL_OK){  	    ForwardSetObjError(paramPtr, resObj);  	    paramPtr->seek.offset = -1;  	} else { @@ -2923,39 +3012,57 @@ ForwardProc(  		    paramPtr->seek.offset = newLoc;  		}  	    } else { -		ForwardSetObjError(paramPtr, MarshallError(interp)); +		Tcl_DecrRefCount(resObj); +		resObj = MarshallError(interp); +		ForwardSetObjError(paramPtr, resObj);  		paramPtr->seek.offset = -1;  	    }  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(offObj); +        Tcl_DecrRefCount(baseObj);  	break;      }      case ForwardedWatch: {  	Tcl_Obj *maskObj = DecodeEventMask(paramPtr->watch.mask); +        /* assert maskObj.refCount == 1 */ -	(void) InvokeTclMethod(rcPtr, "watch", maskObj, NULL, NULL); +        Tcl_Preserve(rcPtr); +	(void) InvokeTclMethod(rcPtr, METH_WATCH, maskObj, NULL, NULL);  	Tcl_DecrRefCount(maskObj); +        Tcl_Release(rcPtr);  	break;      }      case ForwardedBlock: {  	Tcl_Obj *blockObj = Tcl_NewBooleanObj(!paramPtr->block.nonblocking); -	if (InvokeTclMethod(rcPtr, "blocking", blockObj, NULL, -		&resObj) != TCL_OK) { +        Tcl_IncrRefCount(blockObj); +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_BLOCKING, blockObj, NULL, +                &resObj) != TCL_OK) {  	    ForwardSetObjError(paramPtr, resObj);  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(blockObj);  	break;      }      case ForwardedSetOpt: {  	Tcl_Obj *optionObj = Tcl_NewStringObj(paramPtr->setOpt.name, -1); -	Tcl_Obj *valueObj = Tcl_NewStringObj(paramPtr->setOpt.value, -1); +	Tcl_Obj *valueObj  = Tcl_NewStringObj(paramPtr->setOpt.value, -1); -	if (InvokeTclMethod(rcPtr, "configure", optionObj, valueObj, -		&resObj) != TCL_OK) { +        Tcl_IncrRefCount(optionObj); +        Tcl_IncrRefCount(valueObj); +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_CONFIGURE, optionObj, valueObj, +                &resObj) != TCL_OK) {  	    ForwardSetObjError(paramPtr, resObj);  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(optionObj); +        Tcl_DecrRefCount(valueObj);  	break;      } @@ -2966,12 +3073,15 @@ ForwardProc(  	Tcl_Obj *optionObj = Tcl_NewStringObj(paramPtr->getOpt.name, -1); -	if (InvokeTclMethod(rcPtr, "cget", optionObj, NULL, &resObj)!=TCL_OK){ +        Tcl_IncrRefCount(optionObj); +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_CGET, optionObj, NULL, &resObj)!=TCL_OK){  	    ForwardSetObjError(paramPtr, resObj);  	} else { -	    Tcl_DStringAppend(paramPtr->getOpt.value, -		    TclGetString(resObj), -1); +	    TclDStringAppendObj(paramPtr->getOpt.value, resObj);  	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(optionObj);  	break;      } @@ -2980,7 +3090,8 @@ ForwardProc(  	 * Retrieve all options.  	 */ -	if (InvokeTclMethod(rcPtr, "cgetall", NULL, NULL, &resObj) != TCL_OK){ +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_CGETALL, NULL, NULL, &resObj) != TCL_OK){  	    ForwardSetObjError(paramPtr, resObj);  	} else {  	    /* @@ -2992,8 +3103,10 @@ ForwardProc(  	    Tcl_Obj **listv;  	    if (Tcl_ListObjGetElements(interp, resObj, &listc, -		    &listv) != TCL_OK) { -		ForwardSetObjError(paramPtr, MarshallError(interp)); +                    &listv) != TCL_OK) { +		Tcl_DecrRefCount(resObj); +		resObj = MarshallError(interp); +		ForwardSetObjError(paramPtr, resObj);  	    } else if ((listc % 2) == 1) {  		/*  		 * Odd number of elements is wrong. [x]. @@ -3010,11 +3123,12 @@ ForwardProc(  		const char *str = Tcl_GetStringFromObj(resObj, &len);  		if (len) { -		    Tcl_DStringAppend(paramPtr->getOpt.value, " ", 1); +		    TclDStringAppendLiteral(paramPtr->getOpt.value, " ");  		    Tcl_DStringAppend(paramPtr->getOpt.value, str, len);  		}  	    }  	} +        Tcl_Release(rcPtr);  	break;      default: @@ -3108,7 +3222,7 @@ ForwardSetObjError(      const char *msgStr = Tcl_GetStringFromObj(obj, &len);      len++; -    ForwardSetDynamicError(paramPtr, ckalloc((unsigned) len)); +    ForwardSetDynamicError(paramPtr, ckalloc(len));      memcpy(paramPtr->base.msgStr, msgStr, (unsigned) len);  }  #endif | 
