From: Pat Thoyts
Date: Tue, 2 Apr 2002 23:39:07 +0000 (+0000)
Subject: Added support for multiple interpreters.
X-Git-Tag: winsend-0-4
X-Git-Url: https://privyetmir.co.uk/gitweb?a=commitdiff_plain;h=74081d2c1d6b25a47c0877842f2c26866b5837d7;p=winsend
Added support for multiple interpreters.
Added the appname subcommand to report and set the registered name.
Added in some debugging support functions.
Wrote some more documentation as a .man format file (see tcllib)
Added a second VB script demo.
Incremented the version to 0.4
---
diff --git a/.cvsignore b/.cvsignore
new file mode 100644
index 0000000..392bc9c
--- /dev/null
+++ b/.cvsignore
@@ -0,0 +1,5 @@
+.cvsignore
+Debug
+Release
+*.ilk
+*.dll
\ No newline at end of file
diff --git a/Makefile b/Makefile
index c9cace8..5d525a2 100644
--- a/Makefile
+++ b/Makefile
@@ -8,7 +8,10 @@
#
# @(#)$Id$
-VER =03
+#TCLROOT ="c:/Program Files/Tcl"
+TCLROOT =/opt/tcl
+
+VER =04
DBGX =d
DBGFLAGS=-D_DEBUG
@@ -16,8 +19,8 @@ CC =gcc -g
DLLWRAP =dllwrap
DLLTOOL =dlltool
RM =rm -f
-CFLAGS =-Wall -I/opt/tcl/include -DUSE_TCL_STUBS $(DBGFLAGS)
-LDFLAGS =-L/opt/tcl/lib
+CFLAGS =-Wall -I$(TCLROOT)/include -DUSE_TCL_STUBS $(DBGFLAGS)
+LDFLAGS =-L$(TCLROOT)/lib
LIBS =-ltclstub83${DBGX} -lole32 -loleaut32 -ladvapi32 -luuid
DLL =winsend${VER}${DBGX}.dll
@@ -25,7 +28,7 @@ DEFFILE =winsend.def
WRAPFLAGS =--driver-name $(CC) --def $(DEFFILE)
-CSRCS =winsend.c WinSendCom.c
+CSRCS =winsend.c WinSendCom.c debug.c
OBJS =$(CSRCS:.c=.o)
$(DLL): $(OBJS)
@@ -37,7 +40,9 @@ clean:
%.o: %.c
$(CC) $(CFLAGS) -c $< -o $@
+winsend.o: WinSendCom.h debug.h
WinSendCom.o: WinSendCom.c WinSendCom.h
+debug.o: debug.h
.PHONY: clean
diff --git a/Readme b/Readme
index c1465a6..26828fe 100644
--- a/Readme
+++ b/Readme
@@ -47,11 +47,10 @@ WScript.Echo interp.Send("info tcl")
You should see the message printed in your tkcon window and see your
Tcl version printed in your DOS box.
-The code needs tidying up a bit before I publish the source but this
-shouldn't take long. In the meantime see
+See
http://tclsoap.sourceforge.net/winsend.html
and
- http://tclsoap.sourceforge.net/winsend03.zip
+ http://tclsoap.sourceforge.net/winsend.zip
for the docs (this document) and code respectively.
Pat Thoyts.
diff --git a/WinSendCom.c b/WinSendCom.c
index e0167e1..53f0bae 100644
--- a/WinSendCom.c
+++ b/WinSendCom.c
@@ -13,9 +13,10 @@ static const char rcsid[] =
#include "WinSendCom.h"
-/* ---------------------------------------------------------------------- */
-/* Non-public prototypes.
-/* ---------------------------------------------------------------------- */
+/* ----------------------------------------------------------------------
+ * Non-public prototypes.
+ * ----------------------------------------------------------------------
+ */
static STDMETHODIMP WinSendCom_QueryInterface(IDispatch *This,
REFIID riid, void **ppvObject);
@@ -37,9 +38,10 @@ static STDMETHODIMP WinSendCom_Invoke(IDispatch *This, DISPID dispidMember,
UINT *puArgErr);
static HRESULT Send(WinSendCom* obj, VARIANT vCmd, VARIANT* pvResult);
-/* ---------------------------------------------------------------------- */
-/* COM Class Helpers
-/* ---------------------------------------------------------------------- */
+/* ----------------------------------------------------------------------
+ * COM Class Helpers
+ * ----------------------------------------------------------------------
+ */
/* Description:
* Create and initialises a new instance of the WinSend COM class and
diff --git a/WinSendCom.h b/WinSendCom.h
index efefb8e..d8a4efb 100644
--- a/WinSendCom.h
+++ b/WinSendCom.h
@@ -18,7 +18,7 @@ typedef struct WinSendCom_t {
} WinSendCom;
/*
- * WinSendCom public methods
+ * WinSendCom public functions
*/
HRESULT WinSendCom_CreateInstance(Tcl_Interp *interp, REFIID riid, void **ppv);
void WinSendCom_Destroy(LPDISPATCH pdisp);
diff --git a/debug.c b/debug.c
new file mode 100644
index 0000000..fb64b78
--- /dev/null
+++ b/debug.c
@@ -0,0 +1,17 @@
+#include "debug.h"
+#include
+
+void
+LocalTrace(LPCTSTR format, ...)
+{
+ int n;
+ const int max = sizeof(TCHAR) * 512;
+ TCHAR buffer[512];
+ va_list args;
+ va_start (args, format);
+
+ n = _vsntprintf(buffer, max, format, args);
+ _ASSERTE(n < max);
+ OutputDebugString(buffer);
+ va_end(args);
+}
diff --git a/debug.h b/debug.h
new file mode 100644
index 0000000..0d868ad
--- /dev/null
+++ b/debug.h
@@ -0,0 +1,30 @@
+/* debug.h - Copyright (C) 2002 Pat Thoyts
+ *
+ * $Id$
+ */
+
+#ifndef _Debug_h_INCLUDE
+#define _Debug_h_INCLUDE
+
+#include
+#include
+#include
+#include
+
+/* Debug related stuff */
+#ifndef _DEBUG
+
+#define _ASSERTE(x) ((void)0)
+#define LTRACE 1 ? ((void)0) : LocalTrace
+
+#else /* _DEBUG */
+
+#define _ASSERTE(x) if (!(x)) _assert(#x, __FILE__, __LINE__)
+#define LTRACE LocalTrace
+
+#endif /* _DEBUG */
+
+/* Debug functions. */
+void LocalTrace(LPCTSTR szFormat, ...);
+
+#endif /* _Debug_h_INCLUDE */
\ No newline at end of file
diff --git a/pkgIndex.tcl b/pkgIndex.tcl
index 9efa827..ef069c1 100644
--- a/pkgIndex.tcl
+++ b/pkgIndex.tcl
@@ -9,8 +9,8 @@
# full path name of this file's directory.
if {[info exists ::tcl_platform(debug)] && $::tcl_platform(debug)} {
- package ifneeded winsend 0.3 [list load [file join $dir winsend03d.dll]]
+ package ifneeded winsend 0.4 [list load [file join $dir winsend04d.dll]]
} else {
- package ifneeded winsend 0.3 [list load [file join $dir winsend03.dll]]
+ package ifneeded winsend 0.4 [list load [file join $dir winsend04.dll]]
}
diff --git a/test.vbs b/test.vbs
index 51ac78d..f602394 100644
--- a/test.vbs
+++ b/test.vbs
@@ -1,10 +1,3 @@
Dim o
-Set o = GetObject("\\.\TclInterp\tkcon.tcl")
-WScript.Echo o.Send("puts {Hello, from VB} ; tkcon master wm title .")
-
-Dim p
-Set p = GetObject("\\.\TclInterp\tkcon.tcl #2")
-WScript.Echo p.Send("puts {Hello, 2}; tkcon master wm title .")
-
-
-
+Set o = GetObject("\\.\TclInterp\tkchat")
+WScript.Echo o.Send("puts {Hello, from VB} ; info tcl")
diff --git a/test2.vbs b/test2.vbs
new file mode 100644
index 0000000..614b8f3
--- /dev/null
+++ b/test2.vbs
@@ -0,0 +1,21 @@
+' test2.vbs - Copyright (C) 2002 Pat Thoyts
+'
+' Demo accessing Tcl using the winsend package from the Visual Basic
+' Scripting engine via the Windown Scripting Host (WSH).
+'
+' Run this using 'cscript test2.vbs' or 'wscript test2.vbs' when you
+' have a wish session using the winsend appname wish.
+'
+' $Id$
+
+Dim o
+Set o = GetObject("\\.\TclInterp\wish")
+
+cmd = "package require Tk"
+cmd = cmd & vbNewline & "toplevel .t"
+cmd = cmd & vbNewline & "button .t.b -text OK -command {destroy .t}"
+cmd = cmd & vbNewline & "pack .t.b -side top -fill both"
+cmd = cmd & vbNewline & "tkwait window .t"
+cmd = cmd & vbNewline & "info tcl"
+
+WScript.Echo o.Send(cmd)
diff --git a/winsend.c b/winsend.c
index ba58467..8c6aa24 100644
--- a/winsend.c
+++ b/winsend.c
@@ -19,6 +19,7 @@ static const char rcsid[] =
#include
#include "WinSendCom.h"
+#include "debug.h"
#ifndef DECLSPEC_EXPORT
#define DECLSPEC_EXPORT __declspec(dllexport)
@@ -26,23 +27,36 @@ static const char rcsid[] =
/* Should be defined in WTypes.h but mingw 1.0 is missing them */
#ifndef _ROTFLAGS_DEFINED
+#define _ROTFLAGS_DEFINED
#define ROTFLAGS_REGISTRATIONKEEPSALIVE 0x01
#define ROTFLAGS_ALLOWANYCLIENT 0x02
#endif /* ! _ROTFLAGS_DEFINED */
-#define WINSEND_PACKAGE_VERSION "0.3"
+#define WINSEND_PACKAGE_VERSION "0.4"
#define WINSEND_PACKAGE_NAME "winsend"
#define WINSEND_CLASS_NAME "TclEval"
-DWORD g_dwROTCookie;
-
-static void Winsend_PkgDeleteProc _ANSI_ARGS_((ClientData clientData));
+/* Package information structure.
+ * This is used to keep interpreter specific details for use when releasing
+ * the package resources upon interpreter deletion or package removal.
+ */
+typedef struct WinsendPkg_t {
+ Tcl_Obj *appname; /* the registered application name */
+ DWORD ROT_cookie; /* ROT cookie returned on registration */
+ LPUNKNOWN obj; /* Interface for the registration object */
+ Tcl_Command token; /* Winsend command token */
+} WinsendPkg;
+
+static void Winsend_InterpDeleteProc (ClientData clientData, Tcl_Interp *interp);
+static void Winsend_PkgDeleteProc (ClientData clientData);
static int Winsend_CmdProc(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[]);
static int Winsend_CmdInterps(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[]);
static int Winsend_CmdSend(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[]);
+static int Winsend_CmdAppname(ClientData clientData, Tcl_Interp *interp,
+ int objc, Tcl_Obj *CONST objv[]);
static int Winsend_CmdTest(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[]);
static int Winsend_ObjSendCmd(LPDISPATCH pdispInterp, Tcl_Interp *interp,
@@ -50,9 +64,10 @@ static int Winsend_ObjSendCmd(LPDISPATCH pdispInterp, Tcl_Interp *interp,
static HRESULT BuildMoniker(LPCOLESTR name, LPMONIKER *pmk);
static Tcl_Obj* Winsend_Win32ErrorObj(HRESULT hrError);
-// -------------------------------------------------------------------------
-// DllMain
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * DllMain
+ * -------------------------------------------------------------------------
+ */
EXTERN_C BOOL APIENTRY
DllMain(HANDLE hModule, DWORD dwReason, LPVOID lpReserved)
@@ -68,30 +83,38 @@ DllMain(HANDLE hModule, DWORD dwReason, LPVOID lpReserved)
return TRUE;
}
-// -------------------------------------------------------------------------
-// Winsend_Init
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * Winsend_Init
+ * -------------------------------------------------------------------------
+ */
EXTERN_C int DECLSPEC_EXPORT
Winsend_Init(Tcl_Interp* interp)
{
HRESULT hr = S_OK;
int r = TCL_OK;
- IUnknown *pUnk = NULL;
+ WinsendPkg *pkg = NULL;
#ifdef USE_TCL_STUBS
Tcl_InitStubs(interp, "8.3", 0);
#endif
- // Initialize COM
+ pkg = (WinsendPkg*)Tcl_Alloc(sizeof(WinsendPkg));
+ if (pkg == NULL) {
+ Tcl_SetResult(interp, "out of memory", TCL_STATIC);
+ return TCL_ERROR;
+ }
+ memset(pkg, 0, sizeof(WinsendPkg));
+
+ /* Initialize COM */
hr = CoInitialize(0);
if (FAILED(hr)) {
Tcl_SetResult(interp, "failed to initialize the " WINSEND_PACKAGE_NAME " package", TCL_STATIC);
return TCL_ERROR;
}
- // Create our registration object.
- hr = WinSendCom_CreateInstance(interp, &IID_IUnknown, (void**)&pUnk);
+ /* Create our registration object. */
+ hr = WinSendCom_CreateInstance(interp, &IID_IUnknown, (void**)&pkg->obj);
if (SUCCEEDED(hr))
{
@@ -123,9 +146,14 @@ Winsend_Init(Tcl_Interp* interp)
if (SUCCEEDED(hr)) {
hr = pROT->lpVtbl->Register(pROT,
- ROTFLAGS_REGISTRATIONKEEPSALIVE
- | ROTFLAGS_ALLOWANYCLIENT,
- pUnk, pmk, &g_dwROTCookie);
+ ROTFLAGS_REGISTRATIONKEEPSALIVE,
+ pkg->obj,
+ pmk,
+ &pkg->ROT_cookie);
+ if (SUCCEEDED(hr)) {
+ pkg->appname = Tcl_NewUnicodeObj(oleName, -1);
+ Tcl_IncrRefCount(pkg->appname);
+ }
pmk->lpVtbl->Release(pmk);
}
@@ -133,32 +161,33 @@ Winsend_Init(Tcl_Interp* interp)
* try again.
*/
if (hr == MK_S_MONIKERALREADYREGISTERED)
- pROT->lpVtbl->Revoke(pROT, g_dwROTCookie);
+ pROT->lpVtbl->Revoke(pROT, pkg->ROT_cookie);
} while (hr == MK_S_MONIKERALREADYREGISTERED);
pROT->lpVtbl->Release(pROT);
}
- pUnk->lpVtbl->Release(pUnk);
-
- // Create our winsend command
+ /* Create our winsend command */
if (SUCCEEDED(hr)) {
- Tcl_CreateObjCommand(interp, "winsend", Winsend_CmdProc, (ClientData)0, (Tcl_CmdDeleteProc*)0);
+ pkg->token = Tcl_CreateObjCommand(interp,
+ "winsend",
+ Winsend_CmdProc,
+ (ClientData)pkg,
+ (Tcl_CmdDeleteProc*)NULL);
}
/* Create an exit procedure to handle unregistering when the
* Tcl interpreter terminates.
*/
- Tcl_CreateExitHandler(Winsend_PkgDeleteProc, NULL);
+ Tcl_CallWhenDeleted(interp, Winsend_InterpDeleteProc, (ClientData)pkg);
+ Tcl_CreateExitHandler(Winsend_PkgDeleteProc, (ClientData)pkg);
r = Tcl_PkgProvide(interp, WINSEND_PACKAGE_NAME, WINSEND_PACKAGE_VERSION);
}
if (FAILED(hr))
{
- // TODO: better error handling - rip code from w32_exception.h
- Tcl_Obj *err = Winsend_Win32ErrorObj(hr);
- Tcl_SetObjResult(interp, err);
+ Tcl_SetObjResult(interp, Winsend_Win32ErrorObj(hr));
r = TCL_ERROR;
} else {
Tcl_SetResult(interp, "", TCL_STATIC);
@@ -167,9 +196,9 @@ Winsend_Init(Tcl_Interp* interp)
return r;
}
-// -------------------------------------------------------------------------
-// Winsend_SafeInit
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * Winsend_SafeInit
+ * ------------------------------------------------------------------------- */
EXTERN_C int DECLSPEC_EXPORT
Winsend_SafeInit(Tcl_Interp* interp)
@@ -178,33 +207,100 @@ Winsend_SafeInit(Tcl_Interp* interp)
return TCL_ERROR;
}
-// -------------------------------------------------------------------------
-// WinsendExitProc
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * Winsend_InterpDeleteProc
+ * -------------------------------------------------------------------------
+ * Description:
+ * Called when the interpreter is deleted, this procedure clean up the COM
+ * registration for us. We need to revoke our registered object, and release
+ * our own object reference. The WinSendCom object should delete itself now.
+ */
-static void
-Winsend_PkgDeleteProc(ClientData clientData)
+static void
+Winsend_InterpDeleteProc (ClientData clientData, Tcl_Interp *interp)
{
+ WinsendPkg *pkg = (WinsendPkg*)clientData;
LPRUNNINGOBJECTTABLE pROT = NULL;
- HRESULT hr = GetRunningObjectTable(0, &pROT);
- if (SUCCEEDED(hr))
- {
- hr = pROT->lpVtbl->Revoke(pROT, g_dwROTCookie);
- pROT->lpVtbl->Release(pROT);
+ HRESULT hr = S_OK;
+
+ LTRACE(_T("Winsend_InterpDeleteProc( {ROT_cookie: 0x%08X, obj: 0x%08X})\n"),
+ pkg->ROT_cookie, pkg->obj);
+
+ /* Lock the package structure in memory */
+ Tcl_Preserve((ClientData)pkg);
+
+ if (pkg->ROT_cookie != 0) {
+ hr = GetRunningObjectTable(0, &pROT);
+ if (SUCCEEDED(hr))
+ {
+ hr = pROT->lpVtbl->Revoke(pROT, pkg->ROT_cookie);
+ pROT->lpVtbl->Release(pROT);
+ pkg->ROT_cookie = 0;
+ }
+ _ASSERTE(SUCCEEDED(hr));
}
- //ASSERT
+
+ /* Remove the winsend command and release the interp pointer */
+ /* I get assertions from this from within tcl 8.3 - it's not
+ * really needed anyway
+ */
+ /* if (interp != NULL) {
+ Tcl_DeleteCommandFromToken(interp, pkg->token);
+ pkg->token = 0;
+ }*/
+
+ /* release the appname object */
+ if (pkg->appname != NULL) {
+ Tcl_DecrRefCount(pkg->appname);
+ pkg->appname = NULL;
+ }
+
+ /* Release the registration object */
+ pkg->obj->lpVtbl->Release(pkg->obj);
+ pkg->obj = NULL;
+
+ /* unlock the package data structure. */
+ Tcl_Release((ClientData)pkg);
}
-// -------------------------------------------------------------------------
-// Winsend_CmdProc
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * Winsend_PkgDeleteProc
+ * -------------------------------------------------------------------------
+ * Description:
+ * Called when the package is removed, we clean up any outstanding memory
+ * and release any handles.
+ */
+static void
+Winsend_PkgDeleteProc(ClientData clientData)
+{
+ WinsendPkg *pkg = (WinsendPkg*)clientData;
+ LTRACE(_T("Winsend_PkgDeleteProc( {ROT_cookie: 0x%08X, obj: 0x%08X})\n"),
+ pkg->ROT_cookie, pkg->obj);
+
+ if (pkg->obj != NULL) {
+ Winsend_InterpDeleteProc(clientData, NULL);
+ }
+
+ _ASSERTE(pkg->obj == NULL);
+ Tcl_Free((char*)pkg);
+
+ CoUninitialize();
+}
+
+/* -------------------------------------------------------------------------
+ * Winsend_CmdProc
+ * -------------------------------------------------------------------------
+ * Description:
+ * The Tcl winsend proc is first processed here and then passed on depending
+ * upong the command argument.
+ */
static int
Winsend_CmdProc(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[])
{
- enum {WINSEND_INTERPS, WINSEND_SEND, WINSEND_TEST};
- static char* cmds[] = { "interps", "send", "test", NULL };
+ enum {WINSEND_INTERPS, WINSEND_SEND, WINSEND_APPNAME, WINSEND_TEST};
+ static char* cmds[] = { "interps", "send", "appname", "test", NULL };
int index = 0, r = TCL_OK;
if (objc < 2) {
@@ -223,6 +319,9 @@ Winsend_CmdProc(ClientData clientData, Tcl_Interp *interp,
case WINSEND_SEND:
r = Winsend_CmdSend(clientData, interp, objc, objv);
break;
+ case WINSEND_APPNAME:
+ r = Winsend_CmdAppname(clientData, interp, objc, objv);
+ break;
case WINSEND_TEST:
r = Winsend_CmdTest(clientData, interp, objc, objv);
break;
@@ -231,10 +330,14 @@ Winsend_CmdProc(ClientData clientData, Tcl_Interp *interp,
return r;
}
-// -------------------------------------------------------------------------
-// Winsend_CmdInterps
-// -------------------------------------------------------------------------
-
+/* -------------------------------------------------------------------------
+ * Winsend_CmdInterps
+ * -------------------------------------------------------------------------
+ * Description:
+ * Iterate over the running object table and identify all the Tcl registered
+ * objects. We build a list using the tail part of the file moniker and return
+ * this as the list of registered interpreters.
+ */
static int
Winsend_CmdInterps(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[])
@@ -274,8 +377,9 @@ Winsend_CmdInterps(ClientData clientData, Tcl_Interp *interp,
if (wcsncmp(olestr, oleszStub, wcslen(oleszStub)) == 0)
{
- LPOLESTR p = olestr + wcslen(oleszStub) + 1;
- r = Tcl_ListObjAppendElement(interp, objList, Tcl_NewUnicodeObj(p, -1));
+ LPOLESTR p = olestr + wcslen(oleszStub);
+ if (*p)
+ r = Tcl_ListObjAppendElement(interp, objList, Tcl_NewUnicodeObj(p + 1, -1));
}
hr = CoGetMalloc(1, &pMalloc);
@@ -310,10 +414,10 @@ Winsend_CmdInterps(ClientData clientData, Tcl_Interp *interp,
return r;
}
-// -------------------------------------------------------------------------
-// Winsend_CmdSend
-// -------------------------------------------------------------------------
-
+/* -------------------------------------------------------------------------
+ * Winsend_CmdSend
+ * -------------------------------------------------------------------------
+ */
static int
Winsend_CmdSend(ClientData clientData, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[])
@@ -345,7 +449,6 @@ Winsend_CmdSend(ClientData clientData, Tcl_Interp *interp,
IDispatch* pdispInterp = NULL;
hr = pROT->lpVtbl->IsRunning(pROT, pmk);
hr = pmk->lpVtbl->BindToObject(pmk, pBindCtx, NULL, &IID_IUnknown, (void**)&punkInterp);
- //hr = pROT->lpVtbl->GetObject(pROT, pmk, &punkInterp);
if (SUCCEEDED(hr))
hr = punkInterp->lpVtbl->QueryInterface(punkInterp, &IID_IDispatch, (void**)&pdispInterp);
if (SUCCEEDED(hr))
@@ -370,6 +473,80 @@ Winsend_CmdSend(ClientData clientData, Tcl_Interp *interp,
return r;
}
+/* -------------------------------------------------------------------------
+ * Winsend_CmdAppname
+ * -------------------------------------------------------------------------
+ * Description:
+ * If called with no additional parameters returns the registered application
+ * name. If a new name is given then we attempt to register the new app name
+ * If the name is already registered then we cancel the new registration and
+ * continue using the old appname.
+ */
+static int
+Winsend_CmdAppname(ClientData clientData, Tcl_Interp *interp,
+ int objc, Tcl_Obj *CONST objv[])
+{
+ int r = TCL_OK;
+ HRESULT hr = S_OK;
+ WinsendPkg* pkg = (WinsendPkg*)clientData;
+
+ if (objc == 2) {
+
+ Tcl_SetObjResult(interp, Tcl_DuplicateObj(pkg->appname));
+
+ } else if (objc > 3) {
+
+ Tcl_WrongNumArgs(interp, 2, objv, "?appname?");
+ r = TCL_ERROR;
+
+ } else {
+
+ LPRUNNINGOBJECTTABLE pROT = NULL;
+ hr = GetRunningObjectTable(0, &pROT);
+ if (SUCCEEDED(hr)) {
+ LPMONIKER pmk;
+ LPOLESTR szNewName;
+
+ szNewName = Tcl_GetUnicode(objv[2]);
+
+ /* construct a new moniker */
+ hr = BuildMoniker(szNewName, &pmk);
+ if (SUCCEEDED(hr)) {
+ DWORD cookie = 0;
+
+ /* register the new name */
+ hr = pROT->lpVtbl->Register(pROT, ROTFLAGS_REGISTRATIONKEEPSALIVE, pkg->obj, pmk, &cookie);
+ if (SUCCEEDED(hr)) {
+ if (hr == MK_S_MONIKERALREADYREGISTERED) {
+ pROT->lpVtbl->Revoke(pROT, cookie);
+ Tcl_SetObjResult(interp, Winsend_Win32ErrorObj(hr));
+ r = TCL_ERROR;
+ } else {
+
+ /* revoke the old name */
+ hr = pROT->lpVtbl->Revoke(pROT, pkg->ROT_cookie);
+ if (SUCCEEDED(hr)) {
+
+ /* update the package structure */
+ pkg->ROT_cookie = cookie;
+ Tcl_SetUnicodeObj(pkg->appname, szNewName, -1);
+ Tcl_SetObjResult(interp, pkg->appname);
+ }
+ }
+ }
+ pmk->lpVtbl->Release(pmk);
+ }
+ pROT->lpVtbl->Release(pROT);
+ }
+ if (FAILED(hr))
+ {
+ Tcl_SetObjResult(interp, Winsend_Win32ErrorObj(hr));
+ r = TCL_ERROR;
+ }
+ }
+ return r;
+}
+
/* -------------------------------------------------------------------------
* Winsend_CmdTest
* -------------------------------------------------------------------------
@@ -404,15 +581,19 @@ Winsend_CmdTest(ClientData clientData, Tcl_Interp *interp,
return r;
}
-// -------------------------------------------------------------------------
-// Helpers.
-// -------------------------------------------------------------------------
+/* -------------------------------------------------------------------------
+ * Helpers.
+ * -------------------------------------------------------------------------
+ */
-// -------------------------------------------------------------------------
-// Winsend_ObjSendCmd
-// -------------------------------------------------------------------------
-// Description:
-//
+/* -------------------------------------------------------------------------
+ * Winsend_ObjSendCmd
+ * -------------------------------------------------------------------------
+ * Description:
+ * Perform an interface call to the server object. We convert the Tcl arguments
+ * into a BSTR using 'concat'. The result should be a BSTR that we can set as the
+ * interp's result string.
+ */
static int
Winsend_ObjSendCmd(LPDISPATCH pdispInterp, Tcl_Interp *interp,
int objc, Tcl_Obj *CONST objv[])
@@ -450,7 +631,13 @@ Winsend_ObjSendCmd(LPDISPATCH pdispInterp, Tcl_Interp *interp,
return (SUCCEEDED(hr) ? TCL_OK : TCL_ERROR);
}
-// Construct a Tcl interpreter moniker from an interp name.
+/* -------------------------------------------------------------------------
+ * BuildMoniker
+ * -------------------------------------------------------------------------
+ * Description:
+ * Construct a moniker from the given name. This ensures that all our
+ * monikers have the same prefix.
+ */
static HRESULT
BuildMoniker(LPCOLESTR name, LPMONIKER *pmk)
{
@@ -475,6 +662,12 @@ BuildMoniker(LPCOLESTR name, LPMONIKER *pmk)
return hr;
}
+/* -------------------------------------------------------------------------
+ * Winsend_Win32ErrorObj
+ * -------------------------------------------------------------------------
+ * Description:
+ * Convert COM or Win32 API errors into Tcl strings.
+ */
static Tcl_Obj*
Winsend_Win32ErrorObj(HRESULT hrError)
{
diff --git a/winsend.dsp b/winsend.dsp
index 36bb995..9d143a8 100644
--- a/winsend.dsp
+++ b/winsend.dsp
@@ -53,7 +53,7 @@ BSC32=bscmake.exe
# ADD BSC32 /nologo
LINK32=link.exe
# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386
-# ADD LINK32 tclstub83.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386 /out:"winsend03.dll"
+# ADD LINK32 tclstub83.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386 /out:"winsend04.dll"
!ELSEIF "$(CFG)" == "winsend - Win32 Debug"
@@ -68,8 +68,8 @@ LINK32=link.exe
# PROP Intermediate_Dir "Debug"
# PROP Ignore_Export_Lib 0
# PROP Target_Dir ""
-# ADD BASE CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /D "WIN32" /D "_DEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "WINSEND_EXPORTS" /YX /FD /GZ /c
-# ADD CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /D "_DEBUG" /D "WIN32" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "WINSEND_EXPORTS" /D "USE_TCL_STUBS" /YX /FD /GZ /c
+# ADD BASE CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /D "WIN32" /D "_DEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "WINSEND_EXPORTS" /YX /FD /GZ /c
+# ADD CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /D "_DEBUG" /D "WIN32" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "WINSEND_EXPORTS" /D "USE_TCL_STUBS" /YX /FD /GZ /c
# ADD BASE MTL /nologo /D "_DEBUG" /mktyplib203 /win32
# ADD MTL /nologo /D "_DEBUG" /mktyplib203 /win32
# ADD BASE RSC /l 0x809 /d "_DEBUG"
@@ -79,7 +79,7 @@ BSC32=bscmake.exe
# ADD BSC32 /nologo
LINK32=link.exe
# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /debug /machine:I386 /pdbtype:sept
-# ADD LINK32 tclstub83d.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /debug /machine:I386 /out:"winsend03d.dll" /pdbtype:sept
+# ADD LINK32 tclstub83d.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /debug /machine:I386 /out:"winsend04d.dll" /pdbtype:sept
!ENDIF
@@ -92,6 +92,10 @@ LINK32=link.exe
# PROP Default_Filter "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat"
# Begin Source File
+SOURCE=.\debug.c
+# End Source File
+# Begin Source File
+
SOURCE=.\winsend.c
# End Source File
# Begin Source File
@@ -108,6 +112,10 @@ SOURCE=.\WinSendCom.c
# PROP Default_Filter "h;hpp;hxx;hm;inl"
# Begin Source File
+SOURCE=.\debug.h
+# End Source File
+# Begin Source File
+
SOURCE=.\WinSendCom.h
# End Source File
# End Group
diff --git a/winsend.html b/winsend.html
index e5376fe..0e229d9 100644
--- a/winsend.html
+++ b/winsend.html
@@ -58,8 +58,8 @@ WScript.Echo interp.Send("info tcl")
Tcl version printed in your DOS box.
The package containing binaries and source is available from
-
-http://tclsoap.sf.net/winsend03.zip
+
+http://tclsoap.sf.net/winsend.zip
Pat Thoyts.
diff --git a/winsend.man b/winsend.man
new file mode 100644
index 0000000..09b485e
--- /dev/null
+++ b/winsend.man
@@ -0,0 +1,56 @@
+[manpage_begin winsend n 0.4]
+[moddesc {winsend}]
+[titledesc {Windows send Command}]
+[require Tcl 8.3]
+[require winsend [opt 0.4]]
+[description]
+[para]
+
+The [cmd winsend] package is an attempt to provide the complete
+functionality of the Tk [cmd send] command using the Microsoft Windows
+platform. The two key points are inter-process communications between
+one interpreter and another, and automatic registration of
+interpreters with a central source. With the Tk send command this is
+done via the X Windows server. The [cmd winfo interps] command returns
+a list of registered interpreters and the [cmd send] command allows
+you to execute Tcl commands in the named interpreter.
+
+[section COMMANDS]
+
+[list_begin definitions]
+
+[call [cmd winsend] [arg command] [opt [arg "arguments ..."]]]
+
+[list_begin definitions]
+[lst_item "[cmd interps]"]
+ Get the list of registered interpreter names for use with send.
+[lst_item "[cmd appname] [opt [arg name]]"]
+ When called with no arguments this returns the registered name of
+this interpreter. If a single argument is given, this is used to
+re-register the application using the new name. If the registration
+fails for some reason, such as the name is already in use, then an
+error is returned and the old appname remains in use.
+[lst_item "[cmd send]"]
+ send
+[list_end]
+
+[list_end]
+
+[section EXAMPLES]
+
+[para]
+[example {
+% package require winsend
+0.4
+% winsend appname MyApp
+MyApp
+% winsend send MyApp info tcl
+8.3
+}]
+
+[see_also send(n)]
+[section AUTHORS]
+Pat Thoyts
+
+[keywords send appname]
+[manpage_end]