diff options
Diffstat (limited to 'generic/tclIORChan.c')
| -rw-r--r-- | generic/tclIORChan.c | 3947 | 
1 files changed, 2261 insertions, 1686 deletions
| diff --git a/generic/tclIORChan.c b/generic/tclIORChan.c index 0a57eb3..94428bb 100644 --- a/generic/tclIORChan.c +++ b/generic/tclIORChan.c @@ -1,99 +1,85 @@ -/*  +/*   * tclIORChan.c --   * - *	This file contains the implementation of Tcl's generic - *	channel reflection code, which allows the implementation - *	of Tcl channels in Tcl code. + *	This file contains the implementation of Tcl's generic channel + *	reflection code, which allows the implementation of Tcl channels in + *	Tcl code.   * - *	Parts of this file are based on code contributed by  - *	Jean-Claude Wippler. + *	Parts of this file are based on code contributed by Jean-Claude + *	Wippler.   * - *      See TIP #219 for the specification of this functionality. + *	See TIP #219 for the specification of this functionality.   *   * Copyright (c) 2004-2005 ActiveState, a divison of Sophos   * - * 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.5 2005/09/15 16:40:02 dgp Exp $ + * See the file "license.terms" for information on usage and redistribution of + * this file, and for a DISCLAIMER OF ALL WARRANTIES.   */ -#include <tclInt.h> -#include <tclIO.h> +#include "tclInt.h" +#include "tclIO.h"  #include <assert.h>  #ifndef EINVAL -#define EINVAL 9 +#define EINVAL	9  #endif  #ifndef EOK -#define EOK    0 +#define EOK	0  #endif  /*   * Signatures of all functions used in the C layer of the reflection.   */ -/* Required */ -static int	RcClose _ANSI_ARGS_((ClientData clientData, -		   Tcl_Interp *interp)); - -/* Required, "read" is optional despite this. */ -static int	RcInput _ANSI_ARGS_((ClientData clientData, -		    char *buf, int toRead, int *errorCodePtr)); - -/* Required, "write" is optional despite this. */ -static int	RcOutput _ANSI_ARGS_((ClientData clientData, -	            CONST char *buf, int toWrite, int *errorCodePtr)); - -/* Required */ -static void	RcWatch _ANSI_ARGS_((ClientData clientData, int mask)); - -/* NULL'able - "blocking", is optional */ -static int	RcBlock _ANSI_ARGS_((ClientData clientData, -				       int mode)); - -/* NULL'able - "seek", is optional */ -static Tcl_WideInt RcSeekWide _ANSI_ARGS_((ClientData clientData, -		    Tcl_WideInt offset, -		    int mode, int *errorCodePtr)); - -static int RcSeek _ANSI_ARGS_((ClientData clientData, -		    long offset, int mode, int *errorCodePtr)); - -/* NULL'able - "cget" / "cgetall", are optional */ -static int	RcGetOption _ANSI_ARGS_((ClientData clientData, -				       Tcl_Interp* interp, -				       CONST char *optionName, -				       Tcl_DString *dsPtr)); - -/* NULL'able - "configure", is optional */ -static int	RcSetOption _ANSI_ARGS_((ClientData clientData, -				       Tcl_Interp* interp, -				       CONST char *optionName, -				       CONST char *newValue)); - +static int		ReflectClose(ClientData clientData, +			    Tcl_Interp *interp); +static int		ReflectInput(ClientData clientData, char *buf, +			    int toRead, int *errorCodePtr); +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, +			    int mode, int *errorCodePtr); +static int		ReflectGetOption(ClientData clientData, +			    Tcl_Interp *interp, const char *optionName, +			    Tcl_DString *dsPtr); +static int		ReflectSetOption(ClientData clientData, +			    Tcl_Interp *interp, const char *optionName, +			    const char *newValue);  /* - * The C layer channel type/driver definition used by the reflection. - * This is a version 3 structure. + * The C layer channel type/driver definition used by the reflection. This is + * a version 3 structure.   */ -static Tcl_ChannelType tclRChannelType = { -  "tclrchannel",  /* Type name.                                    */ -  TCL_CHANNEL_VERSION_3, -  RcClose,        /* Close channel, clean instance data            */ -  RcInput,        /* Handle read request                           */ -  RcOutput,       /* Handle write request                          */ -  RcSeek,         /* Move location of access point.    NULL'able   */ -  RcSetOption,    /* Set options.                      NULL'able   */ -  RcGetOption,    /* Get options.                      NULL'able   */ -  RcWatch,        /* Initialize notifier                           */ -  NULL,           /* Get OS handle from the channel.   NULL'able   */ -  NULL,           /* No close2 support.                NULL'able   */ -  RcBlock,        /* Set blocking/nonblocking.         NULL'able   */ -  NULL,           /* Flush channel. Not used by core.  NULL'able   */ -  NULL,           /* Handle events.                    NULL'able   */ -  RcSeekWide      /* Move access point (64 bit).       NULL'able   */ +static const Tcl_ChannelType tclRChannelType = { +    "tclrchannel",	   /* Type name.				  */ +    TCL_CHANNEL_VERSION_5, /* v5 channel */ +    ReflectClose,	   /* Close channel, clean instance data	  */ +    ReflectInput,	   /* Handle read request			  */ +    ReflectOutput,	   /* Handle write request			  */ +    ReflectSeek,	   /* Move location of access point.	NULL'able */ +    ReflectSetOption,	   /* Set options.			NULL'able */ +    ReflectGetOption,	   /* Get options.			NULL'able */ +    ReflectWatch,	   /* Initialize notifier			  */ +    NULL,		   /* Get OS handle from the channel.	NULL'able */ +    NULL,		   /* No close2 support.		NULL'able */ +    ReflectBlock,	   /* Set blocking/nonblocking.		NULL'able */ +    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 */  };  /* @@ -101,125 +87,124 @@ static Tcl_ChannelType tclRChannelType = {   */  typedef struct { -  Tcl_Channel chan;    /* Back reference to generic channel structure. -		        */ -  Tcl_Interp* interp;  /* Reference to the interpreter containing the -		        * Tcl level part of the channel. */ +    Tcl_Channel chan;		/* Back reference to generic channel +				 * structure. */ +    Tcl_Interp *interp;		/* Reference to the interpreter containing the +				 * Tcl level part of the channel. NULL here +				 * signals the channel is dead because the +				 * interpreter/thread containing its Tcl +				 * 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 +    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. +     * +     * Most channel implementations need a timer in the C level to ensure that +     * data in buffers is flushed out through the generation of fake file +     * events. +     * +     * See 'rechan', 'memchan', etc. +     * +     * Here this is _not_ required. Interest in events is posted to the Tcl +     * level via 'watch'. And posting of events is possible from the Tcl level +     * as well, via 'chan postevent'. This means that the generation of all +     * events, fake or not, timer based or not, is completely in the hands of +     * the Tcl level. Therefore no timer here. +     */ +} ReflectedChannel; -  /* 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 -   * NOTE (9): for the method names ? -   */ - -  /* ---------------------------------------- */ - -  int mode;             /* Mask of R/W mode */ -  int interest;         /* Mask of events the channel is interested in. */ - -  /* Note regarding the usage of timers. -   * -   * Most channel implementations need a timer in the -   * C level to ensure that data in buffers is flushed -   * out through the generation of fake file events. -   * -   * See 'rechan', 'memchan', etc. -   * -   * Here this is _not_ required. Interest in events is -   * posted to the Tcl level via 'watch'. And posting of -   * events is possible from the Tcl level as well, via -   * 'chan postevent'. This means that the generation of -   * all events, fake or not, timer based or not, is -   * completely in the hands of the Tcl level. Therefore -   * no timer here. -   */ - -} ReflectingChannel; +/* + * Structure of the table maping from channel handles to reflected + * channels. Each interpreter which has the handler command for one or more + * reflected channels records them in such a table, so that 'chan postevent' + * is able to find them even if the actual channel was moved to a different + * interpreter and/or thread. + * + * The table is reachable via the standard interpreter AssocData, the key is + * defined below. + */ + +typedef struct { +    Tcl_HashTable map; +} ReflectedChannelMap; + +#define RCMKEY "ReflectedChannelMap"  /*   * Event literals. ==================================================   */ -static CONST char *eventOptions[] = { -  "read", "write", (char *) NULL +static const char *const eventOptions[] = { +    "read", "write", NULL  };  typedef enum { -  EVENT_READ, EVENT_WRITE +    EVENT_READ, EVENT_WRITE  } EventOption;  /*   * Method literals. ==================================================   */ -static CONST char *methodNames[] = { -  "blocking",	/* OPT */ -  "cget",	/* OPT \/ Together or none */ -  "cgetall",	/* OPT /\ of these two     */ -  "configure",	/* OPT */ -  "finalize",	/*     */ -  "initialize",	/*     */ -  "read",	/* OPT */ -  "seek",	/* OPT */ -  "watch",	/*     */ -  "write",	/* OPT */ -  (char *) NULL +static const char *const methodNames[] = { +    "blocking",		/* OPT */ +    "cget",		/* OPT \/ Together or none */ +    "cgetall",		/* OPT /\ of these two     */ +    "configure",	/* OPT */ +    "finalize",		/*     */ +    "initialize",	/*     */ +    "read",		/* OPT */ +    "seek",		/* OPT */ +    "watch",		/*     */ +    "write",		/* OPT */ +    NULL  };  typedef enum { -  METH_BLOCKING, -  METH_CGET, -  METH_CGETALL, -  METH_CONFIGURE, -  METH_FINAL, -  METH_INIT, -  METH_READ, -  METH_SEEK, -  METH_WATCH, -  METH_WRITE, +    METH_BLOCKING, +    METH_CGET, +    METH_CGETALL, +    METH_CONFIGURE, +    METH_FINAL, +    METH_INIT, +    METH_READ, +    METH_SEEK, +    METH_WATCH, +    METH_WRITE  } MethodName;  #define FLAG(m) (1 << (m)) -#define REQUIRED_METHODS (FLAG (METH_INIT) | FLAG (METH_FINAL) | FLAG (METH_WATCH)) -#define NULLABLE_METHODS (FLAG (METH_BLOCKING) | FLAG (METH_SEEK) | \ -	FLAG (METH_CONFIGURE) | FLAG (METH_CGET) | FLAG (METH_CGETALL)) +#define REQUIRED_METHODS \ +	(FLAG(METH_INIT) | FLAG(METH_FINAL) | FLAG(METH_WATCH)) +#define NULLABLE_METHODS \ +	(FLAG(METH_BLOCKING) | FLAG(METH_SEEK) | \ +	FLAG(METH_CONFIGURE) | FLAG(METH_CGET) | FLAG(METH_CGETALL)) -#define RANDW (TCL_READABLE|TCL_WRITABLE) +#define RANDW \ +	(TCL_READABLE | TCL_WRITABLE) -#define IMPLIES(a,b) ((!(a)) || (b)) +#define IMPLIES(a,b)	((!(a)) || (b))  #define NEGIMPL(a,b) -#define HAS(x,f) (x & FLAG(f)) - +#define HAS(x,f)	(x & FLAG(f))  #ifdef TCL_THREADS  /*   * Thread specific types and structures.   * - * We are here essentially creating a very specific implementation of - * 'thread send'. + * We are here essentially creating a very specific implementation of 'thread + * send'.   */  /* @@ -227,216 +212,230 @@ typedef enum {   */  typedef enum { -  RcOpClose, -  RcOpInput, -  RcOpOutput, -  RcOpSeek, -  RcOpWatch, -  RcOpBlock, -  RcOpSetOpt, -  RcOpGetOpt, -  RcOpGetOptAll -} RcOperation; +    ForwardedClose, +    ForwardedInput, +    ForwardedOutput, +    ForwardedSeek, +    ForwardedWatch, +    ForwardedBlock, +    ForwardedSetOpt, +    ForwardedGetOpt, +    ForwardedGetOptAll +} ForwardedOperation;  /* - * Event used to forward driver invocations to the thread actually - * managing the channel. We cannot construct the command to execute - * and forward that. Because then it will contain a mixture of - * Tcl_Obj's belonging to both the command handler thread (CT), and - * the thread managing the channel (MT), executed in CT. Tcl_Obj's are - * not allowed to cross thread boundaries. So we forward an operation - * code, the argument details ,and reference to results. The command - * is assembled in the CT and belongs fully to that thread. No sharing - * problems. + * Event used to forward driver invocations to the thread actually managing + * the channel. We cannot construct the command to execute and forward that. + * Because then it will contain a mixture of Tcl_Obj's belonging to both the + * command handler thread (CT), and the thread managing the channel (MT), + * executed in CT. Tcl_Obj's are not allowed to cross thread boundaries. So we + * forward an operation code, the argument details, and reference to results. + * The command is assembled in the CT and belongs fully to that thread. No + * sharing problems.   */ -typedef struct RcForwardParamBase { -  int   code; /* O: Ok/Fail of the cmd handler */ -  char* msg;  /* O: Error message for handler failure */ -  int   vol;  /* O: True - msg is allocated, False - msg is static */ -} RcForwardParamBase; +typedef struct ForwardParamBase { +    int code;			/* O: Ok/Fail of the cmd handler */ +    char *msgStr;		/* O: Error message for handler failure */ +    int mustFree;		/* O: True if msgStr is allocated, false if +				 * otherwise (static). */ +} ForwardParamBase;  /* - * Operation specific parameter/result structures. + * Operation specific parameter/result structures. (These are "subtypes" of + * ForwardParamBase. Where an operation does not need any special types, it + * has no "subtype" and just uses ForwardParamBase, as listed above.)   */ -typedef struct RcForwardParamClose { -  RcForwardParamBase b; -} RcForwardParamClose; - -typedef struct RcForwardParamInput { -  RcForwardParamBase b; -  char* buf;    /* O: Where to store the read bytes */ -  int   toRead; /* I: #bytes to read, -		 * O: #bytes actually read */ -} RcForwardParamInput; - -typedef struct RcForwardParamOutput { -  RcForwardParamBase b; -  CONST char* buf;     /* I: Where the bytes to write come from */ -  int         toWrite; /* I: #bytes to write, -			* O: #bytes actually written */ -} RcForwardParamOutput; - -typedef struct RcForwardParamSeek { -  RcForwardParamBase b; -  int         seekMode; /* I: How to seek */ -  Tcl_WideInt offset;   /* I: Where to seek, -			 * O: New location */ -} RcForwardParamSeek; - -typedef struct RcForwardParamWatch { -  RcForwardParamBase b; -  int mask; /* I: What events to watch for */ -} RcForwardParamWatch; - -typedef struct RcForwardParamBlock { -  RcForwardParamBase b; -  int nonblocking; /* I: What mode to activate */ -} RcForwardParamBlock; - -typedef struct RcForwardParamSetOpt { -  RcForwardParamBase b; -  CONST char* name;  /* Name of option to set */ -  CONST char* value; /* Value to set */ -} RcForwardParamSetOpt; - -typedef struct RcForwardParamGetOpt { -  RcForwardParamBase b; -  CONST char*  name;  /* Name of option to get, maybe NULL */ -  Tcl_DString* value; /* Result */ -} RcForwardParamGetOpt; +struct ForwardParamInput { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    char *buf;			/* O: Where to store the read bytes */ +    int toRead;			/* I: #bytes to read, +				 * O: #bytes actually read */ +}; +struct ForwardParamOutput { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    const char *buf;		/* I: Where the bytes to write come from */ +    int toWrite;		/* I: #bytes to write, +				 * O: #bytes actually written */ +}; +struct ForwardParamSeek { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    int seekMode;		/* I: How to seek */ +    Tcl_WideInt offset;		/* I: Where to seek, +				 * O: New location */ +}; +struct ForwardParamWatch { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    int mask;			/* I: What events to watch for */ +}; +struct ForwardParamBlock { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    int nonblocking;		/* I: What mode to activate */ +}; +struct ForwardParamSetOpt { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    const char *name;		/* Name of option to set */ +    const char *value;		/* Value to set */ +}; +struct ForwardParamGetOpt { +    ForwardParamBase base;	/* "Supertype". MUST COME FIRST. */ +    const char *name;		/* Name of option to get, maybe NULL */ +    Tcl_DString *value;		/* Result */ +};  /* - * General event structure, with reference to - * operation specific data. + * Now join all these together in a single union for convenience.   */ -typedef struct RcForwardingEvent { -  Tcl_Event                  event; /* Basic event data, has to be first item */ -  struct RcForwardingResult* resultPtr; - -  RcOperation               op;    /* Forwarded driver operation */ -  ReflectingChannel*        rcPtr; /* Channel instance */ -  CONST RcForwardParamBase* param; /* Arguments, a RcForwardParamXXX pointer */ -} RcForwardingEvent; +typedef union ForwardParam { +    ForwardParamBase base; +    struct ForwardParamInput input; +    struct ForwardParamOutput output; +    struct ForwardParamSeek seek; +    struct ForwardParamWatch watch; +    struct ForwardParamBlock block; +    struct ForwardParamSetOpt setOpt; +    struct ForwardParamGetOpt getOpt; +} ForwardParam;  /* - * Structure to manage the result of the forwarding.  This is not the - * result of the operation itself, but about the success of the - * forward event itself. The event can be successful, even if the - * operation which was forwarded failed. It is also there to manage - * the synchronization between the involved threads. + * Forward declaration.   */ -typedef struct RcForwardingResult { - -  Tcl_ThreadId  src;    /* Originating thread. */ -  Tcl_ThreadId  dst;    /* Thread the op was forwarded to. */ -  Tcl_Condition done;   /* Condition variable the forwarder blocks on. */ -  int           result; /* TCL_OK or TCL_ERROR */ - -  struct RcForwardingEvent*  evPtr; /* Event the result belongs to. */ - -  struct RcForwardingResult* prevPtr; /* Links into the list of pending */ -  struct RcForwardingResult* nextPtr; /* forwarded results. */ - -} RcForwardingResult; +typedef struct ForwardingResult ForwardingResult;  /* - * List of forwarded operations which have not completed yet, plus the - * mutex to protect the access to this process global list. + * General event structure, with reference to operation specific data.   */ -static RcForwardingResult* forwardList = (RcForwardingResult*) NULL; -TCL_DECLARE_MUTEX (rcForwardMutex) +typedef struct ForwardingEvent { +    Tcl_Event event;		/* Basic event data, has to be first item */ +    ForwardingResult *resultPtr; +    ForwardedOperation op;	/* Forwarded driver operation */ +    ReflectedChannel *rcPtr;	/* Channel instance */ +    ForwardParam *param;	/* Packaged arguments and return values, a +				 * ForwardParam pointer. */ +} ForwardingEvent;  /* - * Function containing the generic code executing a forward, and - * wrapper macros for the actual operations we wish to forward. + * Structure to manage the result of the forwarding. This is not the result of + * the operation itself, but about the success of the forward event itself. + * The event can be successful, even if the operation which was forwarded + * failed. It is also there to manage the synchronization between the involved + * threads.   */ -static void -RcForwardOp _ANSI_ARGS_ ((ReflectingChannel* rcPtr, RcOperation op, -			  Tcl_ThreadId dst, CONST VOID* param)); +struct ForwardingResult { +    Tcl_ThreadId src;		/* Originating thread. */ +    Tcl_ThreadId dst;		/* Thread the op was forwarded to. */ +    Tcl_Interp *dsti;		/* Interpreter in the thread the op was +				 * forwarded to. */ +    /* +     * Note regarding 'dsti' above: Its information is also available via the +     * chain evPtr->rcPtr->interp, however, as can be seen, two more +     * indirections are needed to retrieve it. And the evPtr may be gone, +     * breaking the chain. +     */ +    Tcl_Condition done;		/* Condition variable the forwarder blocks +				 * on. */ +    int result;			/* TCL_OK or TCL_ERROR */ +    ForwardingEvent *evPtr;	/* Event the result belongs to. */ +    ForwardingResult *prevPtr, *nextPtr; +				/* Links into the list of pending forwarded +				 * results. */ +}; -/* - * The event function executed by the thread receiving a forwarding - * event. Executes the appropriate function and collects the result, - * if any. - */ +typedef struct ThreadSpecificData { +    /* +     * Table of all reflected channels owned by this thread. This is the +     * per-thread version of the per-interpreter map. +     */ -static int -RcForwardProc _ANSI_ARGS_ ((Tcl_Event *evPtr, int mask)); +    ReflectedChannelMap *rcmPtr; +} ThreadSpecificData; + +static Tcl_ThreadDataKey dataKey;  /* - * Helpers which intercept when threads are going away, and clean up - * after pending forwarding events. Different actions depending on - * which thread went away, originator (src), or receiver (dst). + * List of forwarded operations which have not completed yet, plus the mutex + * to protect the access to this process global list.   */ -static void -RcSrcExitProc _ANSI_ARGS_ ((ClientData clientData)); +static ForwardingResult *forwardList = NULL; +TCL_DECLARE_MUTEX(rcForwardMutex) -static void -RcDstExitProc _ANSI_ARGS_ ((ClientData clientData)); - -#define RcFreeReceivedError(pb) \ -	if ((pb).vol) {ckfree ((pb).msg);} - -#define RcPassReceivedErrorInterp(i,pb) \ -	if ((i)) {Tcl_SetChannelErrorInterp ((i), Tcl_NewStringObj ((pb).msg,-1));} \ -	RcFreeReceivedError (pb) - -#define RcPassReceivedError(c,pb) \ -	Tcl_SetChannelError ((c), Tcl_NewStringObj ((pb).msg,-1)); \ -	RcFreeReceivedError (pb) - -#define RcForwardSetStaticError(p,emsg) \ -       (p)->code = TCL_ERROR; (p)->vol  = 0; (p)->msg  = (char*) (emsg); +/* + * Function containing the generic code executing a forward, and wrapper + * macros for the actual operations we wish to forward. Uses ForwardProc as + * the event function executed by the thread receiving a forwarding event + * (which executes the appropriate function and collects the result, if any). + * + * The ExitProc ensures that things do not deadlock when the sending thread + * involved in the forwarding exits. It also clean things up so that we don't + * leak resources when threads go away. + */ -#define RcForwardSetDynError(p,emsg) \ -       (p)->code = TCL_ERROR; (p)->vol  = 1; (p)->msg  = (char*) (emsg); +static void		ForwardOpToHandlerThread(ReflectedChannel *rcPtr, +			    ForwardedOperation op, const void *param); +static int		ForwardProc(Tcl_Event *evPtr, int mask); +static void		SrcExitProc(ClientData clientData); -static void -RcForwardSetObjError _ANSI_ARGS_ ((RcForwardParamBase* p, -				   Tcl_Obj*            obj)); +#define FreeReceivedError(p) \ +	if ((p)->base.mustFree) {                               \ +	    ckfree((p)->base.msgStr);                           \ +	} +#define PassReceivedErrorInterp(i,p) \ +	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.msgStr = (char *) (emsg) +#define ForwardSetDynamicError(p,emsg) \ +	(p)->base.code = TCL_ERROR;                             \ +	(p)->base.mustFree = 1;                                 \ +	(p)->base.msgStr = (char *) (emsg) + +static void		ForwardSetObjError(ForwardParam *p, Tcl_Obj *objPtr); + +static ReflectedChannelMap *	GetThreadReflectedChannelMap(void); +static void		DeleteThreadReflectedChannelMap(ClientData clientData);  #endif /* TCL_THREADS */ -#define RcSetChannelErrorStr(c,msg) \ -	Tcl_SetChannelError ((c), Tcl_NewStringObj ((msg),-1)) - -static Tcl_Obj* RcErrorMarshall _ANSI_ARGS_ ((Tcl_Interp *interp)); -static void     RcErrorReturn   _ANSI_ARGS_ ((Tcl_Interp* interp, Tcl_Obj* msg)); +#define SetChannelErrorStr(c,msgStr) \ +	Tcl_SetChannelError((c), Tcl_NewStringObj((msgStr), -1)) - +static Tcl_Obj *	MarshallError(Tcl_Interp *interp); +static void		UnmarshallErrorResult(Tcl_Interp *interp, +			    Tcl_Obj *msgObj);  /*   * Static functions for this file:   */ -static int RcEncodeEventMask _ANSI_ARGS_((Tcl_Interp* interp, -		 CONST char* objName, Tcl_Obj* obj, -		 int* mask)); - -static Tcl_Obj* RcDecodeEventMask _ANSI_ARGS_ ((int mask)); - -static ReflectingChannel* RcNew _ANSI_ARGS_ ((Tcl_Interp* interp, -	     Tcl_Obj* cmdpfxObj, int mode, -	     Tcl_Obj* id)); - -static Tcl_Obj* RcNewHandle _ANSI_ARGS_ ((void)); - -static void RcFree _ANSI_ARGS_ ((ReflectingChannel* rcPtr)); - -static void -RcInvokeTclMethod _ANSI_ARGS_((ReflectingChannel* rcPtr, -       CONST char* method, Tcl_Obj* argone, Tcl_Obj* argtwo, -       int* result, Tcl_Obj** resultObj, int capture)); - -#define NO_CAPTURE (0) -#define DO_CAPTURE (1) +static int		EncodeEventMask(Tcl_Interp *interp, +			    const char *objName, Tcl_Obj *obj, int *mask); +static Tcl_Obj *	DecodeEventMask(int mask); +static ReflectedChannel * NewReflectedChannel(Tcl_Interp *interp, +			    Tcl_Obj *cmdpfxObj, int mode, Tcl_Obj *handleObj); +static Tcl_Obj *	NextHandle(void); +static void		FreeReflectedChannel(ReflectedChannel *rcPtr); +static int		InvokeTclMethod(ReflectedChannel *rcPtr, +			    MethodName method, Tcl_Obj *argOneObj, +			    Tcl_Obj *argTwoObj, Tcl_Obj **resultObjPtr); + +static ReflectedChannelMap *	GetReflectedChannelMap(Tcl_Interp *interp); +static void		DeleteReflectedChannelMap(ClientData clientData, +			    Tcl_Interp *interp); +static int		ErrnoReturn(ReflectedChannel *rcPtr, Tcl_Obj *resObj);  /*   * Global constant strings (messages). ================== @@ -445,17 +444,16 @@ RcInvokeTclMethod _ANSI_ARGS_((ReflectingChannel* rcPtr,   * 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_seek_beforestart = "{Tried to seek before origin}"; - +static const char *msg_read_toomuch = "{read delivered more than requested}"; +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}";  #ifdef TCL_THREADS -static CONST char* msg_send_originlost  = "{Origin thread lost}"; -static CONST char* msg_send_dstlost     = "{Destination thread lost}"; +static const char *msg_send_originlost = "{Channel thread lost}"; +static const char *msg_send_dstlost    = "{Owner lost}";  #endif /* TCL_THREADS */ - +static const char *msg_dstlost    = "-code 1 -level 0 -errorcode NONE -errorinfo {} -errorline 1 {Owner lost}"; +  /*   * Main methods to plug into the 'chan' ensemble'. ==================   */ @@ -465,12 +463,12 @@ static CONST char* msg_send_dstlost     = "{Destination thread lost}";   *   * TclChanCreateObjCmd --   * - *	This procedure is invoked to process the "chan create" Tcl - *	command. See the user documentation for details on what it does. + *	This function is invoked to process the "chan create" Tcl command. + *	See the user documentation for details on what it does.   *   * Results: - *	A standard Tcl result. - *	The handle of the new channel is placed in the interp result. + *	A standard Tcl result. The handle of the new channel is placed in the + *	interp result.   *   * Side effects:   *	Creates a new channel. @@ -479,223 +477,252 @@ static CONST char* msg_send_dstlost     = "{Destination thread lost}";   */  int -TclChanCreateObjCmd (/*ignored*/ clientData, interp, objc, objv) -     ClientData      clientData; -     Tcl_Interp*     interp; -     int             objc; -     Tcl_Obj* CONST* objv; +TclChanCreateObjCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const *objv)  { -    ReflectingChannel* rcPtr;       /* Instance data of the new channel */ -    Tcl_Obj*           rcId;        /* Handle of the new channel */ -    int                mode;        /* R/W mode of new channel. Has to -				     * match abilities of handler commands */ -    Tcl_Obj*           cmdObj;      /* Command prefix, list of words */ -    Tcl_Obj*           cmdNameObj;  /* Command name */ -    Tcl_Channel        chan;        /* Token for the new channel */ -    Tcl_Obj*           modeObj;     /* mode in obj form for method call */ -    int                listc;       /* Result of 'initialize', and of */ -    Tcl_Obj**          listv;       /* its sublist in the 2nd element */ -    int                methIndex;   /* Encoded method name */ -    int                res;         /* Result code for 'initialize' */ -    Tcl_Obj*           resObj;      /* Result data for 'initialize' */ -    int                methods;     /* Bitmask for supported methods. */ -    Channel*           chanPtr;     /* 'chan' resolved to internal struct. */ - -    /* Syntax:   chan create MODE CMDPREFIX +    ReflectedChannel *rcPtr;	/* Instance data of the new channel */ +    Tcl_Obj *rcId;		/* Handle of the new channel */ +    int mode;			/* R/W mode of new channel. Has to match +				 * abilities of handler commands */ +    Tcl_Obj *cmdObj;		/* Command prefix, list of words */ +    Tcl_Obj *cmdNameObj;	/* Command name */ +    Tcl_Channel chan;		/* Token for the new channel */ +    Tcl_Obj *modeObj;		/* mode in obj form for method call */ +    int listc;			/* Result of 'initialize', and of */ +    Tcl_Obj **listv;		/* its sublist in the 2nd element */ +    int methIndex;		/* Encoded method name */ +    int result;			/* Result code for 'initialize' */ +    Tcl_Obj *resObj;		/* Result data for 'initialize' */ +    int methods;		/* Bitmask for supported methods. */ +    Channel *chanPtr;		/* 'chan' resolved to internal struct. */ +    Tcl_Obj *err;		/* Error message */ +    ReflectedChannelMap *rcmPtr; +				/* Map of reflected channels with handlers in +				 * this interp. */ +    Tcl_HashEntry *hPtr;	/* Entry in the above map */ +    int isNew;			/* Placeholder. */ + +    /* +     * Syntax:   chan create MODE CMDPREFIX       *           [0]  [1]    [2]  [3]       *       * Actually: rCreate MODE CMDPREFIX       *           [0]     [1]  [2]       */ -#define MODE (1) -#define CMD  (2) +#define MODE	(1) +#define CMD	(2) -    /* Number of arguments ... */ +    /* +     * Number of arguments... +     */      if (objc != 3) { -        Tcl_WrongNumArgs(interp, 1, objv, "mode cmdprefix"); +	Tcl_WrongNumArgs(interp, 1, objv, "mode cmdprefix");  	return TCL_ERROR;      } -    /* First argument is a list of modes. Allowed entries are "read", -     * "write". Expect at least one list element.  Abbreviations are -     * ok. +    /* +     * First argument is a list of modes. Allowed entries are "read", "write". +     * Expect at least one list element. Abbreviations are ok.       */ -    modeObj = objv [MODE]; -    if (RcEncodeEventMask (interp, "mode", objv [MODE], &mode) != TCL_OK) { -        return TCL_ERROR; +    modeObj = objv[MODE]; +    if (EncodeEventMask(interp, "mode", objv[MODE], &mode) != TCL_OK) { +	return TCL_ERROR;      } -    /* Second argument is command prefix, i.e. list of words, first -     * word is name of handler command, other words are fixed -     * arguments. Run 'initialize' method to get the list of supported -     * methods. Validate this. +    /* +     * Second argument is command prefix, i.e. list of words, first word is +     * name of handler command, other words are fixed arguments. Run the +     * 'initialize' method to get the list of supported methods. Validate +     * this.       */ -    cmdObj = objv [CMD]; +    cmdObj = objv[CMD]; -    /* Basic check that the command prefix truly is a list. */ +    /* +     * Basic check that the command prefix truly is a list. +     */      if (Tcl_ListObjIndex(interp, cmdObj, 0, &cmdNameObj) != TCL_OK) { -        return TCL_ERROR; +	return TCL_ERROR;      } -    /* Now create the channel. +    /* +     * Now create the channel.       */ -    rcId  = RcNewHandle (); -    rcPtr = RcNew (interp, cmdObj, mode, rcId); -    chan  = Tcl_CreateChannel (&tclRChannelType, -			       Tcl_GetString (rcId), -			       rcPtr, mode); -    rcPtr->chan = chan; -    chanPtr = (Channel*) chan; - -    /* Invoke 'initialize' and validate that the handler -     * is present and ok. Squash the channel if not. -     */ +    rcId = NextHandle(); +    rcPtr = NewReflectedChannel(interp, cmdObj, mode, rcId); -    /* Note: The conversion of 'mode' back into a Tcl_Obj ensures that +    /* +     * Invoke 'initialize' and validate that the handler is present and ok. +     * Squash the channel if not. +     * +     * Note: The conversion of 'mode' back into a Tcl_Obj ensures that       * 'initialize' is invoked with canonical mode names, and no -     * abbreviations. Using modeObj directly could feed abbreviations -     * into the handler, and the handler is not specified to handle -     * such. +     * abbreviations. Using modeObj directly could feed abbreviations into the +     * handler, and the handler is not specified to handle such.       */ -    modeObj = RcDecodeEventMask (mode); -    RcInvokeTclMethod (rcPtr, "initialize", modeObj, NULL, -		       &res, &resObj, NO_CAPTURE); -    Tcl_DecrRefCount (modeObj); - -    if (res != TCL_OK) { -        Tcl_Obj* err = Tcl_NewStringObj ("Initialize failure: ",-1); +    modeObj = DecodeEventMask(mode); +    /* assert modeObj.refCount == 1 */ +    result = InvokeTclMethod(rcPtr, METH_INIT, modeObj, NULL, &resObj); +    Tcl_DecrRefCount(modeObj); -	Tcl_AppendObjToObj(err,resObj); -	Tcl_SetObjResult (interp,err); -	Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ +    if (result != TCL_OK) { +	UnmarshallErrorResult(interp, resObj); +	Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */  	goto error;      } -    /* Verify the result. +    /* +     * Verify the result.       * - List, of method names. Convert to mask.       *   Check for non-optionals through the mask.       *   Compare open mode against optional r/w.       */ -    Tcl_AppendResult (interp, "Initialize failure: ", (char*) NULL); - -    if (Tcl_ListObjGetElements (interp, resObj, -			&listc, &listv) != TCL_OK) { -        /* The function above replaces my prefix in case of an error, -	 * so more work for us to get the prefix back into the error -	 * message -	 */ - -        Tcl_Obj* err = Tcl_NewStringObj ("Initialize failure: ",-1); - -	Tcl_AppendObjToObj(err,Tcl_GetObjResult (interp)); -	Tcl_SetObjResult (interp,err); +    if (Tcl_ListObjGetElements(NULL, resObj, &listc, &listv) != TCL_OK) { +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s initialize\" returned non-list: %s", +                Tcl_GetString(cmdObj), Tcl_GetString(resObj))); +	Tcl_DecrRefCount(resObj);  	goto error;      }      methods = 0;      while (listc > 0) { -        if (Tcl_GetIndexFromObj (interp, listv [listc-1], -				 methodNames, "method", TCL_EXACT, &methIndex) != TCL_OK) { -	    Tcl_Obj* err = Tcl_NewStringObj ("Initialize failure: ",-1); - -	    Tcl_AppendObjToObj(err,Tcl_GetObjResult (interp)); -	    Tcl_SetObjResult (interp,err); +	if (Tcl_GetIndexFromObj(interp, listv[listc-1], methodNames, +		"method", TCL_EXACT, &methIndex) != TCL_OK) { +	    TclNewLiteralStringObj(err, "chan handler \""); +	    Tcl_AppendObjToObj(err, cmdObj); +	    Tcl_AppendToObj(err, " initialize\" returned ", -1); +	    Tcl_AppendObjToObj(err, Tcl_GetObjResult(interp)); +	    Tcl_SetObjResult(interp, err); +	    Tcl_DecrRefCount(resObj);  	    goto error;  	} -	methods |= FLAG (methIndex); -	listc --; +	methods |= FLAG(methIndex); +	listc--;      } +    Tcl_DecrRefCount(resObj);      if ((REQUIRED_METHODS & methods) != REQUIRED_METHODS) { -        Tcl_AppendResult (interp, "Not all required methods supported", -			  (char*) NULL); +        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)) { -        Tcl_AppendResult (interp, "Reading not supported, but requested", -			  (char*) NULL); +    if ((mode & TCL_READABLE) && !HAS(methods, METH_READ)) { +        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)) { -        Tcl_AppendResult (interp, "Writing not supported, but requested", -			  (char*) NULL); +    if ((mode & TCL_WRITABLE) && !HAS(methods, METH_WRITE)) { +        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))) { -        Tcl_AppendResult (interp, "'cgetall' not supported, but should be, as 'cget' is", -			  (char*) NULL); +    if (!IMPLIES(HAS(methods, METH_CGET), HAS(methods, METH_CGETALL))) { +        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))) { -        Tcl_AppendResult (interp, "'cget' not supported, but should be, as 'cgetall' is", -			  (char*) NULL); +    if (!IMPLIES(HAS(methods, METH_CGETALL), HAS(methods, METH_CGET))) { +        Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "chan handler \"%s\" supports \"cgetall\" but not \"cget\"", +                Tcl_GetString(cmdObj)));  	goto error;      } -    Tcl_ResetResult (interp); +    Tcl_ResetResult(interp); -    /* Everything is fine now */ +    /* +     * 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) { -        /* Some of the nullable methods are not supported.  We clone -	 * the channel type, null the associated C functions, and use -	 * the result as the actual channel type. +	/* +	 * Some of the nullable methods are not supported. We clone the +	 * channel type, null the associated C functions, and use the result +	 * as the actual channel type.  	 */ -        Tcl_ChannelType* clonePtr = (Tcl_ChannelType*) ckalloc (sizeof (Tcl_ChannelType)); -	if (clonePtr == (Tcl_ChannelType*) NULL) { -	    Tcl_Panic ("Out of memory in Tcl_RcCreate"); -	} +	Tcl_ChannelType *clonePtr = ckalloc(sizeof(Tcl_ChannelType)); -	memcpy (clonePtr, &tclRChannelType, sizeof (Tcl_ChannelType)); +	memcpy(clonePtr, &tclRChannelType, sizeof(Tcl_ChannelType)); -	if (!(methods & FLAG (METH_CONFIGURE))) { -	  clonePtr->setOptionProc = NULL; +	if (!(methods & FLAG(METH_CONFIGURE))) { +	    clonePtr->setOptionProc = NULL;  	} -	if ( -	    !(methods & FLAG (METH_CGET)) && -	    !(methods & FLAG (METH_CGETALL)) -	    ) { +	if (!(methods & FLAG(METH_CGET)) && !(methods & FLAG(METH_CGETALL))) {  	    clonePtr->getOptionProc = NULL;  	} -	if (!(methods & FLAG (METH_BLOCKING))) { +	if (!(methods & FLAG(METH_BLOCKING))) {  	    clonePtr->blockModeProc = NULL;  	} -	if (!(methods & FLAG (METH_SEEK))) { -	    clonePtr->seekProc     = NULL; +	if (!(methods & FLAG(METH_SEEK))) { +	    clonePtr->seekProc = NULL;  	    clonePtr->wideSeekProc = NULL;  	}  	chanPtr->typePtr = clonePtr;      } -    Tcl_RegisterChannel (interp, chan); +    /* +     * Register the channel in the I/O system, and in our our map for 'chan +     * postevent'. +     */ + +    Tcl_RegisterChannel(interp, chan); -    /* Return handle as result of command */ +    rcmPtr = GetReflectedChannelMap(interp); +    hPtr = Tcl_CreateHashEntry(&rcmPtr->map, chanPtr->state->channelName, +	    &isNew); +    if (!isNew && chanPtr != Tcl_GetHashValue(hPtr)) { +	Tcl_Panic("TclChanCreateObjCmd: duplicate channel names"); +    } +    Tcl_SetHashValue(hPtr, chan); +#ifdef TCL_THREADS +    rcmPtr = GetThreadReflectedChannelMap(); +    hPtr = Tcl_CreateHashEntry(&rcmPtr->map, chanPtr->state->channelName, +	    &isNew); +    Tcl_SetHashValue(hPtr, chan); +#endif -    Tcl_SetObjResult (interp, rcId); +    /* +     * Return handle as result of command. +     */ + +    Tcl_SetObjResult(interp, +            Tcl_NewStringObj(chanPtr->state->channelName, -1));      return TCL_OK; - error: -    /* Signal to RcClose to not call 'finalize' */ -    rcPtr->methods = 0; -    Tcl_Close (interp, chan); +  error: +    Tcl_DecrRefCount(rcPtr->name); +    Tcl_DecrRefCount(rcPtr->methods); +    Tcl_DecrRefCount(rcPtr->cmd); +    ckfree((char*) rcPtr);      return TCL_ERROR;  #undef MODE @@ -707,190 +734,306 @@ TclChanCreateObjCmd (/*ignored*/ clientData, interp, objc, objv)   *   * TclChanPostEventObjCmd --   * - *	This procedure is invoked to process the "chan postevent" - *	Tcl command. See the user documentation for details on what it does. + *	This function is invoked to process the "chan postevent" Tcl command. + *	See the user documentation for details on what it does.   *   * Results:   *	A standard Tcl result.   *   * Side effects: - *	Posts events to a reflected channel, invokes event handlers. - *	The latter implies that arbitrary side effects are possible. + *	Posts events to a reflected channel, invokes event handlers. The + *	latter implies that arbitrary side effects are possible.   *   *----------------------------------------------------------------------   */ +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 (/*ignored*/ clientData, interp, objc, objv) -     ClientData      clientData; -     Tcl_Interp*     interp; -     int             objc; -     Tcl_Obj* CONST* objv; +TclChanPostEventObjCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const *objv)  { -    /* Syntax:   chan postevent CHANNEL EVENTSPEC +    /* +     * Ensure -> HANDLER thread +     * +     * Syntax:   chan postevent CHANNEL EVENTSPEC       *           [0]  [1]       [2]     [3]       *       * Actually: rPostevent CHANNEL EVENTSPEC       *           [0]        [1]     [2]       * -     *         where EVENTSPEC = {read write ...} (Abbreviations allowed as well. +     * where EVENTSPEC = {read write ...} (Abbreviations allowed as well).       */ -#define CHAN  (1) -#define EVENT (2) +#define CHAN	(1) +#define EVENT	(2) -    CONST char*        chanId;      /* Tcl level channel handle */ -    Tcl_Channel        chan;        /* Channel associated to the handle */ -    Tcl_ChannelType*   chanTypePtr; /* Its associated driver structure */ -    ReflectingChannel* rcPtr;       /* Associated instance data */ -    int                mode;        /* Dummy, r|w mode of the channel */ -    int                events;      /* Mask of events to post */ +    const char *chanId;		/* Tcl level channel handle */ +    Tcl_Channel chan;		/* Channel associated to the handle */ +    const Tcl_ChannelType *chanTypePtr; +				/* Its associated driver structure */ +    ReflectedChannel *rcPtr;	/* Associated instance data */ +    int events;			/* Mask of events to post */ +    ReflectedChannelMap *rcmPtr;/* Map of reflected channels with handlers in +				 * this interp. */ +    Tcl_HashEntry *hPtr;	/* Entry in the above map */ -    /* Number of arguments ... */ +    /* +     * Number of arguments... +     */      if (objc != 3) { -        Tcl_WrongNumArgs(interp, 1, objv, "channel eventspec"); +	Tcl_WrongNumArgs(interp, 1, objv, "channel eventspec");  	return TCL_ERROR;      } -    /* First argument is a channel, a reflected channel, and the call -     * of this command is done from the interp defining the channel -     * handler cmd. +    /* +     * First argument is a channel, a reflected channel, and the call of this +     * command is done from the interp defining the channel handler cmd.       */ -    chanId = Tcl_GetString (objv [CHAN]); -    chan   = Tcl_GetChannel(interp, chanId, &mode); +    chanId = TclGetString(objv[CHAN]); + +    rcmPtr = GetReflectedChannelMap(interp); +    hPtr = Tcl_FindHashEntry(&rcmPtr->map, chanId); -    if (chan == (Tcl_Channel) NULL) { -        return TCL_ERROR; +    if (hPtr == 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;      } -    chanTypePtr = Tcl_GetChannelType (chan); +    /* +     * Note that the search above subsumes several of the older checks, namely: +     * +     * (1) Does the channel handle refer to a reflected channel ? +     * (2) Is the post event issued from the interpreter holding the handler +     *     of the reflected channel ? +     * +     * A successful search answers yes to both. Because the map holds only +     * handles of reflected channels, and only of such whose handler is +     * defined in this interpreter. +     * +     * We keep the old checks for both, for paranioa, but abort now instead of +     * throwing errors, as failure now means that our internal datastructures +     * have gone seriously haywire. +     */ + +    chan = Tcl_GetHashValue(hPtr); +    chanTypePtr = Tcl_GetChannelType(chan); -    /* We use a function referenced by the channel type as our cookie -     * to detect calls to non-reflecting channels. The channel type -     * itself is not suitable, as it might not be the static -     * definition in this file, but a clone thereof. And while we have -     * reserved the name of the type nothing in the core checks -     * against violation, so someone else might have created a channel -     * type using our name, clashing with ourselves. +    /* +     * We use a function referenced by the channel type as our cookie to +     * detect calls to non-reflecting channels. The channel type itself is not +     * suitable, as it might not be the static definition in this file, but a +     * clone thereof. And while we have reserved the name of the type nothing +     * in the core checks against violation, so someone else might have +     * created a channel type using our name, clashing with ourselves.       */ -    if (chanTypePtr->watchProc != &RcWatch) { -        Tcl_AppendResult(interp, "channel \"", chanId, -			 "\" is not a reflected channel", -			 (char *) NULL); -	return TCL_ERROR; +    if (chanTypePtr->watchProc != &ReflectWatch) { +	Tcl_Panic("TclChanPostEventObjCmd: channel is not a reflected channel");      } -    rcPtr = (ReflectingChannel*) Tcl_GetChannelInstanceData (chan); +    rcPtr = Tcl_GetChannelInstanceData(chan);      if (rcPtr->interp != interp) { -        Tcl_AppendResult(interp, "postevent for channel \"", chanId, -			 "\" called from outside interpreter", -			 (char *) NULL); -	return TCL_ERROR; +	Tcl_Panic("TclChanPostEventObjCmd: postevent accepted for call from outside interpreter");      } -    /* Second argument is a list of events. Allowed entries are -     * "read", "write". Expect at least one list element. -     * Abbreviations are ok. +    /* +     * Second argument is a list of events. Allowed entries are "read", +     * "write". Expect at least one list element. Abbreviations are ok.       */ -    if (RcEncodeEventMask (interp, "event", objv [EVENT], &events) != TCL_OK) { -        return TCL_ERROR; +    if (EncodeEventMask(interp, "event", objv[EVENT], &events) != TCL_OK) { +	return TCL_ERROR;      } -     -    /* Check that the channel is actually interested in the provided -     * events. + +    /* +     * Check that the channel is actually interested in the provided events.       */      if (events & ~rcPtr->interest) { -        Tcl_AppendResult(interp, "tried to post events channel \"", chanId, -			 "\" is not interested in", -			 (char *) NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "tried to post events channel \"%s\" is not interested in", +                chanId));  	return TCL_ERROR;      } -    /* We have the channel and the events to post. +    /* +     * 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. +    /* +     * Squash interp results left by the event script.       */ -    Tcl_ResetResult (interp); +    Tcl_ResetResult(interp);      return TCL_OK;  #undef CHAN  #undef EVENT  } +/* + * Channel error message marshalling utilities. + */ -static Tcl_Obj* -RcErrorMarshall (interp) -     Tcl_Interp *interp; +static Tcl_Obj * +MarshallError( +    Tcl_Interp *interp)  { -    /* Capture the result status of the interpreter into a string. -     * => List of options and values, followed by the error message. -     * The result has refCount 0. +    /* +     * Capture the result status of the interpreter into a string. => List of +     * options and values, followed by the error message. The result has +     * refCount 0.       */ -     -    Tcl_Obj* returnOpt = Tcl_GetReturnOptions (interp, TCL_ERROR); -    /* => returnOpt.refCount == 0. We can append directly. +    Tcl_Obj *returnOpt = Tcl_GetReturnOptions(interp, TCL_ERROR); + +    /* +     * => returnOpt.refCount == 0. We can append directly.       */ -    Tcl_ListObjAppendElement (NULL, returnOpt, Tcl_GetObjResult (interp)); +    Tcl_ListObjAppendElement(NULL, returnOpt, Tcl_GetObjResult(interp));      return returnOpt;  }  static void -RcErrorReturn (interp, msg) -     Tcl_Interp *interp; -     Tcl_Obj    *msg; +UnmarshallErrorResult( +    Tcl_Interp *interp, +    Tcl_Obj *msgObj)  { -    int       res; -    int       lc; -    Tcl_Obj** lv; -    int       explicitResult; -    int       numOptions; +    int lc; +    Tcl_Obj **lv; +    int explicitResult; +    int numOptions; -    /* Process the caught message. +    /* +     * Process the caught message.       *       * Syntax = (option value)... ?message?       * -     * Bad syntax causes a panic. Because the other side uses +     * Bad syntax causes a panic. This is OK because the other side uses       * Tcl_GetReturnOptions and list construction functions to marshall the -     * information. +     * information; if we panic here, something has gone badly wrong already.       */ -    res = Tcl_ListObjGetElements (interp, msg, &lc, &lv); -    if (res != TCL_OK) { -	Tcl_Panic ("TclChanCaughtErrorBypass: Bad syntax of caught result"); +    if (Tcl_ListObjGetElements(interp, msgObj, &lc, &lv) != TCL_OK) { +	Tcl_Panic("TclChanCaughtErrorBypass: Bad syntax of caught result"); +    } +    if (interp == NULL) { +	return;      } -    explicitResult = (1 == (lc % 2)); -    numOptions     = lc - explicitResult; +    explicitResult = lc & 1;		/* Odd number of values? */ +    numOptions = lc - explicitResult;      if (explicitResult) { -	Tcl_SetObjResult (interp, lv [lc-1]); +	Tcl_SetObjResult(interp, lv[lc-1]);      } -    (void) Tcl_SetReturnOptions(interp, Tcl_NewListObj (numOptions, lv)); +    (void) Tcl_SetReturnOptions(interp, Tcl_NewListObj(numOptions, lv)); +    ((Interp *) interp)->flags &= ~ERR_ALREADY_LOGGED;  }  int -TclChanCaughtErrorBypass (interp, chan) -     Tcl_Interp *interp; -     Tcl_Channel chan; +TclChanCaughtErrorBypass( +    Tcl_Interp *interp, +    Tcl_Channel chan)  { -    Tcl_Obj* msgc = NULL; -    Tcl_Obj* msgi = NULL; -    Tcl_Obj* msg  = NULL; +    Tcl_Obj *chanMsgObj = NULL; +    Tcl_Obj *interpMsgObj = NULL; +    Tcl_Obj *msgObj = NULL; -    /* Get a bypassed error message from channel and/or interpreter, save the +    /* +     * Get a bypassed error message from channel and/or interpreter, save the       * reference, then kill the returned objects, if there were any. If there       * are messages in both the channel has preference.       */ @@ -900,37 +1043,39 @@ TclChanCaughtErrorBypass (interp, chan)      }      if (chan != NULL) { -	Tcl_GetChannelError       (chan,   &msgc); +	Tcl_GetChannelError(chan, &chanMsgObj);      }      if (interp != NULL) { -	Tcl_GetChannelErrorInterp (interp, &msgi); +	Tcl_GetChannelErrorInterp(interp, &interpMsgObj);      } -    if (msgc != NULL) { -	msg = msgc; -	Tcl_IncrRefCount (msg); -    } else if (msgi != NULL) { -	msg = msgi; -	Tcl_IncrRefCount (msg); +    if (chanMsgObj != NULL) { +	msgObj = chanMsgObj; +    } else if (interpMsgObj != NULL) { +	msgObj = interpMsgObj; +    } +    if (msgObj != NULL) { +	Tcl_IncrRefCount(msgObj);      } -    if (msgc != NULL) { -	Tcl_DecrRefCount (msgc); +    if (chanMsgObj != NULL) { +	Tcl_DecrRefCount(chanMsgObj);      } -    if (msgi != NULL) { -	Tcl_DecrRefCount (msgi); +    if (interpMsgObj != NULL) { +	Tcl_DecrRefCount(interpMsgObj);      } -    /* No message returned, nothing caught. +    /* +     * No message returned, nothing caught.       */ -    if (msg == NULL) { +    if (msgObj == NULL) {  	return 0;      } -    RcErrorReturn (interp, msg); +    UnmarshallErrorResult(interp, msgObj); -    Tcl_DecrRefCount (msg); +    Tcl_DecrRefCount(msgObj);      return 1;  } @@ -941,10 +1086,10 @@ TclChanCaughtErrorBypass (interp, chan)  /*   *----------------------------------------------------------------------   * - * RcClose -- + * ReflectClose --   * - *	This function is invoked when the channel is closed, to delete - *	the driver specific instance data. + *	This function is invoked when the channel is closed, to delete the + *	driver specific instance data.   *   * Results:   *	A posix error. @@ -956,110 +1101,131 @@ TclChanCaughtErrorBypass (interp, chan)   */  static int -RcClose (clientData, interp) -     ClientData  clientData; -     Tcl_Interp* interp; +ReflectClose( +    ClientData clientData, +    Tcl_Interp *interp)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    int                res;         /* Result code for 'close' */ -    Tcl_Obj*           resObj;      /* Result data for 'close' */ - -    if (interp == (Tcl_Interp*) NULL) { -        /* This call comes from TclFinalizeIOSystem. There are no -	 * interpreters, and therefore we cannot call upon the handler -	 * command anymore. Threading is irrelevant as well.  We -	 * simply clean up all our C level data structures and leave -	 * the Tcl level to the other finalization functions. +    ReflectedChannel *rcPtr = clientData; +    int result;			/* Result code for 'close' */ +    Tcl_Obj *resObj;		/* Result data for 'close' */ +    ReflectedChannelMap *rcmPtr;/* Map of reflected channels with handlers in +				 * this interp */ +    Tcl_HashEntry *hPtr;	/* Entry in the above map */ + +    if (TclInThreadExit()) { +	/* +	 * This call comes from TclFinalizeIOSystem. There are no +	 * interpreters, and therefore we cannot call upon the handler command +	 * anymore. Threading is irrelevant as well. We simply clean up all +	 * our C level data structures and leave the Tcl level to the other +	 * finalization functions.  	 */ -      /* THREADED => Forward this to the origin thread */ -      /* Note: Have a thread delete handler for the origin -       * thread. Use this to clean up the structure! -       */ +	/* +	 * THREADED => Forward this to the origin thread +	 * +	 * Note: DeleteThreadReflectedChannelMap() is the thread exit handler +	 * for the origin thread. Use this to clean up the structure? Except +	 * if lost? +	 */  #ifdef TCL_THREADS -        /* Are we in the correct thread ? -	 */ +	if (rcPtr->thread != Tcl_GetCurrentThread()) { +	    ForwardParam p; -        if (rcPtr->thread != Tcl_GetCurrentThread ()) { -	    RcForwardParamClose p; +	    ForwardOpToHandlerThread(rcPtr, ForwardedClose, &p); +	    result = p.base.code; -	    RcForwardOp (rcPtr, RcOpClose, rcPtr->thread, &p); -	    res = p.b.code; +            /* +             * Now squash the pending reflection events for this channel. +             */ -	    /* RcFree is done in the forwarded operation!, -	     * in the other thread. rcPtr here is gone! -	     */ +            Tcl_DeleteEvents(ReflectEventDelete, rcPtr); -	    if (res != TCL_OK) { -	        RcFreeReceivedError (p.b); +	    if (result != TCL_OK) { +		FreeReceivedError(&p);  	    } -	} else { -#endif -	    RcFree (rcPtr); -#ifdef TCL_THREADS +	    return EOK;  	}  #endif + +        Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel);  	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. +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->methods == 0) { -        RcFree (rcPtr); -        return EOK; -    } else {  #ifdef TCL_THREADS -        /* Are we in the correct thread ? -	 */ +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -        if (rcPtr->thread != Tcl_GetCurrentThread ()) { -	    RcForwardParamClose p; +	ForwardOpToHandlerThread(rcPtr, ForwardedClose, &p); +	result = p.base.code; -	    RcForwardOp (rcPtr, RcOpClose, rcPtr->thread, &p); -	    res = p.b.code; +        /* +         * Now squash the pending reflection events for this channel. +         */ -	    /* RcFree is done in the forwarded operation!, -	     * in the other thread. rcPtr here is gone! -	     */ +        Tcl_DeleteEvents(ReflectEventDelete, rcPtr); -	    if (res != TCL_OK) { -	        RcPassReceivedErrorInterp (interp, p.b); -	    } -	} else { +	if (result != TCL_OK) { +	    PassReceivedErrorInterp(interp, &p); +	} +    } else {  #endif -	    RcInvokeTclMethod (rcPtr, "finalize", NULL, NULL, -			       &res, &resObj, DO_CAPTURE); +	result = InvokeTclMethod(rcPtr, METH_FINAL, NULL, NULL, &resObj); +	if ((result != TCL_OK) && (interp != NULL)) { +	    Tcl_SetChannelErrorInterp(interp, resObj); +	} -	    if ((res != TCL_OK) && (interp != NULL)) { -	        Tcl_SetChannelErrorInterp (interp, resObj); -	    } +	Tcl_DecrRefCount(resObj);	/* Remove reference we held from the +					 * invoke */ + +	/* +	 * Remove the channel from the map before releasing the memory, to +	 * prevent future accesses (like by 'postevent') from finding and +	 * dereferencing a dangling pointer. +	 * +	 * NOTE: The channel may not be in the map. This is ok, that happens +	 * when the channel was created in a different interpreter and/or +	 * thread and then was moved here. +	 * +	 * NOTE: The channel may have been removed from the map already via +	 * the per-interp DeleteReflectedChannelMap exit-handler. +	 */ -	    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ +	if (!rcPtr->dead) { +	    rcmPtr = GetReflectedChannelMap(rcPtr->interp); +	    hPtr = Tcl_FindHashEntry(&rcmPtr->map, +		    Tcl_GetChannelName(rcPtr->chan)); +	    if (hPtr) { +		Tcl_DeleteHashEntry(hPtr); +	    } +	}  #ifdef TCL_THREADS -	    RcFree (rcPtr); +	rcmPtr = GetThreadReflectedChannelMap(); +	hPtr = Tcl_FindHashEntry(&rcmPtr->map, +		Tcl_GetChannelName(rcPtr->chan)); +	if (hPtr) { +	    Tcl_DeleteHashEntry(hPtr);  	}  #endif -	return (res == TCL_OK) ? EOK : EINVAL; -    } -    Tcl_Panic ("Should not be reached"); -    return EINVAL; +        Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel); +#ifdef TCL_THREADS +    } +#endif +    return (result == TCL_OK) ? EOK : EINVAL;  }  /*   *----------------------------------------------------------------------   * - * RcInput -- + * ReflectInput --   * - *	This function is invoked when more data is requested from the - *	channel. + *	This function is invoked when more data is requested from the channel.   *   * Results:   *	The number of bytes read. @@ -1071,99 +1237,99 @@ RcClose (clientData, interp)   */  static int -RcInput (clientData, buf, toRead, errorCodePtr) -     ClientData clientData; -     char* buf; -     int toRead; -     int* errorCodePtr; +ReflectInput( +    ClientData clientData, +    char *buf, +    int toRead, +    int *errorCodePtr)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*           toReadObj; -    int                bytec;       /* Number of returned bytes */ -    unsigned char*     bytev;       /* Array of returned bytes */ -    int                res;         /* Result code for 'read' */ -    Tcl_Obj*           resObj;      /* Result data for 'read' */ +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *toReadObj; +    int bytec;			/* Number of returned bytes */ +    unsigned char *bytev;	/* Array of returned bytes */ +    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. +    /* +     * Are we in the correct thread?       */ -    if (!(rcPtr->methods & FLAG (METH_READ))) { -        RcSetChannelErrorStr (rcPtr->chan, msg_read_unsup); -	*errorCodePtr = EINVAL; -	return -1; -    } -  #ifdef TCL_THREADS -    /* Are we in the correct thread ? -     */ +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        RcForwardParamInput p; +	p.input.buf = buf; +	p.input.toRead = toRead; -	p.buf    = buf; -	p.toRead = toRead; +	ForwardOpToHandlerThread(rcPtr, ForwardedInput, &p); -	RcForwardOp (rcPtr, RcOpInput, rcPtr->thread, &p); - -	if (p.b.code != TCL_OK) { -	    RcPassReceivedError (rcPtr->chan, p.b); -	    *errorCodePtr = EINVAL; +	if (p.base.code != TCL_OK) { +	    if (p.base.code < 0) { +		/* No error message, this is an errno signal. */ +		*errorCodePtr = -p.base.code; +	    } else { +		PassReceivedError(rcPtr->chan, &p); +		*errorCodePtr = EINVAL; +	    } +	    p.input.toRead = -1;  	} else {  	    *errorCodePtr = EOK;  	} -	return p.toRead; +	return p.input.toRead;      }  #endif -    /* -------- */ - -    /* ASSERT: rcPtr->method & FLAG (METH_READ) */ +    /* ASSERT: rcPtr->method & FLAG(METH_READ) */      /* ASSERT: rcPtr->mode & TCL_READABLE */ +    Tcl_Preserve(rcPtr); +      toReadObj = Tcl_NewIntObj(toRead); -    if (toReadObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcInput"); -    } +    Tcl_IncrRefCount(toReadObj); -    RcInvokeTclMethod (rcPtr, "read", toReadObj, NULL, -		       &res, &resObj, DO_CAPTURE); +    if (InvokeTclMethod(rcPtr, METH_READ, toReadObj, NULL, &resObj)!=TCL_OK) { +	int code = ErrnoReturn(rcPtr, resObj); -    if (res != TCL_OK) { -	Tcl_SetChannelError (rcPtr->chan, resObj); -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	*errorCodePtr = EINVAL; -	return -1; +	if (code < 0) { +	    *errorCodePtr = -code; +            goto error; +	} + +	Tcl_SetChannelError(rcPtr->chan, resObj); +        goto invalid;      }      bytev = Tcl_GetByteArrayFromObj(resObj, &bytec);      if (toRead < bytec) { -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	RcSetChannelErrorStr (rcPtr->chan, msg_read_toomuch); -	*errorCodePtr = EINVAL; -	return -1; +	SetChannelErrorStr(rcPtr->chan, msg_read_toomuch); +        goto invalid;      }      *errorCodePtr = EOK;      if (bytec > 0) { -        memcpy (buf, bytev, bytec); +	memcpy(buf, bytev, (size_t) bytec);      } -    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ + 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;  }  /*   *----------------------------------------------------------------------   * - * RcOutput -- + * ReflectOutput --   * - *	This function is invoked when data is writen to the - *	channel. + *	This function is invoked when data is writen to the channel.   *   * Results:   *	The number of bytes actually written. @@ -1175,105 +1341,111 @@ RcInput (clientData, buf, toRead, errorCodePtr)   */  static int -RcOutput (clientData, buf, toWrite, errorCodePtr) -     ClientData clientData; -     CONST char* buf; -     int toWrite; -     int* errorCodePtr; +ReflectOutput( +    ClientData clientData, +    const char *buf, +    int toWrite, +    int *errorCodePtr)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*       bufObj; -    int            res;         /* Result code for 'write' */ -    Tcl_Obj*       resObj;      /* Result data for 'write' */ -    int            written; +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *bufObj; +    Tcl_Obj *resObj;		/* Result data for 'write' */ +    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. +    /* +     * Are we in the correct thread?       */ -    if (!(rcPtr->methods & FLAG (METH_WRITE))) { -        RcSetChannelErrorStr (rcPtr->chan, msg_write_unsup); -        *errorCodePtr = EINVAL; -	return -1; -    } -  #ifdef TCL_THREADS -    /* Are we in the correct thread ? -     */ - -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -      RcForwardParamOutput p; - -      p.buf     = buf; -      p.toWrite = toWrite; - -      RcForwardOp (rcPtr, RcOpOutput, rcPtr->thread, &p); - -      if (p.b.code != TCL_OK) { -	  RcPassReceivedError (rcPtr->chan, p.b); -	  *errorCodePtr = EINVAL; -      } else { -	  *errorCodePtr = EOK; -      } +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; + +	p.output.buf = buf; +	p.output.toWrite = toWrite; + +	ForwardOpToHandlerThread(rcPtr, ForwardedOutput, &p); + +	if (p.base.code != TCL_OK) { +	    if (p.base.code < 0) { +		/* No error message, this is an errno signal. */ +		*errorCodePtr = -p.base.code; +	    } else { +                PassReceivedError(rcPtr->chan, &p); +                *errorCodePtr = EINVAL; +            } +	    p.output.toWrite = -1; +	} else { +	    *errorCodePtr = EOK; +	} -      return p.toWrite; +	return p.output.toWrite;      }  #endif -    /* -------- */ - -    /* ASSERT: rcPtr->method & FLAG (METH_WRITE) */ +    /* ASSERT: rcPtr->method & FLAG(METH_WRITE) */      /* ASSERT: rcPtr->mode & TCL_WRITABLE */ -     -    bufObj = Tcl_NewByteArrayObj((unsigned char*) buf, toWrite); -    if (bufObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcOutput"); -    } -    RcInvokeTclMethod (rcPtr, "write", bufObj, NULL, -		       &res, &resObj, DO_CAPTURE); +    Tcl_Preserve(rcPtr); + +    bufObj = Tcl_NewByteArrayObj((unsigned char *) buf, toWrite); +    Tcl_IncrRefCount(bufObj); + +    if (InvokeTclMethod(rcPtr, METH_WRITE, bufObj, NULL, &resObj) != TCL_OK) { +	int code = ErrnoReturn(rcPtr, resObj); + +	if (code < 0) { +	    *errorCodePtr = -code; +            goto error; +	} -    if (res != TCL_OK) { -	Tcl_SetChannelError (rcPtr->chan, resObj); -	Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	*errorCodePtr = EINVAL; -	return -1; +	Tcl_SetChannelError(rcPtr->chan, resObj); +        goto invalid;      } -    res = Tcl_GetIntFromObj (rcPtr->interp, resObj, &written); -    if (res != TCL_OK) { -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	Tcl_SetChannelError (rcPtr->chan, RcErrorMarshall (rcPtr->interp)); -	*errorCodePtr = EINVAL; -	return -1; +    if (Tcl_GetIntFromObj(rcPtr->interp, resObj, &written) != TCL_OK) { +	Tcl_SetChannelError(rcPtr->chan, MarshallError(rcPtr->interp)); +        goto invalid;      } -    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ +    if ((written == 0) && (toWrite > 0)) { +	/* +	 * The handler claims to have written nothing of what it was +	 * given. That is bad. +	 */ -    if ((written == 0) || (toWrite < written)) { -        /* The handler claims to have written more than it was given. -	 * That is bad. Note that the I/O core would crash if we were -	 * to return this information, trying to write -nnn bytes in -	 * the next iteration. +	SetChannelErrorStr(rcPtr->chan, msg_write_nothing); +        goto invalid; +    } +    if (toWrite < written) { +	/* +	 * The handler claims to have written more than it was given. That is +	 * bad. Note that the I/O core would crash if we were to return this +	 * information, trying to write -nnn bytes in the next iteration.  	 */ -        RcSetChannelErrorStr (rcPtr->chan, msg_write_toomuch); -	*errorCodePtr = EINVAL; -	return -1; +	SetChannelErrorStr(rcPtr->chan, msg_write_toomuch); +        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;  }  /*   *----------------------------------------------------------------------   * - * RcSeekWide / RcSeek -- + * ReflectSeekWide / ReflectSeek --   * - *	This function is invoked when the user wishes to seek on - *	the channel. + *	This function is invoked when the user wishes to seek on the channel.   *   * Results:   *	The new location of the access point. @@ -1285,115 +1457,106 @@ RcOutput (clientData, buf, toWrite, errorCodePtr)   */  static Tcl_WideInt -RcSeekWide (clientData, offset, seekMode, errorCodePtr) -     ClientData  clientData; -     Tcl_WideInt offset; -     int         seekMode; -     int*        errorCodePtr; +ReflectSeekWide( +    ClientData clientData, +    Tcl_WideInt offset, +    int seekMode, +    int *errorCodePtr)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*       offObj; -    Tcl_Obj*       baseObj; -    int            res;         /* Result code for 'seek' */ -    Tcl_Obj*       resObj;      /* Result data for 'seek' */ -    Tcl_WideInt    newLoc; +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *offObj, *baseObj; +    Tcl_Obj *resObj;		/* Result for 'seek' */ +    Tcl_WideInt newLoc; -#ifdef TCL_THREADS -    /* Are we in the correct thread ? +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        RcForwardParamSeek p; +#ifdef TCL_THREADS +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -	p.seekMode = seekMode; -	p.offset   = offset; +	p.seek.seekMode = seekMode; +	p.seek.offset = offset; -	RcForwardOp (rcPtr, RcOpSeek, rcPtr->thread, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedSeek, &p); -	if (p.b.code != TCL_OK) { -	    RcPassReceivedError (rcPtr->chan, p.b); +	if (p.base.code != TCL_OK) { +	    PassReceivedError(rcPtr->chan, &p);  	    *errorCodePtr = EINVAL; +	    p.seek.offset = -1;  	} else {  	    *errorCodePtr = EOK;  	} -	return p.offset; +	return p.seek.offset;      }  #endif -    /* -------- */ - -    /* ASSERT: rcPtr->method & FLAG (METH_SEEK) */ - -    offObj = Tcl_NewWideIntObj(offset); -    if (offObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcSeekWide"); -    } - -    baseObj = Tcl_NewStringObj((seekMode == SEEK_SET) ? -			       "start" : -			       ((seekMode == SEEK_CUR) ? -				"current" : -				"end"), -1); +    /* ASSERT: rcPtr->method & FLAG(METH_SEEK) */ -    if (baseObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcSeekWide"); -    } +    Tcl_Preserve(rcPtr); -    RcInvokeTclMethod (rcPtr, "seek", offObj, baseObj, -		       &res, &resObj, DO_CAPTURE); +    offObj  = Tcl_NewWideIntObj(offset); +    baseObj = Tcl_NewStringObj( +            (seekMode == SEEK_SET) ? "start" : +            (seekMode == SEEK_CUR) ? "current" : "end", -1); +    Tcl_IncrRefCount(offObj); +    Tcl_IncrRefCount(baseObj); -    if (res != TCL_OK) { -	Tcl_SetChannelError (rcPtr->chan, resObj); -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	*errorCodePtr = EINVAL; -	return -1; +    if (InvokeTclMethod(rcPtr, METH_SEEK, offObj, baseObj, &resObj)!=TCL_OK) { +	Tcl_SetChannelError(rcPtr->chan, resObj); +        goto invalid;      } -    res = Tcl_GetWideIntFromObj (rcPtr->interp, resObj, &newLoc); -    if (res != TCL_OK) { -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	Tcl_SetChannelError (rcPtr->chan, RcErrorMarshall (rcPtr->interp)); -	*errorCodePtr = EINVAL; -	return -1; +    if (Tcl_GetWideIntFromObj(rcPtr->interp, resObj, &newLoc) != TCL_OK) { +	Tcl_SetChannelError(rcPtr->chan, MarshallError(rcPtr->interp)); +        goto invalid;      } -    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ - -    if (newLoc < Tcl_LongAsWide (0)) { -        RcSetChannelErrorStr (rcPtr->chan, msg_seek_beforestart); -        *errorCodePtr = EINVAL; -	return -1; +    if (newLoc < Tcl_LongAsWide(0)) { +	SetChannelErrorStr(rcPtr->chan, msg_seek_beforestart); +        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 -RcSeek (clientData, offset, seekMode, errorCodePtr) -     ClientData  clientData; -     long        offset; -     int         seekMode; -     int*        errorCodePtr; +ReflectSeek( +    ClientData clientData, +    long offset, +    int seekMode, +    int *errorCodePtr)  { -  /* This function can be invoked from a transformation which is based -   * on standard seeking, i.e. non-wide. Because o this we have to -   * implement it, a dummy is not enough. We simply delegate the call -   * to the wide routine. -   */ - -  return (int) RcSeekWide (clientData, Tcl_LongAsWide (offset), -			   seekMode, errorCodePtr); +    /* +     * This function can be invoked from a transformation which is based on +     * standard seeking, i.e. non-wide. Because of this we have to implement +     * it, a dummy is not enough. We simply delegate the call to the wide +     * routine. +     */ + +    return (int) ReflectSeekWide(clientData, Tcl_LongAsWide(offset), seekMode, +	    errorCodePtr);  }  /*   *----------------------------------------------------------------------   * - * RcWatch -- + * ReflectWatch --   * - *	This function is invoked to tell the channel what events - *	the I/O system is interested in. + *	This function is invoked to tell the channel what events the I/O + *	system is interested in.   *   * Results:   *	None. @@ -1405,62 +1568,68 @@ RcSeek (clientData, offset, seekMode, errorCodePtr)   */  static void -RcWatch (clientData, mask) -     ClientData clientData; -     int mask; +ReflectWatch( +    ClientData clientData, +    int mask)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj* maskObj; +    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. Analoguous for read events. +    /* +     * We restrict the interest to what the channel can support. IOW there +     * will never be write events for a channel which is not writable. +     * Analoguously for read events and non-readable channels.       */ -    mask = mask & rcPtr->mode;  +    mask &= rcPtr->mode;      if (mask == rcPtr->interest) { -        /* Same old, same old, why should we do something ? */ -      return; +	/* +	 * Same old, same old, why should we do something? +	 */ + +	return;      }      rcPtr->interest = mask; -#ifdef TCL_THREADS -    /* Are we in the correct thread ? +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        RcForwardParamWatch p; - -	p.mask = mask; +#ifdef TCL_THREADS +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -	RcForwardOp (rcPtr, RcOpWatch, rcPtr->thread, &p); +	p.watch.mask = mask; +	ForwardOpToHandlerThread(rcPtr, ForwardedWatch, &p); -	/* Any failure from the forward is ignored. We have no place to -	 * put this. +	/* +	 * Any failure from the forward is ignored. We have no place to put +	 * this.  	 */ +  	return;      }  #endif -    /* -------- */ +    Tcl_Preserve(rcPtr); + +    maskObj = DecodeEventMask(mask); +    /* assert maskObj.refCount == 1 */ +    (void) InvokeTclMethod(rcPtr, METH_WATCH, maskObj, NULL, NULL); +    Tcl_DecrRefCount(maskObj); -    maskObj = RcDecodeEventMask (mask); -    RcInvokeTclMethod (rcPtr, "watch", maskObj, NULL, -		       NULL, NULL, NO_CAPTURE); -    Tcl_DecrRefCount (maskObj); +    Tcl_Release(rcPtr);  }  /*   *----------------------------------------------------------------------   * - * RcBlock -- + * ReflectBlock --   * - *	This function is invoked to tell the channel which blocking - *	behaviour is required of it. + *	This function is invoked to tell the channel which blocking behaviour + *	is required of it.   *   * Results:   *	A posix error number. @@ -1472,60 +1641,97 @@ RcWatch (clientData, mask)   */  static int -RcBlock (clientData, nonblocking) -     ClientData clientData; -     int nonblocking; +ReflectBlock( +    ClientData clientData, +    int nonblocking)  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*           blockObj; -    int                res;         /* Result code for 'blocking' */ -    Tcl_Obj*           resObj;      /* Result data for 'blocking' */ +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *blockObj; +    int errorNum;		/* EINVAL or EOK (success). */ +    Tcl_Obj *resObj;		/* Result data for 'blocking' */ -#ifdef TCL_THREADS -    /* Are we in the correct thread ? +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        RcForwardParamBlock p; +#ifdef TCL_THREADS +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -	p.nonblocking = nonblocking; +	p.block.nonblocking = nonblocking; -	RcForwardOp (rcPtr, RcOpBlock, rcPtr->thread, &p); +	ForwardOpToHandlerThread(rcPtr, ForwardedBlock, &p); -	if (p.b.code != TCL_OK) { -	    RcPassReceivedError (rcPtr->chan, p.b); +	if (p.base.code != TCL_OK) { +	    PassReceivedError(rcPtr->chan, &p);  	    return EINVAL; -	} else { -	    return EOK;  	} + +	return EOK;      }  #endif -    /* -------- */ -      blockObj = Tcl_NewBooleanObj(!nonblocking); -    if (blockObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcBlock"); -    } +    Tcl_IncrRefCount(blockObj); -    RcInvokeTclMethod (rcPtr, "blocking", blockObj, NULL, -		       &res, &resObj, DO_CAPTURE); +    Tcl_Preserve(rcPtr); -    if (res != TCL_OK) { -	Tcl_SetChannelError (rcPtr->chan, resObj); -	res = EINVAL; +    if (InvokeTclMethod(rcPtr,METH_BLOCKING,blockObj,NULL,&resObj)!=TCL_OK) { +	Tcl_SetChannelError(rcPtr->chan, resObj); +	errorNum = EINVAL;      } else { -        res = EOK; +	errorNum = EOK;      } -    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -    return res; +    Tcl_DecrRefCount(blockObj); +    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ + +    Tcl_Release(rcPtr); +    return errorNum;  } +#ifdef TCL_THREADS  /*   *----------------------------------------------------------------------   * - * RcSetOption -- + * 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 +/* + *---------------------------------------------------------------------- + * + * ReflectSetOption --   *   *	This function is invoked to configure a channel option.   * @@ -1539,70 +1745,65 @@ RcBlock (clientData, nonblocking)   */  static int -RcSetOption (clientData, interp, optionName, newValue) -     ClientData   clientData;	/* Channel to query */ -     Tcl_Interp   *interp;	/* Interpreter to leave error messages in */ -     CONST char *optionName;	/* Name of requested option */ -     CONST char *newValue;	/* The new value */ +ReflectSetOption( +    ClientData clientData,	/* Channel to query */ +    Tcl_Interp *interp,		/* Interpreter to leave error messages in */ +    const char *optionName,	/* Name of requested option */ +    const char *newValue)	/* The new value */  { -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*           optionObj; -    Tcl_Obj*           valueObj; -    int                res;         /* Result code for 'configure' */ -    Tcl_Obj*           resObj;      /* Result data for 'configure' */ +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *optionObj, *valueObj; +    int result;			/* Result code for 'configure' */ +    Tcl_Obj *resObj;		/* Result data for 'configure' */ -#ifdef TCL_THREADS -    /* Are we in the correct thread ? +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        RcForwardParamSetOpt p; - -	p.name  = optionName; -	p.value = newValue; +#ifdef TCL_THREADS +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	ForwardParam p; -	RcForwardOp (rcPtr, RcOpSetOpt, rcPtr->thread, &p); +	p.setOpt.name = optionName; +	p.setOpt.value = newValue; -	if (p.b.code != TCL_OK) { -	    Tcl_Obj* err = Tcl_NewStringObj (p.b.msg, -1); +	ForwardOpToHandlerThread(rcPtr, ForwardedSetOpt, &p); -	    RcErrorReturn (interp, err); +	if (p.base.code != TCL_OK) { +	    Tcl_Obj *err = Tcl_NewStringObj(p.base.msgStr, -1); -	    Tcl_DecrRefCount (err); -	    if (p.b.vol) {ckfree (p.b.msg);} +	    UnmarshallErrorResult(interp, err); +	    Tcl_DecrRefCount(err); +	    FreeReceivedError(&p);  	} -	return p.b.code; +	return p.base.code;      }  #endif +    Tcl_Preserve(rcPtr); -  /* -------- */ - -  optionObj = Tcl_NewStringObj(optionName,-1); -  if (optionObj == (Tcl_Obj*) NULL) { -    Tcl_Panic ("Out of memory in RcSetOption"); -  } +    optionObj = Tcl_NewStringObj(optionName, -1); +    valueObj = Tcl_NewStringObj(newValue, -1); -  valueObj = Tcl_NewStringObj(newValue,-1); -  if (valueObj == (Tcl_Obj*) NULL) { -    Tcl_Panic ("Out of memory in RcSetOption"); -  } +    Tcl_IncrRefCount(optionObj); +    Tcl_IncrRefCount(valueObj); -  RcInvokeTclMethod (rcPtr, "configure", optionObj, valueObj, -		     &res, &resObj, DO_CAPTURE); - -  if (res != TCL_OK) { -    RcErrorReturn (interp, resObj); -  } +    result = InvokeTclMethod(rcPtr, METH_CONFIGURE,optionObj,valueObj, &resObj); +    if (result != TCL_OK) { +	UnmarshallErrorResult(interp, resObj); +    } -  Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -  return res; +    Tcl_DecrRefCount(optionObj); +    Tcl_DecrRefCount(valueObj); +    Tcl_DecrRefCount(resObj);		/* Remove reference held from invoke */ +    Tcl_Release(rcPtr); +    return result;  }  /*   *----------------------------------------------------------------------   * - * RcGetOption -- + * ReflectGetOption --   *   *	This function is invoked to retrieve all or a channel option.   * @@ -1616,135 +1817,139 @@ RcSetOption (clientData, interp, optionName, newValue)   */  static int -RcGetOption (clientData, interp, optionName, dsPtr) -     ClientData   clientData;	/* Channel to query */ -     Tcl_Interp*  interp;	/* Interpreter to leave error messages in */ -     CONST char* optionName;	/* Name of reuqested option */ -     Tcl_DString* dsPtr;	/* String to place the result into */ +ReflectGetOption( +    ClientData clientData,	/* Channel to query */ +    Tcl_Interp *interp,		/* Interpreter to leave error messages in */ +    const char *optionName,	/* Name of reuqested option */ +    Tcl_DString *dsPtr)		/* String to place the result into */  { -    /* This code is special. It has regular passing of Tcl result, and -     * errors. The bypass functions are not required. +    /* +     * This code is special. It has regular passing of Tcl result, and errors. +     * The bypass functions are not required.       */ -    ReflectingChannel* rcPtr = (ReflectingChannel*) clientData; -    Tcl_Obj*           optionObj; -    int                res;         /* Result code for 'configure' */ -    Tcl_Obj*           resObj;      /* Result data for 'configure' */ -    int                listc; -    Tcl_Obj**          listv; -    const char*        method; +    ReflectedChannel *rcPtr = clientData; +    Tcl_Obj *optionObj; +    Tcl_Obj *resObj;		/* Result data for 'configure' */ +    int listc, result = TCL_OK; +    Tcl_Obj **listv; +    MethodName method; -#ifdef TCL_THREADS -    /* Are we in the correct thread ? +    /* +     * Are we in the correct thread?       */ -    if (rcPtr->thread != Tcl_GetCurrentThread ()) { -        int opcode; -        RcForwardParamGetOpt p; +#ifdef TCL_THREADS +    if (rcPtr->thread != Tcl_GetCurrentThread()) { +	int opcode; +	ForwardParam p; -	p.name  = optionName; -	p.value = dsPtr; +	p.getOpt.name = optionName; +	p.getOpt.value = dsPtr; -	if (optionName == (char*) NULL) { -	    opcode = RcOpGetOptAll; +	if (optionName == NULL) { +	    opcode = ForwardedGetOptAll;  	} else { -	    opcode = RcOpGetOpt; +	    opcode = ForwardedGetOpt;  	} -	RcForwardOp (rcPtr, opcode, rcPtr->thread, &p); - -	if (p.b.code != TCL_OK) { -	    Tcl_Obj* err = Tcl_NewStringObj (p.b.msg, -1); +	ForwardOpToHandlerThread(rcPtr, opcode, &p); -	    RcErrorReturn (interp, err); +	if (p.base.code != TCL_OK) { +	    Tcl_Obj *err = Tcl_NewStringObj(p.base.msgStr, -1); -	    Tcl_DecrRefCount (err); -	    if (p.b.vol) {ckfree (p.b.msg);} +	    UnmarshallErrorResult(interp, err); +	    Tcl_DecrRefCount(err); +	    FreeReceivedError(&p);  	} -	return p.b.code; +	return p.base.code;      }  #endif -    /* -------- */ +    if (optionName == NULL) { +	/* +	 * Retrieve all options. +	 */ -    if (optionName == (char*) NULL) { -        /* Retrieve all options. */ -        method    = "cgetall"; +	method = METH_CGETALL;  	optionObj = NULL;      } else { -        /* Retrieve the value of one option */ -       -        method    = "cget"; -	optionObj = Tcl_NewStringObj(optionName,-1); -	if (optionObj == (Tcl_Obj*) NULL) { -	    Tcl_Panic ("Out of memory in RcGetOption"); -	} +	/* +	 * Retrieve the value of one option. +	 */ + +	method = METH_CGET; +	optionObj = Tcl_NewStringObj(optionName, -1); +        Tcl_IncrRefCount(optionObj);      } -    RcInvokeTclMethod (rcPtr, method, optionObj, NULL, -			 &res, &resObj, DO_CAPTURE); +    Tcl_Preserve(rcPtr); -    if (res != TCL_OK) { -        RcErrorReturn (interp, resObj); -	Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	return res; +    if (InvokeTclMethod(rcPtr, method, optionObj, NULL, &resObj)!=TCL_OK) { +	UnmarshallErrorResult(interp, resObj); +        goto error;      } -    /* The result has to go into the 'dsPtr' for propagation to the -     * caller of the driver. +    /* +     * The result has to go into the 'dsPtr' for propagation to the caller of +     * the driver.       */      if (optionObj != NULL) { -        Tcl_DStringAppend (dsPtr, Tcl_GetString (resObj), -1); -	Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	return res; +	TclDStringAppendObj(dsPtr, resObj); +        goto ok;      } -    /* Extract the list and append each item as element. +    /* +     * Extract the list and append each item as element.       */ -    /* NOTE (4): If we extract the string rep we can assume a -     * NOTE (4): properly quoted string. Together with a separating -     * NOTE (4): space this way of simply appending the whole string -     * NOTE (4): rep might be faster. It also doesn't check if the -     * NOTE (4): result is a valid list. Nor that the list has an -     * NOTE (4): even number elements. -     * NOTE (4): --- +    /* +     * NOTE (4): If we extract the string rep we can assume a properly quoted +     * string. Together with a separating space this way of simply appending +     * the whole string rep might be faster. It also doesn't check if the +     * result is a valid list. Nor that the list has an even number elements.       */ -    res = Tcl_ListObjGetElements (interp, resObj, &listc, &listv); - -    if (res != TCL_OK) { -        Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	return res; +    if (Tcl_ListObjGetElements(interp, resObj, &listc, &listv) != TCL_OK) { +        goto error;      }      if ((listc % 2) == 1) { -        /* Odd number of elements is wrong. +	/* +	 * Odd number of elements is wrong.  	 */ -	Tcl_Obj *objPtr = Tcl_NewObj(); -	Tcl_ResetResult(interp); -	TclObjPrintf(NULL, objPtr, "Expected list with even number of " -		"elements, got %d element%s instead", listc,  -		(listc == 1 ? "" : "s")); -	Tcl_SetObjResult(interp, objPtr); -	Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -	return TCL_ERROR; -    } - -    { -        int len; -	char* str = Tcl_GetStringFromObj (resObj, &len); +	Tcl_ResetResult(interp); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"Expected list with even number of " +		"elements, got %d element%s instead", listc, +		(listc == 1 ? "" : "s"))); +        goto error; +    } else { +	int len; +	const char *str = Tcl_GetStringFromObj(resObj, &len);  	if (len) { -	    Tcl_DStringAppend (dsPtr, " ", 1); -	    Tcl_DStringAppend (dsPtr, str, len); +	    TclDStringAppendLiteral(dsPtr, " "); +	    Tcl_DStringAppend(dsPtr, str, len);  	} +        goto ok; +    } + + ok: +    result = TCL_OK; + stop: +    if (optionObj) { +        Tcl_DecrRefCount(optionObj);      } -    Tcl_DecrRefCount (resObj); /* Remove reference we held from the invoke */ -    return res; +    Tcl_DecrRefCount(resObj);	/* Remove reference held from invoke */ +    Tcl_Release(rcPtr); +    return result; + error: +    result = TCL_ERROR; +    goto stop;  }  /* @@ -1754,58 +1959,61 @@ RcGetOption (clientData, interp, optionName, dsPtr)  /*   *----------------------------------------------------------------------   * - * RcEncodeEventMask -- + * EncodeEventMask --   *   *	This function takes a list of event items and constructs the - *	equivalent internal bitmask. The list has to contain at - *	least one element. Elements are "read", "write", or any unique - *	abbreviation thereof. Note that the bitmask is not changed if - *	problems are encountered. + *	equivalent internal bitmask. The list must contain at least one + *	element. Elements are "read", "write", or any unique abbreviation of + *	them. Note that the bitmask is not changed if problems are + *	encountered.   *   * Results: - *	A standard Tcl error code. A bitmask where TCL_READABLE - *	and/or TCL_WRITABLE can be set. + *	A standard Tcl error code. A bitmask where TCL_READABLE and/or + *	TCL_WRITABLE can be set.   *   * Side effects: - *	May shimmer 'obj' to a list representation. May place an - *	error message into the interp result. + *	May shimmer 'obj' to a list representation. May place an error message + *	into the interp result.   *   *----------------------------------------------------------------------   */  static int -RcEncodeEventMask (interp, objName, obj, mask) -     Tcl_Interp* interp; -     CONST char* objName; -     Tcl_Obj*    obj; -     int*        mask; +EncodeEventMask( +    Tcl_Interp *interp, +    const char *objName, +    Tcl_Obj *obj, +    int *mask)  { -    int        events;	/* Mask of events to post */ -    int        listc;     /* #elements in eventspec list */ -    Tcl_Obj**  listv;     /* Elements of eventspec list */ -    int        evIndex;   /* Id of event for an element of the -			 * eventspec list */ +    int events;			/* Mask of events to post */ +    int listc;			/* #elements in eventspec list */ +    Tcl_Obj **listv;		/* Elements of eventspec list */ +    int evIndex;		/* Id of event for an element of the eventspec +				 * list. */ -    if (Tcl_ListObjGetElements (interp, obj, -				&listc, &listv) != TCL_OK) { -        return TCL_ERROR; +    if (Tcl_ListObjGetElements(interp, obj, &listc, &listv) != TCL_OK) { +	return TCL_ERROR;      }      if (listc < 1) { -        Tcl_AppendResult(interp, "bad ", objName, " list: is empty", -			 (char *) NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "bad %s list: is empty", objName));  	return TCL_ERROR;      }      events = 0;      while (listc > 0) { -        if (Tcl_GetIndexFromObj (interp, listv [listc-1], -				 eventOptions, objName, 0, &evIndex) != TCL_OK) { +	if (Tcl_GetIndexFromObj(interp, listv[listc-1], eventOptions, +		objName, 0, &evIndex) != TCL_OK) {  	    return TCL_ERROR;  	}  	switch (evIndex) { -	    case EVENT_READ:  events |= TCL_READABLE; break; -	    case EVENT_WRITE: events |= TCL_WRITABLE; break; +	case EVENT_READ: +	    events |= TCL_READABLE; +	    break; +	case EVENT_WRITE: +	    events |= TCL_WRITABLE; +	    break;  	}  	listc --;      } @@ -1817,14 +2025,14 @@ RcEncodeEventMask (interp, objName, obj, mask)  /*   *----------------------------------------------------------------------   * - * RcDecodeEventMask -- + * DecodeEventMask --   * - *	This function takes an internal bitmask of events and - *	constructs the equivalent list of event items. + *	This function takes an internal bitmask of events and constructs the + *	equivalent list of event items.   * - * Results: - *	A Tcl_Obj reference. The object will have a refCount of - *	one. The user has to decrement it to release the object. + * Results, Contract: + *	A Tcl_Obj reference. The object will have a refCount of one. The user + *	has to decrement it to release the object.   *   * Side effects:   *	None. @@ -1832,31 +2040,41 @@ RcEncodeEventMask (interp, objName, obj, mask)   *----------------------------------------------------------------------   */ -static Tcl_Obj* -RcDecodeEventMask (mask) -     int mask; +static Tcl_Obj * +DecodeEventMask( +    int mask)  { -    Tcl_Obj* evObj = Tcl_NewStringObj (((mask & RANDW) == RANDW) ? -				       "read write" : -				       ((mask & TCL_READABLE) ? -					"read" : -					((mask & TCL_WRITABLE) ? -					 "write" : "")), -1); -    if (evObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcDecodeEventMask"); +    register const char *eventStr; +    Tcl_Obj *evObj; + +    switch (mask & RANDW) { +    case RANDW: +	eventStr = "read write"; +	break; +    case TCL_READABLE: +	eventStr = "read"; +	break; +    case TCL_WRITABLE: +	eventStr = "write"; +	break; +    default: +	eventStr = ""; +	break;      } -    Tcl_IncrRefCount (evObj); +    evObj = Tcl_NewStringObj(eventStr, -1); +    Tcl_IncrRefCount(evObj); +    /* assert evObj.refCount == 1 */      return evObj;  }  /*   *----------------------------------------------------------------------   * - * RcNew -- + * NewReflectedChannel --   * - *	This function is invoked to allocate and initialize the - *	instance data of a new reflected channel. + *	This function is invoked to allocate and initialize the instance data + *	of a new reflected channel.   *   * Results:   *	A heap-allocated channel instance. @@ -1867,691 +2085,1078 @@ RcDecodeEventMask (mask)   *----------------------------------------------------------------------   */ -static ReflectingChannel* -RcNew (interp, cmdpfxObj, mode, id) -       Tcl_Interp* interp; -       Tcl_Obj*    cmdpfxObj; -       int         mode; -       Tcl_Obj*    id; +static ReflectedChannel * +NewReflectedChannel( +    Tcl_Interp *interp, +    Tcl_Obj *cmdpfxObj, +    int mode, +    Tcl_Obj *handleObj)  { -    ReflectingChannel* rcPtr; -    int                listc; -    Tcl_Obj**          listv; -    Tcl_Obj*           word; -    int                i; +    ReflectedChannel *rcPtr; +    MethodName mn = METH_BLOCKING; -    rcPtr = (ReflectingChannel*) ckalloc (sizeof(ReflectingChannel)); +    rcPtr = ckalloc(sizeof(ReflectedChannel)); -    /* rcPtr->chan    : Assigned by caller. Dummy data here. */ -    /* rcPtr->methods : Assigned by caller. Dummy data here. */ +    /* rcPtr->chan: Assigned by caller. Dummy data here. */ -    rcPtr->chan     = (Tcl_Channel) NULL; -    rcPtr->methods  = 0; -    rcPtr->interp   = interp; +    rcPtr->chan = NULL; +    rcPtr->interp = interp; +    rcPtr->dead = 0;  #ifdef TCL_THREADS -    rcPtr->thread   = Tcl_GetCurrentThread (); +    rcPtr->thread = Tcl_GetCurrentThread();  #endif -    rcPtr->mode     = mode; -    rcPtr->interest = 0; /* Initially no interest registered */ - -    /* Method placeholder */ +    rcPtr->mode = mode; +    rcPtr->interest = 0;		/* Initially no interest registered */      /* ASSERT: cmdpfxObj is a Tcl List */ +    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; +} + +/* + *---------------------------------------------------------------------- + * + * NextHandle -- + * + *	This function is invoked to generate a channel handle for a new + *	reflected channel. + * + * Results: + *	A Tcl_Obj containing the string of the new channel handle. The + *	refcount of the returned object is -- zero --. + * + * Side effects: + *	May allocate memory. Mutex protected critical section locks out other + *	threads for a short time. + * + *---------------------------------------------------------------------- + */ -    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 +static Tcl_Obj * +NextHandle(void) +{ +    /* +     * Count number of generated reflected channels. Used for id generation. +     * Ids are never reclaimed and there is no dealing with wrap around. On +     * the other hand, "unsigned long" should be big enough except for +     * absolute longrunners (generate a 100 ids per second => overflow will +     * occur in 1 1/3 years).       */ -    rcPtr->argc = listc + 2; -    rcPtr->argv = (Tcl_Obj**) ckalloc (sizeof (Tcl_Obj*) * (listc+4)); - -    for (i = 0; i < listc ; i++) { -        word = rcPtr->argv [i] = listv [i]; -        Tcl_IncrRefCount (word); -    } +    TCL_DECLARE_MUTEX(rcCounterMutex) +    static unsigned long rcCounter = 0; +    Tcl_Obj *resObj; -    i++; /* Skip placeholder for method */ +    Tcl_MutexLock(&rcCounterMutex); +    resObj = Tcl_ObjPrintf("rc%lu", rcCounter); +    rcCounter++; +    Tcl_MutexUnlock(&rcCounterMutex); -    rcPtr->argv [i] = id ; Tcl_IncrRefCount (id); +    return resObj; +} + +static void +FreeReflectedChannel( +    ReflectedChannel *rcPtr) +{ +    Channel *chanPtr = (Channel *) rcPtr->chan; -    /* The next two objects are kept empty, varying arguments */ +    if (chanPtr->typePtr != &tclRChannelType) { +	/* +	 * Delete a cloned ChannelType structure. +	 */ -    /* Initialization complete */ -    return rcPtr; +	ckfree(chanPtr->typePtr); +	chanPtr->typePtr = NULL; +    } +    Tcl_Release(chanPtr); +    Tcl_DecrRefCount(rcPtr->name); +    Tcl_DecrRefCount(rcPtr->methods); +    Tcl_DecrRefCount(rcPtr->cmd); +    ckfree(rcPtr);  }  /*   *----------------------------------------------------------------------   * - * RcNewHandle -- + * InvokeTclMethod --   * - *	This function is invoked to generate a channel handle for - *	a new reflected channel. + *	This function is used to invoke the Tcl level of a reflected channel. + *	It handles all the command assembly, invokation, and generic state and + *	result mgmt. It does *not* handle thread redirection; that is the + *	responsibility of clients of this function.   *   * Results: - *	A Tcl_Obj containing the string of the new channel handle. - *	The refcount of the returned object is -- zero --. + *	Result code and data as returned by the method.   *   * Side effects: - *	May allocate memory. Mutex protected critical section - *	locks out other threads for a short time. + *	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 Tcl_Obj* -RcNewHandle () +static int +InvokeTclMethod( +    ReflectedChannel *rcPtr, +    MethodName method, +    Tcl_Obj *argOneObj,		/* NULL'able */ +    Tcl_Obj *argTwoObj,		/* NULL'able */ +    Tcl_Obj **resultObjPtr)	/* NULL'able */  { -    /* Count number of generated reflected channels.  Used for id -     * generation. Ids are never reclaimed and there is no dealing -     * with wrap around. On the other hand, "unsigned long" should be -     * big enough except for absolute longrunners (generate a 100 ids -     * per second => overflow will occur in 1 1/3 years). +    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->dead) { +	/* +	 * The channel is marked as dead. Bail out immediately, with an +	 * appropriate error. +	 */ + +	if (resultObjPtr != NULL) { +	    resObj = Tcl_NewStringObj(msg_dstlost,-1); +	    *resultObjPtr = resObj; +	    Tcl_IncrRefCount(resObj); +	} + +        /* +         * Not touching argOneObj, argTwoObj, they have not been used. +         * See the contract as well. +         */ + +	return TCL_ERROR; +    } + +    /* +     * Insert method into the callback command, after the command prefix, +     * before the channel id.       */ -#ifdef TCL_THREADS -    TCL_DECLARE_MUTEX (rcCounterMutex) -#endif -    static unsigned long rcCounter = 0; +    cmd = TclListObjCopy(NULL, rcPtr->cmd); -    Tcl_Obj* res = Tcl_NewObj (); +    Tcl_ListObjIndex(NULL, rcPtr->methods, method, &methObj); +    Tcl_ListObjAppendElement(NULL, cmd, methObj); +    Tcl_ListObjAppendElement(NULL, cmd, rcPtr->name); -#ifdef TCL_THREADS -    Tcl_MutexLock (&rcCounterMutex); -#endif +    /* +     * 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. +     */ -    TclObjPrintf(NULL, res, "rc%lu", rcCounter); -    rcCounter ++; +    if (argOneObj) { +	Tcl_ListObjAppendElement(NULL, cmd, argOneObj); +	if (argTwoObj) { +	    Tcl_ListObjAppendElement(NULL, cmd, argTwoObj); +	} +    } -#ifdef TCL_THREADS -    Tcl_MutexUnlock (&rcCounterMutex); -#endif +    /* +     * And run the handler... This is done in auch a manner which leaves any +     * existing state intact. +     */ + +    Tcl_IncrRefCount(cmd); +    sr = Tcl_SaveInterpState(rcPtr->interp, 0 /* Dummy */); +    Tcl_Preserve(rcPtr->interp); +    result = Tcl_EvalObjEx(rcPtr->interp, cmd, TCL_EVAL_GLOBAL); + +    /* +     * We do not try to extract the result information if the caller has no +     * interest in it. I.e. there is no need to put effort into creating +     * something which is discarded immediately after. +     */ + +    if (resultObjPtr) { +	if (result == TCL_OK) { +	    /* +	     * Ok result taken as is, also if the caller requests that there +	     * is no capture. +	     */ + +	    resObj = Tcl_GetObjResult(rcPtr->interp); +	} else { +	    /* +	     * Non-ok result is always treated as an error. We have to capture +	     * the full state of the result, including additional options. +	     * +	     * This is complex and ugly, and would be completely unnecessary +	     * if we only added support for a TCL_FORBID_EXCEPTIONS flag. +	     */ + +	    if (result != TCL_ERROR) { +		int cmdLen; +		const char *cmdString = Tcl_GetStringFromObj(cmd, &cmdLen); + +		Tcl_IncrRefCount(cmd); +		Tcl_ResetResult(rcPtr->interp); +		Tcl_SetObjResult(rcPtr->interp, Tcl_ObjPrintf( +			"chan handler returned bad code: %d", result)); +		Tcl_LogCommandInfo(rcPtr->interp, cmdString, cmdString, +			cmdLen); +		Tcl_DecrRefCount(cmd); +		result = TCL_ERROR; +	    } +	    Tcl_AppendObjToErrorInfo(rcPtr->interp, Tcl_ObjPrintf( +		    "\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); + +    /* +     * 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). +     */ -    return res; +    if (resultObjPtr != NULL) { +	*resultObjPtr = resObj; +    } + +    /* +     * There no need to handle the case where nothing is returned, because for +     * that case resObj was not set anyway. +     */ + +    return result;  } +/* + *---------------------------------------------------------------------- + * + * ErrnoReturn -- + * + *	Checks a method error result if it returned an 'errno'. + * + * Results: + *	The negative errno found in the error result, or 0. + * + * Side effects: + *	None. + * + * Users: + *	ReflectInput/Output(), to enable the signaling of EAGAIN + *	on 0-sized short reads/writes. + * + *---------------------------------------------------------------------- + */ -static void -RcFree (rcPtr) -     ReflectingChannel* rcPtr; +static int +ErrnoReturn( +    ReflectedChannel *rcPtr, +    Tcl_Obj *resObj)  { -    Channel* chanPtr = (Channel*) rcPtr->chan; -    int      i, n; +    int code; +    Tcl_InterpState sr;		/* State of handler interp */ -    if (chanPtr->typePtr != &tclRChannelType) { -        /* Delete a cloned ChannelType structure. */ -        ckfree ((char*) chanPtr->typePtr); +    if (rcPtr->dead) { +	return 0;      } -    n = rcPtr->argc - 2; -    for (i = 0; i < n; i++) { -        Tcl_DecrRefCount (rcPtr->argv[i]); +    sr = Tcl_SaveInterpState(rcPtr->interp, 0 /* Dummy */); +    UnmarshallErrorResult(rcPtr->interp, resObj); + +    resObj = Tcl_GetObjResult(rcPtr->interp); + +    if (((Tcl_GetIntFromObj(rcPtr->interp, resObj, &code) != TCL_OK) +	    || (code >= 0))) { +	if (strcmp("EAGAIN", Tcl_GetString(resObj)) == 0) { +	    code = -EAGAIN; +	} else { +	    code = 0; +	}      } -    ckfree ((char*) rcPtr->argv); -    ckfree ((char*) rcPtr); -    return; +    Tcl_RestoreInterpState(rcPtr->interp, sr); +    return code; +} + +/* + *---------------------------------------------------------------------- + * + * GetReflectedChannelMap -- + * + *	Gets and potentially initializes the reflected channel map for an + *	interpreter. + * + * Results: + *	A pointer to the map created, for use by the caller. + * + * Side effects: + *	Initializes the reflected channel map for an interpreter. + * + *---------------------------------------------------------------------- + */ + +static ReflectedChannelMap * +GetReflectedChannelMap( +    Tcl_Interp *interp) +{ +    ReflectedChannelMap *rcmPtr = Tcl_GetAssocData(interp, RCMKEY, NULL); + +    if (rcmPtr == NULL) { +	rcmPtr = ckalloc(sizeof(ReflectedChannelMap)); +	Tcl_InitHashTable(&rcmPtr->map, TCL_STRING_KEYS); +	Tcl_SetAssocData(interp, RCMKEY, +		(Tcl_InterpDeleteProc *) DeleteReflectedChannelMap, rcmPtr); +    } +    return rcmPtr;  }  /*   *----------------------------------------------------------------------   * - * RcInvokeTclMethod -- + * DeleteReflectedChannelMap --   * - *	This function is used to invoke the Tcl level of a reflected - *	channel. It handles all the command assembly, invokation, and - *	generic state and result mgmt. + *	Deletes the channel table for an interpreter, closing any open + *	channels whose refcount reaches zero. This procedure is invoked when + *	an interpreter is deleted, via the AssocData cleanup mechanism.   *   * Results: - *      Result code and data as returned by the method. + *	None.   *   * Side effects: - *	Arbitrary, as it calls upo na Tcl script. + *	Deletes the hash table of channels. May close channels. May flush + *	output on closed channels. Removes any channeEvent handlers that were + *	registered in this interpreter.   *   *----------------------------------------------------------------------   */  static void -RcInvokeTclMethod (rcPtr, method, argone, argtwo, result, resultObj, capture) -     ReflectingChannel* rcPtr; -     CONST char*        method; -     Tcl_Obj*           argone;    /* NULL'able */ -     Tcl_Obj*           argtwo;    /* NULL'able */ -     int*               result;    /* NULL'able */ -     Tcl_Obj**          resultObj; /* NULL'able */ -     int                capture; +DeleteReflectedChannelMap( +    ClientData clientData,	/* The per-interpreter data structure. */ +    Tcl_Interp *interp)		/* The interpreter being deleted. */  { -    /* Thread redirection was done by higher layers */ -    /* ASSERT: Tcl_GetCurrentThread () == rcPtr->thread */ +    ReflectedChannelMap *rcmPtr = clientData; +				/* The map */ +    Tcl_HashSearch hSearch;	 /* Search variable. */ +    Tcl_HashEntry *hPtr;	 /* Search variable. */ +    ReflectedChannel *rcPtr; +    Tcl_Channel chan; +#ifdef TCL_THREADS +    ForwardingResult *resultPtr; +    ForwardingEvent *evPtr; +    ForwardParam *paramPtr; +#endif -    int             cmdc;           /* #words in constructed command */ -    Tcl_Obj*        methObj = NULL; /* Method name in object form */ -    Tcl_InterpState sr;             /* State of handler interp */ -    int             res;            /* Result code of method invokation */ -    Tcl_Obj*        resObj  = NULL; /* Result of method invokation. */ +    /* +     * Delete all entries. The channels may have been closed already, or will +     * be closed later, by the standard IO finalization of an interpreter +     * under destruction. Except for the channels which were moved to a +     * different interpreter and/or thread. They do not exist from the IO +     * systems point of view and will not get closed. Therefore mark all as +     * dead so that any future access will cause a proper error. For channels +     * in a different thread we actually do the same as +     * DeleteThreadReflectedChannelMap(), just restricted to the channels of +     * this interp. +     */ + +    for (hPtr = Tcl_FirstHashEntry(&rcmPtr->map, &hSearch); +	    hPtr != NULL; +	    hPtr = Tcl_FirstHashEntry(&rcmPtr->map, &hSearch)) { +	chan = Tcl_GetHashValue(hPtr); +	rcPtr = Tcl_GetChannelInstanceData(chan); + +	rcPtr->dead = 1; +	Tcl_DeleteHashEntry(hPtr); +    } +    Tcl_DeleteHashTable(&rcmPtr->map); +    ckfree(&rcmPtr->map); -    /* NOTE (5): Decide impl. issue: Cache objects with method names ? -     * NOTE (5): Requires TSD data as reflections can be created in -     * NOTE (5): many different threads. -     * NOTE (5): --- +#ifdef TCL_THREADS +    /* +     * The origin interpreter for one or more reflected channels is gone.       */ -    /* Insert method into the pre-allocated area, after the command -     * prefix, before the channel id. +    /* +     * Go through the list of pending results and cancel all whose events were +     * destined for this interpreter. While this is in progress we block any +     * other access to the list of pending results.       */ -    methObj = Tcl_NewStringObj (method, -1); -    if (methObj == (Tcl_Obj*) NULL) { -        Tcl_Panic ("Out of memory in RcInvokeTclMethod"); +    Tcl_MutexLock(&rcForwardMutex); + +    for (resultPtr = forwardList; +	    resultPtr != NULL; +	    resultPtr = resultPtr->nextPtr) { +	if (resultPtr->dsti != interp) { +	    /* +	     * Ignore results/events for other interpreters. +	     */ + +	    continue; +	} + +	/* +	 * The receiver for the event exited, before processing the event. We +	 * detach the result now, wake the originator up and signal failure. +	 */ + +	evPtr = resultPtr->evPtr; + +	/* Basic crash safety until this routine can get revised [3411310] */ +	if (evPtr == NULL) { +	    continue; +	} +	paramPtr = evPtr->param; + +	evPtr->resultPtr = NULL; +	resultPtr->evPtr = NULL; +	resultPtr->result = TCL_ERROR; + +	ForwardSetStaticError(paramPtr, msg_send_dstlost); + +	Tcl_ConditionNotify(&resultPtr->done);      } -    Tcl_IncrRefCount (methObj); -    rcPtr->argv [rcPtr->argc - 2] = methObj; +    Tcl_MutexUnlock(&rcForwardMutex); -    /* Append the additional argument containing method specific -     * details behind the channel id. If specified. +    /* +     * Get the map of all channels handled by the current thread. This is a +     * ReflectedChannelMap, but on a per-thread basis, not per-interp. Go +     * through the channels and remove all which were handled by this +     * interpreter. They have already been marked as dead.       */ -    cmdc = rcPtr->argc ; -    if (argone) { -        Tcl_IncrRefCount (argone); -	rcPtr->argv [cmdc] = argone; -	cmdc++; +    rcmPtr = GetThreadReflectedChannelMap(); +    for (hPtr = Tcl_FirstHashEntry(&rcmPtr->map, &hSearch); +	    hPtr != NULL; +	    hPtr = Tcl_NextHashEntry(&hSearch)) { +	chan = Tcl_GetHashValue(hPtr); +	rcPtr = Tcl_GetChannelInstanceData(chan); + +	if (rcPtr->interp != interp) { +	    /* +	     * Ignore entries for other interpreters. +	     */ + +	    continue; +	} + +	rcPtr->dead = 1; +	Tcl_DeleteHashEntry(hPtr);      } -    if (argtwo) { -        Tcl_IncrRefCount (argtwo); -	rcPtr->argv [cmdc] = argtwo; -	cmdc++; +#endif +} + +#ifdef TCL_THREADS +/* + *---------------------------------------------------------------------- + * + * GetThreadReflectedChannelMap -- + * + *	Gets and potentially initializes the reflected channel map for a + *	thread. + * + * Results: + *	A pointer to the map created, for use by the caller. + * + * Side effects: + *	Initializes the reflected channel map for a thread. + * + *---------------------------------------------------------------------- + */ + +static ReflectedChannelMap * +GetThreadReflectedChannelMap(void) +{ +    ThreadSpecificData *tsdPtr = TCL_TSD_INIT(&dataKey); + +    if (!tsdPtr->rcmPtr) { +	tsdPtr->rcmPtr = ckalloc(sizeof(ReflectedChannelMap)); +	Tcl_InitHashTable(&tsdPtr->rcmPtr->map, TCL_STRING_KEYS); +	Tcl_CreateThreadExitHandler(DeleteThreadReflectedChannelMap, NULL);      } -    /* And run the handler ...  This is done in auch a manner which -     * leaves any existing state intact. -     */ +    return tsdPtr->rcmPtr; +} + +/* + *---------------------------------------------------------------------- + * + * DeleteThreadReflectedChannelMap -- + * + *	Deletes the channel table for a thread. This procedure is invoked when + *	a thread is deleted. The channels have already been marked as dead, in + *	DeleteReflectedChannelMap(). + * + * Results: + *	None. + * + * Side effects: + *	Deletes the hash table of channels. + * + *---------------------------------------------------------------------- + */ -    sr  = Tcl_SaveInterpState (rcPtr->interp, 0 /* Dummy */); -    res = Tcl_EvalObjv        (rcPtr->interp, cmdc, rcPtr->argv, TCL_EVAL_GLOBAL); +static void +DeleteThreadReflectedChannelMap( +    ClientData clientData)	/* The per-thread data structure. */ +{ +    Tcl_HashSearch hSearch;	 /* Search variable. */ +    Tcl_HashEntry *hPtr;	 /* Search variable. */ +    Tcl_ThreadId self = Tcl_GetCurrentThread(); +    ReflectedChannelMap *rcmPtr; /* The map */ +    ForwardingResult *resultPtr; -    /* We do not try to extract the result information if the caller has no -     * interest in it. I.e. there is no need to put effort into creating -     * something which is discarded immediately after. +    /* +     * The origin thread for one or more reflected channels is gone. +     * NOTE: If this function is called due to a thread getting killed the +     *       per-interp DeleteReflectedChannelMap is apparently not called.       */ -    if (resultObj) { -	if ((res == TCL_OK) || !capture) { -	    /* Ok result taken as is, also if the caller requests that there -	     * is no capture. -	     */ +    /* +     * Go through the list of pending results and cancel all whose events were +     * destined for this thread. While this is in progress we block any +     * other access to the list of pending results. +     */ -	    resObj = Tcl_GetObjResult (rcPtr->interp); -	} else { -	    /* Non-ok ressult is always treated as an error. -	     * We have to capture the full state of the result, -	     * including additional options. +    Tcl_MutexLock(&rcForwardMutex); + +    for (resultPtr = forwardList; +	    resultPtr != NULL; +	    resultPtr = resultPtr->nextPtr) { +	ForwardingEvent *evPtr; +	ForwardParam *paramPtr; + +	if (resultPtr->dst != self) { +	    /* +	     * Ignore results/events for other threads.  	     */ -	    res    = TCL_ERROR; -	    resObj = RcErrorMarshall (rcPtr->interp); +	    continue;  	} -	Tcl_IncrRefCount(resObj); -    } -    Tcl_RestoreInterpState (rcPtr->interp, sr); -    /* ... */ +	/* +	 * The receiver for the event exited, before processing the event. We +	 * detach the result now, wake the originator up and signal failure. +	 */ -    /* Cleanup of the dynamic parts of the command */ +	evPtr = resultPtr->evPtr; -    Tcl_DecrRefCount (methObj); -    if (argone) {Tcl_DecrRefCount (argone);} -    if (argtwo) {Tcl_DecrRefCount (argtwo);} +	/* Basic crash safety until this routine can get revised [3411310] */ +	if (evPtr == NULL ) { +	    continue; +	} +	paramPtr = evPtr->param; -    /* The resObj has a ref count of 1 at this location.  This means -     * that the caller of RcInvoke has to dispose of it (but only if -     * it was returned to it). -     */ +	evPtr->resultPtr = NULL; +	resultPtr->evPtr = NULL; +	resultPtr->result = TCL_ERROR; -    if (result) { -        *result = res; -    } -    if (resultObj) { -        *resultObj = resObj; +	ForwardSetStaticError(paramPtr, msg_send_dstlost); + +	Tcl_ConditionNotify(&resultPtr->done);      } -    /* There no need to handle the case where nothing is returned, because for -     * that case resObj was not set anyway. +    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 +     * ReflectedChannelMap, but on a per-thread basis, not per-interp. Go +     * through the channels, remove all, mark them as dead.       */ + +    rcmPtr = GetThreadReflectedChannelMap(); +    for (hPtr = Tcl_FirstHashEntry(&rcmPtr->map, &hSearch); +	    hPtr != NULL; +	    hPtr = Tcl_FirstHashEntry(&rcmPtr->map, &hSearch)) { +	Tcl_Channel chan = Tcl_GetHashValue(hPtr); +	ReflectedChannel *rcPtr = Tcl_GetChannelInstanceData(chan); + +	rcPtr->dead = 1; +	Tcl_DeleteHashEntry(hPtr); +    } +    ckfree(rcmPtr);  } -#ifdef TCL_THREADS  static void -RcForwardOp (rcPtr, op, dst, param) -  ReflectingChannel* rcPtr; /* Channel instance */ -  RcOperation        op;    /* Forwarded driver operation */ -  Tcl_ThreadId       dst;   /* Destination thread */ -  CONST VOID*        param; /* Arguments */ +ForwardOpToHandlerThread( +    ReflectedChannel *rcPtr,	/* Channel instance */ +    ForwardedOperation op,	/* Forwarded driver operation */ +    const void *param)		/* Arguments */  { -    RcForwardingEvent*  evPtr; -    RcForwardingResult* resultPtr; -    int                 result; +    /* +     * Core of the communication from OWNER to HANDLER thread. +     * The receiver is ForwardProc() below. +     */ -    /* Create and initialize the event and data structures */ +    Tcl_ThreadId dst = rcPtr->thread; +    ForwardingEvent *evPtr; +    ForwardingResult *resultPtr; -    evPtr     = (RcForwardingEvent*)  ckalloc (sizeof (RcForwardingEvent)); -    resultPtr = (RcForwardingResult*) ckalloc (sizeof (RcForwardingResult)); +    /* +     * We gather the lock early. This allows us to check the liveness of the +     * channel without interference from DeleteThreadReflectedChannelMap(). +     */ -    evPtr->event.proc = RcForwardProc; -    evPtr->resultPtr  = resultPtr; -    evPtr->op         = op; -    evPtr->rcPtr      = rcPtr; -    evPtr->param      = param; +    Tcl_MutexLock(&rcForwardMutex); + +    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. +	 */ -    resultPtr->src    = Tcl_GetCurrentThread (); -    resultPtr->dst    = dst; -    resultPtr->done   = (Tcl_Condition) NULL; +	ForwardSetStaticError((ForwardParam *) param, msg_send_dstlost); +	Tcl_MutexUnlock(&rcForwardMutex); +	return; +    } + +    /* +     * Create and initialize the event and data structures. +     */ + +    evPtr = ckalloc(sizeof(ForwardingEvent)); +    resultPtr = ckalloc(sizeof(ForwardingResult)); + +    evPtr->event.proc = ForwardProc; +    evPtr->resultPtr = resultPtr; +    evPtr->op = op; +    evPtr->rcPtr = rcPtr; +    evPtr->param = (ForwardParam *) param; + +    resultPtr->src = Tcl_GetCurrentThread(); +    resultPtr->dst = dst; +    resultPtr->dsti = rcPtr->interp; +    resultPtr->done = NULL;      resultPtr->result = -1; -    resultPtr->evPtr  = evPtr; +    resultPtr->evPtr = evPtr; -    /* Now execute the forward */ +    /* +     * Now execute the forward. +     */ -    Tcl_MutexLock(&rcForwardMutex);      TclSpliceIn(resultPtr, forwardList); +    /* Do not unlock here. That is done by the ConditionWait */      /* -     * Ensure cleanup of the event if any of the two involved threads -     * exits while this event is pending or in progress. +     * 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 +     * DeleteThreadReflectedChannelMap(), this is set up by +     * GetThreadReflectedChannelMap(). This is what we use the 'forwardList' +     * (see above) for.       */ -    Tcl_CreateThreadExitHandler(RcSrcExitProc, (ClientData) evPtr); -    Tcl_CreateThreadExitHandler(RcDstExitProc, (ClientData) evPtr); +    Tcl_CreateThreadExitHandler(SrcExitProc, evPtr);      /*       * Queue the event and poke the other thread's notifier.       */ -    Tcl_ThreadQueueEvent(dst, (Tcl_Event*)evPtr, TCL_QUEUE_TAIL); +    Tcl_ThreadQueueEvent(dst, (Tcl_Event *) evPtr, TCL_QUEUE_TAIL);      Tcl_ThreadAlert(dst);      /* -     * (*) Block until the other thread has either processed the transfer -     * or rejected it. +     * (*) Block until the handler thread has either processed the transfer or +     * rejected it.       */      while (resultPtr->result < 0) { -        /* NOTE (1): Is it possible that the current thread goes away while waiting here ? -	 * NOTE (1): IOW Is it possible that "RcSrcExitProc" is called while we are here ? -	 * NOTE (1): See complementary note (2) in "RcSrcExitProc" -	 * NOTE (1): --- +	/* +	 * NOTE (1): Is it possible that the current thread goes away while +	 * waiting here? IOW Is it possible that "SrcExitProc" is called while +	 * we are here? See complementary note (2) in "SrcExitProc" +	 * +	 * The ConditionWait unlocks the mutex during the wait and relocks it +	 * immediately after.  	 */ -        Tcl_ConditionWait(&resultPtr->done, &rcForwardMutex, NULL); +	Tcl_ConditionWait(&resultPtr->done, &rcForwardMutex, NULL);      }      /* -     * Unlink result from the forwarder list. +     * Unlink result from the forwarder list. No need to lock. Either still +     * locked, or locked by the ConditionWait       */      TclSpliceOut(resultPtr, forwardList); -    resultPtr->nextPtr  = NULL; -    resultPtr->prevPtr  = NULL; +    resultPtr->nextPtr = NULL; +    resultPtr->prevPtr = NULL;      Tcl_MutexUnlock(&rcForwardMutex);      Tcl_ConditionFinalize(&resultPtr->done);      /* -     * Kill the cleanup handlers now, and the result structure as well, -     * before returning the success code. +     * Kill the cleanup handler now, and the result structure as well, before +     * returning the success code.       *       * Note: The event structure has already been deleted.       */ -    Tcl_DeleteThreadExitHandler(RcSrcExitProc, (ClientData) evPtr); -    Tcl_DeleteThreadExitHandler(RcDstExitProc, (ClientData) evPtr); -     -    result =  resultPtr->result; -    ckfree ((char*) resultPtr); +    Tcl_DeleteThreadExitHandler(SrcExitProc, evPtr); + +    ckfree(resultPtr);  }  static int -RcForwardProc (evGPtr, mask) -     Tcl_Event *evGPtr;  -     int mask; +ForwardProc( +    Tcl_Event *evGPtr, +    int mask)  { -    /* Notes regarding access to the referenced data. +    /* +     * 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 -     * evPtr->src), however this thread is currently blocked at (*), -     * i.e. quiescent. Because of this we can treat the data as -     * belonging to us, without fear of race conditions. I.e. we can -     * read and write as we like. +     * evPtr->src), however this thread is currently blocked at (*), i.e. +     * quiescent. Because of this we can treat the data as belonging to us, +     * without fear of race conditions. I.e. we can read and write as we like.       * -     * The only thing we cannot be sure of is the resultPtr. This can be -     * be NULLed if the originating thread went away while the event -     * is handled here now. +     * The only thing we cannot be sure of is the resultPtr. This can be be +     * NULLed if the originating thread went away while the event is handled +     * here now.       */ -    RcForwardingEvent*  evPtr     = (RcForwardingEvent*) evGPtr; -    RcForwardingResult* resultPtr = evPtr->resultPtr; -    ReflectingChannel*  rcPtr     = evPtr->rcPtr; -    Tcl_Interp*         interp    = rcPtr->interp; -    RcForwardParamBase* paramPtr  = (RcForwardParamBase*) evPtr->param; -    int                 res       = TCL_OK; /* Result code   of RcInvokeTclMethod */ -    Tcl_Obj*            resObj    = NULL;   /* Interp result of RcInvokeTclMethod */ +    ForwardingEvent *evPtr = (ForwardingEvent *) evGPtr; +    ForwardingResult *resultPtr = evPtr->resultPtr; +    ReflectedChannel *rcPtr = evPtr->rcPtr; +    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. */ +    Tcl_HashEntry *hPtr;	/* Entry in the above map */ -    /* Ignore the event if no one is waiting for its result anymore. +    /* +     * Ignore the event if no one is waiting for its result anymore.       */      if (!resultPtr) { -        return 1; +	return 1;      } -    paramPtr->code = TCL_OK; -    paramPtr->msg  = NULL; -    paramPtr->vol  = 0; +    paramPtr->base.code = TCL_OK; +    paramPtr->base.msgStr = NULL; +    paramPtr->base.mustFree = 0;      switch (evPtr->op) { -      /* The destination thread for the following operations is -       * rcPtr->thread, which contains rcPtr->interp, the interp -       * we have to call upon for the driver. -       */ - -    case RcOpClose: -      { -	  /* No parameters/results */ -	  RcInvokeTclMethod (rcPtr, "finalize", NULL, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	     RcForwardSetObjError (paramPtr, resObj); -	  } - -	  /* Freeing is done here, in the origin thread, because the -	   * argv[] objects belong to this thread. Deallocating them -	   * in a different thread is not allowed -	   */ - -	  RcFree (rcPtr); -      } -      break; - -    case RcOpInput: -      { -	  RcForwardParamInput* p = (RcForwardParamInput*) paramPtr; -	  Tcl_Obj*     toReadObj = Tcl_NewIntObj (p->toRead); - -	  if (toReadObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcInput"); -	  } - -	  RcInvokeTclMethod (rcPtr, "read", toReadObj, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	      p->toRead = -1; -	  } else { -	      /* Process a regular result. */ - -	      int            bytec; /* Number of returned bytes */ -	      unsigned char* bytev; /* Array of returned bytes */ - -	      bytev = Tcl_GetByteArrayFromObj(resObj, &bytec); - -	      if (p->toRead < bytec) { -		  RcForwardSetStaticError (paramPtr, msg_read_toomuch); -		  p->toRead   = -1; - -	      } else { -	          if (bytec > 0) { -		      memcpy (p->buf, bytev, bytec); -		  } - -		  p->toRead = bytec; -	      } -	  } -      } -      break; - -    case RcOpOutput: -      { -	  RcForwardParamOutput* p = (RcForwardParamOutput*) paramPtr; -	  Tcl_Obj*         bufObj = Tcl_NewByteArrayObj((unsigned char*) p->buf, p->toWrite); - -	  if (bufObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcOutput"); -	  } - -	  RcInvokeTclMethod (rcPtr, "write", bufObj, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	      p->toWrite = -1; -	  } else { -	      /* Process a regular result. */ - -	      int written; - -	      res = Tcl_GetIntFromObj (interp, resObj, &written); -	      if (res != TCL_OK) { - -		  RcForwardSetObjError (paramPtr, RcErrorMarshall (interp)); -		  p->toWrite = -1; - -	      } else if ((written == 0) || (p->toWrite < written)) { - -		  RcForwardSetStaticError (paramPtr, msg_write_toomuch); -		  p->toWrite = -1; - -	      } else { -		  p->toWrite = written; -	      } -	  } -      } -      break; - -    case RcOpSeek: -      { -	  RcForwardParamSeek* p = (RcForwardParamSeek*) paramPtr; - -	  Tcl_Obj*       offObj; -	  Tcl_Obj*       baseObj; - -	  offObj = Tcl_NewWideIntObj(p->offset); -	  if (offObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcSeekWide"); -	  } - -	  baseObj = Tcl_NewStringObj((p->seekMode == SEEK_SET) ? -				     "start" : -				     ((p->seekMode == SEEK_CUR) ? -				      "current" : -				      "end"), -1); - -	  if (baseObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcSeekWide"); -	  } - -	  RcInvokeTclMethod (rcPtr, "seek", offObj, baseObj, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	      p->offset = -1; -	  } else { -	      /* Process a regular result. If the type is wrong this -	       * may change into an error. -	       */ - -	      Tcl_WideInt newLoc; -	      res = Tcl_GetWideIntFromObj (interp, resObj, &newLoc); - -	      if (res == TCL_OK) { -		  if (newLoc < Tcl_LongAsWide (0)) { -		      RcForwardSetStaticError (paramPtr, msg_seek_beforestart); -		      p->offset = -1; -		  } else { -		      p->offset = newLoc; -		  } -	      } else { -		  RcForwardSetObjError (paramPtr, RcErrorMarshall (interp)); -		  p->offset = -1; -	      } -	  } -      } -      break; - -    case RcOpWatch: -      { -	  RcForwardParamWatch* p = (RcForwardParamWatch*) paramPtr; - -	  Tcl_Obj* maskObj = RcDecodeEventMask (p->mask); -	  RcInvokeTclMethod (rcPtr, "watch", maskObj, NULL, -			     NULL, NULL, NO_CAPTURE); -	  Tcl_DecrRefCount (maskObj); -      } -    break; - -    case RcOpBlock: -      { -	  RcForwardParamBlock* p = (RcForwardParamBlock*) evPtr->param; -	  Tcl_Obj*      blockObj = Tcl_NewBooleanObj(!p->nonblocking); - -	  if (blockObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcBlock"); -	  } - -	  RcInvokeTclMethod (rcPtr, "blocking", blockObj, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	  } -      } -      break; - -    case RcOpSetOpt: -      { -	  RcForwardParamSetOpt* p = (RcForwardParamSetOpt*) paramPtr; -	  Tcl_Obj* optionObj; -	  Tcl_Obj* valueObj; - -	  optionObj = Tcl_NewStringObj(p->name,-1); -	  if (optionObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcSetOption"); -	  } - -	  valueObj = Tcl_NewStringObj(p->value,-1); -	  if (valueObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcSetOption"); -	  } - -	  RcInvokeTclMethod (rcPtr, "configure", optionObj, valueObj, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	  } -      } -      break; - -    case RcOpGetOpt: -      { -	  /* Retrieve the value of one option */ - -	  RcForwardParamGetOpt* p = (RcForwardParamGetOpt*) paramPtr; -	  Tcl_Obj*           optionObj; - -	  optionObj = Tcl_NewStringObj(p->name,-1); -	  if (optionObj == (Tcl_Obj*) NULL) { -	      Tcl_Panic ("Out of memory in RcGetOption"); -	  } - -	  RcInvokeTclMethod (rcPtr, "cget", optionObj, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	  } else { -	      Tcl_DStringAppend (p->value, Tcl_GetString (resObj), -1); -	  } -      } -      break; - -    case RcOpGetOptAll: -      { -	  /* Retrieve all options. */ - -	  RcForwardParamGetOpt* p = (RcForwardParamGetOpt*) paramPtr; - -	  RcInvokeTclMethod (rcPtr, "cgetall", NULL, NULL, -			     &res, &resObj, DO_CAPTURE); - -	  if (res != TCL_OK) { -	      RcForwardSetObjError (paramPtr, resObj); -	  } else { -	      /* Extract list, validate that it is a list, and -	       * #elements. See NOTE (4) as well. -	       */ - -	      int       listc; -	      Tcl_Obj** listv; - -	      res = Tcl_ListObjGetElements (interp, resObj, &listc, &listv); -	      if (res != TCL_OK) { -		  RcForwardSetObjError (paramPtr, RcErrorMarshall (interp)); - -	      } else if ((listc % 2) == 1) { -	          /* Odd number of elements is wrong. -		   * [x]. -		   */ - -	          char* buf = ckalloc (200); -		  sprintf (buf, -			   "{Expected list with even number of elements, got %d %s instead}", -			   listc, -			   (listc == 1 ? "element" : "elements")); -		   -		  RcForwardSetDynError (paramPtr, buf); -	      } else { -		  int len; -		  char* str = Tcl_GetStringFromObj (resObj, &len); - -		  if (len) { -		      Tcl_DStringAppend (p->value, " ", 1); -		      Tcl_DStringAppend (p->value, str, len); -		  } -	      } -	  } -      } -      break; +	/* +	 * The destination thread for the following operations is +	 * rcPtr->thread, which contains rcPtr->interp, the interp we have to +	 * call upon for the driver. +	 */ + +    case ForwardedClose: +	/* +	 * No parameters/results. +	 */ + +	if (InvokeTclMethod(rcPtr, METH_FINAL, NULL, NULL, &resObj)!=TCL_OK) { +	    ForwardSetObjError(paramPtr, resObj); +	} + +	/* +	 * Freeing is done here, in the origin thread, callback command +	 * objects belong to this thread. Deallocating them in a different +	 * thread is not allowed +	 * +	 * We remove the channel from both interpreter and thread maps before +	 * releasing the memory, to prevent future accesses (like by +	 * 'postevent') from finding and dereferencing a dangling pointer. +	 */ + +	rcmPtr = GetReflectedChannelMap(interp); +	hPtr = Tcl_FindHashEntry(&rcmPtr->map, +                Tcl_GetChannelName(rcPtr->chan)); +	Tcl_DeleteHashEntry(hPtr); + +	rcmPtr = GetThreadReflectedChannelMap(); +	hPtr = Tcl_FindHashEntry(&rcmPtr->map, +                Tcl_GetChannelName(rcPtr->chan)); +	Tcl_DeleteHashEntry(hPtr); + +	Tcl_EventuallyFree(rcPtr, (Tcl_FreeProc *) FreeReflectedChannel); +	break; + +    case ForwardedInput: { +	Tcl_Obj *toReadObj = Tcl_NewIntObj(paramPtr->input.toRead); +        Tcl_IncrRefCount(toReadObj); + +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_READ, toReadObj, NULL, &resObj)!=TCL_OK){ +	    int code = ErrnoReturn(rcPtr, resObj); + +	    if (code < 0) { +		paramPtr->base.code = code; +	    } else { +		ForwardSetObjError(paramPtr, resObj); +	    } +	    paramPtr->input.toRead = -1; +	} else { +	    /* +	     * Process a regular result. +	     */ + +	    int bytec;			/* Number of returned bytes */ +	    unsigned char *bytev;	/* Array of returned bytes */ + +	    bytev = Tcl_GetByteArrayFromObj(resObj, &bytec); + +	    if (paramPtr->input.toRead < bytec) { +		ForwardSetStaticError(paramPtr, msg_read_toomuch); +		paramPtr->input.toRead = -1; +	    } else { +		if (bytec > 0) { +		    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); +        Tcl_IncrRefCount(bufObj); + +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_WRITE, bufObj, NULL, &resObj) != TCL_OK) { +	    int code = ErrnoReturn(rcPtr, resObj); + +	    if (code < 0) { +		paramPtr->base.code = code; +	    } else { +		ForwardSetObjError(paramPtr, resObj); +	    } +	    paramPtr->output.toWrite = -1; +	} else { +	    /* +	     * Process a regular result. +	     */ + +	    int written; + +	    if (Tcl_GetIntFromObj(interp, resObj, &written) != TCL_OK) { +		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); +		paramPtr->output.toWrite = -1; +	    } else { +		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); + +        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 { +	    /* +	     * Process a regular result. If the type is wrong this may change +	     * into an error. +	     */ + +	    Tcl_WideInt newLoc; + +	    if (Tcl_GetWideIntFromObj(interp, resObj, &newLoc) == TCL_OK) { +		if (newLoc < Tcl_LongAsWide(0)) { +		    ForwardSetStaticError(paramPtr, msg_seek_beforestart); +		    paramPtr->seek.offset = -1; +		} else { +		    paramPtr->seek.offset = newLoc; +		} +	    } else { +		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 */ + +        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); + +        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_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; +    } + +    case ForwardedGetOpt: { +	/* +	 * Retrieve the value of one option. +	 */ + +	Tcl_Obj *optionObj = Tcl_NewStringObj(paramPtr->getOpt.name, -1); + +        Tcl_IncrRefCount(optionObj); +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_CGET, optionObj, NULL, &resObj)!=TCL_OK){ +	    ForwardSetObjError(paramPtr, resObj); +	} else { +	    TclDStringAppendObj(paramPtr->getOpt.value, resObj); +	} +        Tcl_Release(rcPtr); +        Tcl_DecrRefCount(optionObj); +	break; +    } + +    case ForwardedGetOptAll: +	/* +	 * Retrieve all options. +	 */ + +        Tcl_Preserve(rcPtr); +	if (InvokeTclMethod(rcPtr, METH_CGETALL, NULL, NULL, &resObj) != TCL_OK){ +	    ForwardSetObjError(paramPtr, resObj); +	} else { +	    /* +	     * Extract list, validate that it is a list, and #elements. See +	     * NOTE (4) as well. +	     */ + +	    int listc; +	    Tcl_Obj **listv; + +	    if (Tcl_ListObjGetElements(interp, resObj, &listc, +                    &listv) != TCL_OK) { +		Tcl_DecrRefCount(resObj); +		resObj = MarshallError(interp); +		ForwardSetObjError(paramPtr, resObj); +	    } else if ((listc % 2) == 1) { +		/* +		 * Odd number of elements is wrong. [x]. +		 */ + +		char *buf = ckalloc(200); +		sprintf(buf, +			"{Expected list with even number of elements, got %d %s instead}", +			listc, (listc == 1 ? "element" : "elements")); + +		ForwardSetDynamicError(paramPtr, buf); +	    } else { +		int len; +		const char *str = Tcl_GetStringFromObj(resObj, &len); + +		if (len) { +		    TclDStringAppendLiteral(paramPtr->getOpt.value, " "); +		    Tcl_DStringAppend(paramPtr->getOpt.value, str, len); +		} +	    } +	} +        Tcl_Release(rcPtr); +	break;      default: -      /* Bad operation code */ -      Tcl_Panic ("Bad operation code in RcForwardProc"); -      break; +	/* +	 * Bad operation code. +	 */ + +	Tcl_Panic("Bad operation code in ForwardProc"); +	break;      } -    /* Remove the reference we held on the result of the invoke, if we had -     * such +    /* +     * Remove the reference we held on the result of the invoke, if we had +     * such.       */ +      if (resObj != NULL) { -	Tcl_DecrRefCount (resObj); +	Tcl_DecrRefCount(resObj);      }      if (resultPtr) { -        /* -	 * Report the forwarding result synchronously to the waiting -	 * caller. This unblocks (*) as well. This is wrapped into a -	 * conditional because the caller may have exited in the mean -	 * time. +	/* +	 * Report the forwarding result synchronously to the waiting caller. +	 * This unblocks (*) as well. This is wrapped into a conditional +	 * because the caller may have exited in the mean time.  	 */ -        Tcl_MutexLock(&rcForwardMutex); +	Tcl_MutexLock(&rcForwardMutex);  	resultPtr->result = TCL_OK;  	Tcl_ConditionNotify(&resultPtr->done);  	Tcl_MutexUnlock(&rcForwardMutex); @@ -2560,97 +3165,65 @@ RcForwardProc (evGPtr, mask)      return 1;  } -  static void -RcSrcExitProc (clientData) -     ClientData clientData; +SrcExitProc( +    ClientData clientData)  { -  RcForwardingEvent*  evPtr = (RcForwardingEvent*) clientData; -  RcForwardingResult* resultPtr; -  RcForwardParamBase* paramPtr; - -  /* NOTE (2): Can this handler be called with the originator blocked ? -   * NOTE (2): --- -   */ - -  /* The originator for the event exited. It is not sure if this -   * can happen, as the originator should be blocked at (*) while -   * the event is in transit/pending. -   */ - -  /* -   * We make sure that the event cannot refer to the result anymore, -   * remove it from the list of pending results and free the -   * structure. Locking the access ensures that we cannot get in -   * conflict with "RcForwardProc", should it already execute the -   * event. -   */ - -  Tcl_MutexLock(&rcForwardMutex); +    ForwardingEvent *evPtr = clientData; +    ForwardingResult *resultPtr; +    ForwardParam *paramPtr; -  resultPtr = evPtr->resultPtr; -  paramPtr  = (RcForwardParamBase*) evPtr->param; - -  evPtr->resultPtr  = NULL; -  resultPtr->evPtr  = NULL; -  resultPtr->result = TCL_ERROR; - -  RcForwardSetStaticError (paramPtr, msg_send_originlost); - -  /* See below: TclSpliceOut(resultPtr, forwardList); */ +    /* +     * NOTE (2): Can this handler be called with the originator blocked? +     */ -  Tcl_MutexUnlock(&rcForwardMutex); +    /* +     * The originator for the event exited. It is not sure if this can happen, +     * as the originator should be blocked at (*) while the event is in +     * transit/pending. +     * +     * We make sure that the event cannot refer to the result anymore, remove +     * it from the list of pending results and free the structure. Locking the +     * access ensures that we cannot get in conflict with "ForwardProc", +     * should it already execute the event. +     */ -  /* -   * This unlocks (*). The structure will be spliced out and freed by -   * "RcForwardProc". Maybe. -   */ +    Tcl_MutexLock(&rcForwardMutex); -  Tcl_ConditionNotify(&resultPtr->done); -} - +    resultPtr = evPtr->resultPtr; +    paramPtr = evPtr->param; -static void -RcDstExitProc (clientData) -     ClientData clientData; -{ -  RcForwardingEvent*  evPtr     = (RcForwardingEvent*) clientData; -  RcForwardingResult* resultPtr = evPtr->resultPtr; -  RcForwardParamBase* paramPtr  = (RcForwardParamBase*) evPtr->param; +    evPtr->resultPtr = NULL; +    resultPtr->evPtr = NULL; +    resultPtr->result = TCL_ERROR; -  /* NOTE (3): It is not clear if the event still exists when this handler is called.. -   * NOTE (3): We might have to use 'resultPtr' as our clientData instead. -   * NOTE (3): --- -   */ +    ForwardSetStaticError(paramPtr, msg_send_originlost); -  /* The receiver for the event exited, before processing the -   * event. We detach the result now, wake the originator up -   * and signal failure. -   */ +    /* +     * See below: TclSpliceOut(resultPtr, forwardList); +     */ -  evPtr->resultPtr = NULL; -  resultPtr->evPtr  = NULL; -  resultPtr->result = TCL_ERROR; +    Tcl_MutexUnlock(&rcForwardMutex); -  RcForwardSetStaticError (paramPtr, msg_send_dstlost); +    /* +     * This unlocks (*). The structure will be spliced out and freed by +     * "ForwardProc". Maybe. +     */ -  Tcl_ConditionNotify(&resultPtr->done); +    Tcl_ConditionNotify(&resultPtr->done);  } -  static void -RcForwardSetObjError (p,obj) -     RcForwardParamBase* p; -     Tcl_Obj*            obj; +ForwardSetObjError( +    ForwardParam *paramPtr, +    Tcl_Obj *obj)  { -    int   len; -    char* msg; - -    msg = Tcl_GetStringFromObj (obj, &len); +    int len; +    const char *msgStr = Tcl_GetStringFromObj(obj, &len); -    p->code = TCL_ERROR; -    p->vol  = 1; -    p->msg  = strcpy(ckalloc (1+len), msg); +    len++; +    ForwardSetDynamicError(paramPtr, ckalloc(len)); +    memcpy(paramPtr->base.msgStr, msgStr, (unsigned) len);  }  #endif @@ -2659,5 +3232,7 @@ RcForwardSetObjError (p,obj)   * mode: c   * c-basic-offset: 4   * fill-column: 78 + * tab-width: 8 + * indent-tabs-mode: nil   * End:   */ | 
