summaryrefslogtreecommitdiffstats
path: root/win/tclAppInit.c
diff options
context:
space:
mode:
authorJoe Mistachkin <joe@mistachkin.com>2003-05-10 05:00:11 (GMT)
committerJoe Mistachkin <joe@mistachkin.com>2003-05-10 05:00:11 (GMT)
commit1a9fd777ca966fd412d3d0149ad3d4529ff76ea7 (patch)
tree89aaf34c6162c72ed27ac393e5da1403c202738b /win/tclAppInit.c
parentbd23f3d38d90df4d13ed707fec50e231ce016152 (diff)
downloadtcl-1a9fd777ca966fd412d3d0149ad3d4529ff76ea7.zip
tcl-1a9fd777ca966fd412d3d0149ad3d4529ff76ea7.tar.gz
tcl-1a9fd777ca966fd412d3d0149ad3d4529ff76ea7.tar.bz2
fix for [Bugs 733221, 733156]
Diffstat (limited to 'win/tclAppInit.c')
-rw-r--r--win/tclAppInit.c91
1 files changed, 70 insertions, 21 deletions
diff --git a/win/tclAppInit.c b/win/tclAppInit.c
index 864f59a..bed9ea8 100644
--- a/win/tclAppInit.c
+++ b/win/tclAppInit.c
@@ -1,4 +1,4 @@
-/*
+/*
* tclAppInit.c --
*
* Provides a default version of the main program and Tcl_AppInit
@@ -11,7 +11,7 @@
* See the file "license.terms" for information on usage and redistribution
* of this file, and for a DISCLAIMER OF ALL WARRANTIES.
*
- * RCS: @(#) $Id: tclAppInit.c,v 1.11 2002/12/04 03:59:17 davygrvy Exp $
+ * RCS: @(#) $Id: tclAppInit.c,v 1.11.2.1 2003/05/10 05:00:11 mistachkin Exp $
*/
#include "tcl.h"
@@ -31,9 +31,11 @@ extern int TclThread_Init _ANSI_ARGS_((Tcl_Interp *interp));
static void setargv _ANSI_ARGS_((int *argcPtr, char ***argvPtr));
static BOOL __stdcall sigHandler (DWORD fdwCtrlType);
static Tcl_AsyncProc asyncExit;
+static void AppInitExitHandler(ClientData clientData);
-Tcl_AsyncHandler exitToken;
-DWORD exitErrorCode;
+static char ** argvSave = NULL;
+static Tcl_AsyncHandler exitToken = NULL;
+static DWORD exitErrorCode = 0;
/*
@@ -64,18 +66,18 @@ main(argc, argv)
* of rewriting this entire file. The #if checks for that
* #define and uses Tcl_AppInit if it doesn't exist.
*/
-
+
#ifndef TCL_LOCAL_APPINIT
-#define TCL_LOCAL_APPINIT Tcl_AppInit
+#define TCL_LOCAL_APPINIT Tcl_AppInit
#endif
extern int TCL_LOCAL_APPINIT _ANSI_ARGS_((Tcl_Interp *interp));
-
+
/*
* The following #if block allows you to change how Tcl finds the startup
* script, prime the library or encoding paths, fiddle with the argv,
* etc., without needing to rewrite Tcl_Main()
*/
-
+
#ifdef TCL_LOCAL_MAIN_HOOK
extern int TCL_LOCAL_MAIN_HOOK _ANSI_ARGS_((int *argc, char ***argv));
#endif
@@ -91,6 +93,11 @@ main(argc, argv)
setargv(&argc, &argv);
/*
+ * Save this for later, so we can free it.
+ */
+ argvSave = argv;
+
+ /*
* Replace argv[0] with full pathname of executable, and forward
* slashes substituted for backslashes.
*/
@@ -143,8 +150,14 @@ Tcl_AppInit(interp)
/*
* Install a signal handler to the win32 console tclsh is running in.
*/
- SetConsoleCtrlHandler(sigHandler, TRUE);
- exitToken = Tcl_AsyncCreate(asyncExit, NULL);
+ SetConsoleCtrlHandler(sigHandler, TRUE);
+ exitToken = Tcl_AsyncCreate(asyncExit, NULL);
+
+ /*
+ * This exit handler will be used to free the
+ * resources allocated in this file.
+ */
+ Tcl_CreateExitHandler(AppInitExitHandler, NULL);
#ifdef TCL_TEST
if (Tcltest_Init(interp) == TCL_ERROR) {
@@ -212,12 +225,48 @@ Tcl_AppInit(interp)
}
/*
+ *----------------------------------------------------------------------
+ *
+ * AppInitExitHandler --
+ *
+ * This function is called to cleanup the app init resources before
+ * Tcl is unloaded.
+ *
+ * Results:
+ * None.
+ *
+ * Side effects:
+ * Frees the saved argv and deletes the async exit handler.
+ *
+ *----------------------------------------------------------------------
+ */
+
+static void
+AppInitExitHandler(
+ ClientData clientData)
+{
+ if (argvSave != NULL) {
+ ckfree((char *)argvSave);
+ argvSave = NULL;
+ }
+
+ if (exitToken != NULL) {
+ /*
+ * This should be safe to do even if we
+ * are in an async exit right now.
+ */
+ Tcl_AsyncDelete(exitToken);
+ exitToken = NULL;
+ }
+}
+
+/*
*-------------------------------------------------------------------------
*
* setargv --
*
* Parse the Windows command line string into argc/argv. Done here
- * because we don't trust the builtin argument parser in crt0.
+ * because we don't trust the builtin argument parser in crt0.
* Windows applications are responsible for breaking their command
* line into arguments.
*
@@ -246,7 +295,7 @@ setargv(argcPtr, argvPtr)
char *cmdLine, *p, *arg, *argSpace;
char **argv;
int argc, size, inquote, copy, slashes;
-
+
cmdLine = GetCommandLine(); /* INTL: BUG */
/*
@@ -327,7 +376,7 @@ setargv(argcPtr, argvPtr)
*argcPtr = argc;
*argvPtr = argv;
}
-
+
/*
*----------------------------------------------------------------------
*
@@ -352,7 +401,7 @@ asyncExit (ClientData clientData, Tcl_Interp *interp, int code)
/* NOTREACHED */
return code;
}
-
+
/*
*----------------------------------------------------------------------
*
@@ -385,16 +434,16 @@ sigHandler(DWORD fdwCtrlType)
exitErrorCode = fdwCtrlType;
Tcl_AsyncMark(exitToken);
- /*
- * This will cause Tcl_Gets in Tcl_Main() to drop-out with an <EOF>
- * should it be blocked on input and our Tcl_AsyncMark didn't grab
- * the attention of the interpreter.
+ /*
+ * This will cause Tcl_Gets in Tcl_Main() to drop-out with an <EOF>
+ * should it be blocked on input and our Tcl_AsyncMark didn't grab
+ * the attention of the interpreter.
*/
hStdIn = GetStdHandle(STD_INPUT_HANDLE);
if (hStdIn) {
CloseHandle(hStdIn);
}
- /* indicate to the OS not to call the default terminator */
- return TRUE;
-}
+ /* indicate to the OS not to call the default terminator */
+ return TRUE;
+}