diff options
Diffstat (limited to 'unix/tclUnixFile.c')
| -rw-r--r-- | unix/tclUnixFile.c | 104 | 
1 files changed, 55 insertions, 49 deletions
| diff --git a/unix/tclUnixFile.c b/unix/tclUnixFile.c index a4426b7..2cb0027 100644 --- a/unix/tclUnixFile.c +++ b/unix/tclUnixFile.c @@ -42,27 +42,18 @@ TclpFindExecutable(      Tcl_Encoding encoding;  #ifdef __CYGWIN__      int length; -    char buf[PATH_MAX * TCL_UTF_MAX + 1]; +    char buf[PATH_MAX * 2];      char name[PATH_MAX * TCL_UTF_MAX + 1]; - -    /* Make some symbols available without including <windows.h> */ -#   define CP_UTF8 65001 -    DLLIMPORT extern int cygwin_conv_to_full_posix_path(const char *, char *); -    DLLIMPORT extern __stdcall int GetModuleFileNameW(void *, const char *, int); -    DLLIMPORT extern __stdcall int WideCharToMultiByte(int, int, const char *, int, -		const char *, int, const char *, const char *); - -    GetModuleFileNameW(NULL, name, PATH_MAX); -    WideCharToMultiByte(CP_UTF8, 0, name, -1, buf, PATH_MAX, NULL, NULL); -    cygwin_conv_to_full_posix_path(buf, name); +    GetModuleFileNameW(NULL, buf, PATH_MAX); +    cygwin_conv_path(3, buf, name, PATH_MAX);      length = strlen(name);      if ((length > 4) && !strcasecmp(name + length - 4, ".exe")) {  	/* Strip '.exe' part. */  	length -= 4;      } -	encoding = Tcl_GetEncoding(NULL, NULL); -	TclSetObjNameOfExecutable( -		Tcl_NewStringObj(name, length), encoding); +    encoding = Tcl_GetEncoding(NULL, NULL); +    TclSetObjNameOfExecutable( +	    Tcl_NewStringObj(name, length), encoding);  #else      const char *name, *p;      Tcl_StatBuf statBuf; @@ -114,11 +105,11 @@ TclpFindExecutable(  	while ((*p != ':') && (*p != 0)) {  	    p++;  	} -	Tcl_DStringSetLength(&buffer, 0); +	TclDStringClear(&buffer);  	if (p != name) {  	    Tcl_DStringAppend(&buffer, name, p - name);  	    if (p[-1] != '/') { -		Tcl_DStringAppend(&buffer, "/", 1); +		TclDStringAppendLiteral(&buffer, "/");  	    }  	}  	name = Tcl_DStringAppend(&buffer, argv0, -1); @@ -183,11 +174,10 @@ TclpFindExecutable(      Tcl_UtfToExternalDString(NULL, Tcl_DStringValue(&cwd),  	    Tcl_DStringLength(&cwd), &buffer);      if (Tcl_DStringValue(&cwd)[Tcl_DStringLength(&cwd) -1] != '/') { -	Tcl_DStringAppend(&buffer, "/", 1); +	TclDStringAppendLiteral(&buffer, "/");      }      Tcl_DStringFree(&cwd); -    Tcl_DStringAppend(&buffer, Tcl_DStringValue(&nameString), -	    Tcl_DStringLength(&nameString)); +    TclDStringAppendDString(&buffer, &nameString);      Tcl_DStringFree(&nameString);      encoding = Tcl_GetEncoding(NULL, NULL); @@ -297,7 +287,7 @@ TclpMatchInDirectory(  	     */  	    if (dirName[dirLength-1] != '/') { -		dirName = Tcl_DStringAppend(&dsOrig, "/", 1); +		dirName = TclDStringAppendLiteral(&dsOrig, "/");  		dirLength++;  	    }  	} @@ -320,10 +310,9 @@ TclpMatchInDirectory(  	if (d == NULL) {  	    Tcl_DStringFree(&ds);  	    if (interp != NULL) { -		Tcl_ResetResult(interp); -		Tcl_AppendResult(interp, "couldn't read directory \"", -			Tcl_DStringValue(&dsOrig), "\": ", -			Tcl_PosixError(interp), NULL); +		Tcl_SetObjResult(interp, Tcl_ObjPrintf( +			"couldn't read directory \"%s\": %s", +			Tcl_DStringValue(&dsOrig), Tcl_PosixError(interp)));  	    }  	    Tcl_DStringFree(&dsOrig);  	    Tcl_DecrRefCount(fileNamePtr); @@ -481,7 +470,7 @@ NativeMatchType(  #ifndef MAC_OSX_TCL  	    || ((types->perm & TCL_GLOB_PERM_HIDDEN) &&  		(*nativeName != '.')) -#endif +#endif /* MAC_OSX_TCL */  		) {  	    return 0;  	} @@ -499,12 +488,10 @@ NativeMatchType(  		 * check that here:  		 */ -		if (types->type & TCL_GLOB_TYPE_LINK) { -		    if (TclOSlstat(nativeEntry, &buf) == 0) { -			if (S_ISLNK(buf.st_mode)) { -			    return 1; -			} -		    } +		if ((types->type & TCL_GLOB_TYPE_LINK) +			&& (TclOSlstat(nativeEntry, &buf) == 0) +			&& S_ISLNK(buf.st_mode)) { +		    return 1;  		}  		return 0;  	    } @@ -527,12 +514,10 @@ NativeMatchType(  	     */  	} else {  #ifdef S_ISLNK -	    if (types->type & TCL_GLOB_TYPE_LINK) { -		if (TclOSlstat(nativeEntry, &buf) == 0) { -		    if (S_ISLNK(buf.st_mode)) { -			goto filetypeOK; -		    } -		} +	    if ((types->type & TCL_GLOB_TYPE_LINK) +		    && (TclOSlstat(nativeEntry, &buf) == 0) +		    && S_ISLNK(buf.st_mode)) { +		goto filetypeOK;  	    }  #endif /* S_ISLNK */  	    return 0; @@ -728,9 +713,9 @@ TclpGetNativeCwd(      if (getcwd(buffer, MAXPATHLEN+1) == NULL) {		/* INTL: Native. */  	return NULL;      } -#endif +#endif /* USEGETWD */ -    if ((clientData == NULL) || strcmp(buffer, (const char*)clientData)) { +    if ((clientData == NULL) || strcmp(buffer, (const char *) clientData)) {  	char *newCd = ckalloc(strlen(buffer) + 1);  	strcpy(newCd, buffer); @@ -778,12 +763,12 @@ TclpGetCwd(      if (getwd(buffer) == NULL)				/* INTL: Native. */  #else      if (getcwd(buffer, MAXPATHLEN+1) == NULL)		/* INTL: Native. */ -#endif +#endif /* USEGETWD */      {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, -		    "error getting working directory name: ", -		    Tcl_PosixError(interp), NULL); +	    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		    "error getting working directory name: %s", +		    Tcl_PosixError(interp)));  	}  	return NULL;      } @@ -834,7 +819,7 @@ TclpReadlink(      return Tcl_DStringValue(linkPtr);  #else      return NULL; -#endif +#endif /* !DJGPP */  }  /* @@ -868,7 +853,7 @@ TclpObjStat(  #ifdef S_IFLNK -Tcl_Obj* +Tcl_Obj *  TclpObjLink(      Tcl_Obj *pathPtr,      Tcl_Obj *toPtr, @@ -1120,6 +1105,12 @@ TclNativeCreateNativeRep(      str = Tcl_GetStringFromObj(validPathPtr, &len);      Tcl_UtfToExternalDString(NULL, str, len, &ds);      len = Tcl_DStringLength(&ds) + sizeof(char); +    if (strlen(Tcl_DStringValue(&ds)) < len - sizeof(char)) { +	/* See bug [3118489]: NUL in filenames */ +	Tcl_DecrRefCount(validPathPtr); +	Tcl_DStringFree(&ds); +	return NULL; +    }      Tcl_DecrRefCount(validPathPtr);      nativePathPtr = ckalloc(len);      memcpy(nativePathPtr, Tcl_DStringValue(&ds), (size_t) len); @@ -1190,10 +1181,18 @@ TclpUtime(  {      return utime(Tcl_FSGetNativePath(pathPtr), tval);  } +  #ifdef __CYGWIN__ -int TclOSstat(const char *name, Tcl_StatBuf *statBuf) { + +int +TclOSstat( +    const char *name, +    void *cygstat) +{      struct stat buf; +    Tcl_StatBuf *statBuf = cygstat;      int result = stat(name, &buf); +      statBuf->st_mode = buf.st_mode;      statBuf->st_ino = buf.st_ino;      statBuf->st_dev = buf.st_dev; @@ -1207,9 +1206,16 @@ int TclOSstat(const char *name, Tcl_StatBuf *statBuf) {      statBuf->st_ctime = buf.st_ctime;      return result;  } -int TclOSlstat(const char *name, Tcl_StatBuf *statBuf) { + +int +TclOSlstat( +    const char *name, +    void *cygstat) +{      struct stat buf; +    Tcl_StatBuf *statBuf = cygstat;      int result = lstat(name, &buf); +      statBuf->st_mode = buf.st_mode;      statBuf->st_ino = buf.st_ino;      statBuf->st_dev = buf.st_dev; @@ -1223,7 +1229,7 @@ int TclOSlstat(const char *name, Tcl_StatBuf *statBuf) {      statBuf->st_ctime = buf.st_ctime;      return result;  } -#endif +#endif /* CYGWIN */  /*   * Local Variables: | 
