diff options
Diffstat (limited to 'win/tclWinFile.c')
| -rw-r--r-- | win/tclWinFile.c | 778 | 
1 files changed, 318 insertions, 460 deletions
| diff --git a/win/tclWinFile.c b/win/tclWinFile.c index a772015..5761eeb 100644 --- a/win/tclWinFile.c +++ b/win/tclWinFile.c @@ -15,7 +15,6 @@  #include "tclWinInt.h"  #include "tclFileSystem.h"  #include <winioctl.h> -#include <sys/stat.h>  #include <shlobj.h>  #include <lm.h>		/* For TclpGetUserHome(). */ @@ -160,7 +159,7 @@ static unsigned short	NativeStatMode(DWORD attr, int checkLinks,  			    int isExec);  static int		NativeIsExec(const TCHAR *path);  static int		NativeReadReparse(const TCHAR *LinkDirectory, -			    REPARSE_DATA_BUFFER *buffer); +			    REPARSE_DATA_BUFFER *buffer, DWORD desiredAccess);  static int		NativeWriteReparse(const TCHAR *LinkDirectory,  			    REPARSE_DATA_BUFFER *buffer);  static int		NativeMatchType(int isDrive, DWORD attr, @@ -249,9 +248,7 @@ WinLink(  	 * It is a file.  	 */ -	if (CreateHardLink == NULL) { -	    Tcl_SetErrno(ENOTDIR); -	} else if (linkAction & TCL_CREATE_HARD_LINK) { +	if (linkAction & TCL_CREATE_HARD_LINK) {  	    if (CreateHardLink(linkSourcePath, linkTargetPath, NULL)) {  		/*  		 * Success! @@ -446,7 +443,7 @@ TclWinSymLinkCopyDirectory(      DUMMY_REPARSE_BUFFER dummy;      REPARSE_DATA_BUFFER *reparseBuffer = (REPARSE_DATA_BUFFER *) &dummy; -    if (NativeReadReparse(linkOrigPath, reparseBuffer)) { +    if (NativeReadReparse(linkOrigPath, reparseBuffer, GENERIC_READ)) {  	return -1;      }      return NativeWriteReparse(linkCopyPath, reparseBuffer); @@ -544,7 +541,7 @@ WinReadLinkDirectory(      if (!(attr & FILE_ATTRIBUTE_REPARSE_POINT)) {  	goto invalidError;      } -    if (NativeReadReparse(linkDirPath, reparseBuffer)) { +    if (NativeReadReparse(linkDirPath, reparseBuffer, 0)) {  	return NULL;      } @@ -665,12 +662,13 @@ WinReadLinkDirectory(  static int  NativeReadReparse(      const TCHAR *linkDirPath,	/* The junction to read */ -    REPARSE_DATA_BUFFER *buffer)/* Pointer to buffer. Cannot be NULL */ +    REPARSE_DATA_BUFFER *buffer,/* Pointer to buffer. Cannot be NULL */ +    DWORD desiredAccess)  {      HANDLE hFile;      DWORD returnedLength; -    hFile = CreateFile(linkDirPath, GENERIC_READ, 0, NULL, OPEN_EXISTING, +    hFile = CreateFile(linkDirPath, desiredAccess, 0, NULL, OPEN_EXISTING,  	    FILE_FLAG_OPEN_REPARSE_POINT | FILE_FLAG_BACKUP_SEMANTICS, NULL);      if (hFile == INVALID_HANDLE_VALUE) { @@ -801,7 +799,7 @@ tclWinDebugPanic(      WCHAR msgString[TCL_MAX_WARN_LEN];      va_start(argList, format); -    _vsnprintf(buf, sizeof(buf), format, argList); +    vsnprintf(buf, sizeof(buf), format, argList);      msgString[TCL_MAX_WARN_LEN-1] = L'\0';      MultiByteToWideChar(CP_UTF8, 0, buf, -1, msgString, TCL_MAX_WARN_LEN); @@ -821,6 +819,16 @@ tclWinDebugPanic(  	MessageBoxW(NULL, msgString, L"Fatal Error",  		MB_ICONSTOP | MB_OK | MB_TASKMODAL | MB_SETFOREGROUND);      } +#if defined(__GNUC__) +    __builtin_trap(); +#elif defined(_WIN64) +    __debugbreak(); +#elif defined(_MSC_VER) +    _asm {int 3} +#else +    DebugBreak(); +#endif +    abort();  }  /* @@ -922,24 +930,16 @@ TclpMatchInDirectory(  	    int len;  	    DWORD attr; +	    WIN32_FILE_ATTRIBUTE_DATA data;  	    const char *str = Tcl_GetStringFromObj(norm,&len);  	    native = Tcl_FSGetNativePath(pathPtr); -	    if (GetFileAttributesEx == NULL) { -		attr = GetFileAttributes(native); -		if (attr == INVALID_FILE_ATTRIBUTES) { -		    return TCL_OK; -		} -	    } else { -		WIN32_FILE_ATTRIBUTE_DATA data; - -		if (GetFileAttributesEx(native, -			GetFileExInfoStandard, &data) != TRUE) { -		    return TCL_OK; -		} -		attr = data.dwFileAttributes; +	    if (GetFileAttributesEx(native, +		    GetFileExInfoStandard, &data) != TRUE) { +		return TCL_OK;  	    } +	    attr = data.dwFileAttributes;  	    if (NativeMatchType(WinIsDrive(str,len), attr, native, types)) {  		Tcl_ListObjAppendElement(interp, resultPtr, pathPtr); @@ -996,7 +996,7 @@ TclpMatchInDirectory(  	lastChar = dirName[dirLength -1];  	if ((lastChar != '\\') && (lastChar != '/') && (lastChar != ':')) { -	    Tcl_DStringAppend(&dsOrig, "/", 1); +	    TclDStringAppendLiteral(&dsOrig, "/");  	    dirLength++;  	}  	dirName = Tcl_DStringValue(&dsOrig); @@ -1016,12 +1016,11 @@ TclpMatchInDirectory(  	    dirName = Tcl_DStringAppend(&dsOrig, pattern, -1);  	} else { -	    dirName = Tcl_DStringAppend(&dsOrig, "*.*", 3); +	    dirName = TclDStringAppendLiteral(&dsOrig, "*.*");  	}  	native = Tcl_WinUtfToTChar(dirName, -1, &ds); -	if (FindFirstFileEx == NULL || (types == NULL) -		|| (types->type != TCL_GLOB_TYPE_DIR)) { +	if ((types == NULL) || (types->type != TCL_GLOB_TYPE_DIR)) {  	    handle = FindFirstFile(native, &data);  	} else {  	    /* @@ -1049,10 +1048,9 @@ TclpMatchInDirectory(  	    TclWinConvertError(err);  	    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);  	    return TCL_ERROR; @@ -1296,7 +1294,7 @@ WinIsReserved(   *	because for NTFS root volumes, the getFileAttributesProc returns a   *	'hidden' attribute when it should not.   * - *	We never make any calss to a 'get attributes' routine here, since we + *	We never make any calls to a 'get attributes' routine here, since we   *	have arranged things so that our caller already knows such   *	information.   * @@ -1468,7 +1466,7 @@ TclpGetUserHome(  		GetWindowsDirectoryW(buf, MAX_PATH);  		Tcl_UniCharToUtfDString(buf, 2, bufferPtr); -		Tcl_DStringAppend(bufferPtr, "/users/default", -1); +		TclDStringAppendLiteral(bufferPtr, "/users/default");  	    }  	    result = Tcl_DStringValue(bufferPtr);  	    NetApiBufferFree((void *) uiPtr); @@ -1539,32 +1537,37 @@ NativeAccess(  	 * File might not exist.  	 */ -	WIN32_FIND_DATA ffd; -	HANDLE hFind; -	hFind = FindFirstFile(nativePath, &ffd); -	if (hFind != INVALID_HANDLE_VALUE) { -	    attr = ffd.dwFileAttributes; -	    FindClose(hFind); -	} else { -	    TclWinConvertError(GetLastError()); +	DWORD lasterror = GetLastError(); +	if (lasterror != ERROR_SHARING_VIOLATION) { +	    TclWinConvertError(lasterror);  	    return -1;  	}      } -#ifndef UNICODE +    if (mode == F_OK) { +	/* +	 * File exists, nothing else to check. +	 */ + +	return 0; +    } +      if ((mode & W_OK) -	    && (attr & FILE_ATTRIBUTE_READONLY)) { +	&& (attr & FILE_ATTRIBUTE_READONLY) +	&& !(attr & FILE_ATTRIBUTE_DIRECTORY)) {  	/* -	 * We don't have the advanced 'GetFileSecurity', and our -	 * attributes say the file is not writable. If we do have -	 * 'GetFileSecurity', we'll do a more robust XP-related check -	 * below. +	 * The attributes say the file is not writable.	 If the file is a +	 * regular file (i.e., not a directory), then the file is not +	 * writable, full stop.	 For directories, the read-only bit is +	 * (mostly) ignored by Windows, so we can't ascertain anything about +	 * directory access from the attrib data.  However, if we have the +	 * advanced 'getFileSecurityProc', then more robust ACL checks +	 * will be done below.  	 */  	Tcl_SetErrno(EACCES);  	return -1;      } -#endif /* !UNICODE */      if (mode & X_OK) {  	if (!(attr & FILE_ATTRIBUTE_DIRECTORY) && !NativeIsExec(nativePath)) { @@ -1583,15 +1586,15 @@ NativeAccess(       * we have a more complex permissions structure so we try to check that.       * The code below is remarkably complex for such a simple thing as finding       * what permissions the OS has set for a file. -     * -     * If we are simply checking for file existence, then we don't need all -     * these complications (which are really quite slow: with this code 'file -     * readable' is 5-6 times slower than 'file exists').       */ -    if (mode != F_OK) { +#ifdef UNICODE +    {  	SECURITY_DESCRIPTOR *sdPtr = NULL;  	unsigned long size; +	PSID pSid = 0; +	BOOL SidDefaulted; +	SID_IDENTIFIER_AUTHORITY samba_unmapped = {{0, 0, 0, 0, 0, 22}};  	GENERIC_MAPPING genMap;  	HANDLE hToken = NULL;  	DWORD desiredAccess = 0, grantedAccess = 0; @@ -1607,7 +1610,8 @@ NativeAccess(  	size = 0;  	GetFileSecurity(nativePath,  		OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION -		| DACL_SECURITY_INFORMATION, 0, 0, &size); +		| DACL_SECURITY_INFORMATION | LABEL_SECURITY_INFORMATION, +		0, 0, &size);  	/*  	 * Should have failed with ERROR_INSUFFICIENT_BUFFER @@ -1640,7 +1644,8 @@ NativeAccess(  	if (!GetFileSecurity(nativePath,  		OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION -		| DACL_SECURITY_INFORMATION, sdPtr, size, &size)) { +		| DACL_SECURITY_INFORMATION | LABEL_SECURITY_INFORMATION, +		sdPtr, size, &size)) {  	    /*  	     * Error getting owner SD  	     */ @@ -1649,6 +1654,26 @@ NativeAccess(  	}  	/* +	 * As of Samba 3.0.23 (10-Jul-2006), unmapped users and groups are +	 * assigned to SID domains S-1-22-1 and S-1-22-2, where "22" is the +	 * top-level authority.	 If the file owner and group is unmapped then +	 * the ACL access check below will only test against world access, +	 * which is likely to be more restrictive than the actual access +	 * restrictions.  Since the ACL tests are more likely wrong than +	 * right, skip them.  Moreover, the unix owner access permissions are +	 * usually mapped to the Windows attributes, so if the user is the +	 * file owner then the attrib checks above are correct (as far as they +	 * go). +	 */ + +	if(!GetSecurityDescriptorOwner(sdPtr,&pSid,&SidDefaulted) || +	   memcmp(GetSidIdentifierAuthority(pSid),&samba_unmapped, +		  sizeof(SID_IDENTIFIER_AUTHORITY))==0) { +	    HeapFree(GetProcessHeap(), 0, sdPtr); +	    return 0; /* Attrib tests say access allowed. */ +	} + +	/*  	 * Perform security impersonation of the user and open the resulting  	 * thread token.  	 */ @@ -1725,18 +1750,8 @@ NativeAccess(  	    return -1;  	} -	/* -	 * For directories the above checks are ok. For files, though, we must -	 * still check the 'attr' value. -	 */ - -	if ((mode & W_OK) -		&& !(attr & FILE_ATTRIBUTE_DIRECTORY) -		&& (attr & FILE_ATTRIBUTE_READONLY)) { -	    Tcl_SetErrno(EACCES); -	    return -1; -	}      } +#endif /* !UNICODE */      return 0;  } @@ -1764,7 +1779,7 @@ NativeIsExec(  	return 0;      } -    if (path[len-4] != TEXT('.')) { +    if (path[len-4] != '.') {  	return 0;      } @@ -1798,27 +1813,13 @@ TclpObjChdir(  {      int result;      const TCHAR *nativePath; -#ifdef __CYGWIN__ -    extern int cygwin_conv_to_posix_path(const char *, char *); -    char posixPath[MAX_PATH+1]; -    const char *path; -    Tcl_DString ds; -#endif /* __CYGWIN__ */      nativePath = Tcl_FSGetNativePath(pathPtr); -#ifdef __CYGWIN__ -    /* -     * Cygwin chdir only groks POSIX path. -     */ - -    path = Tcl_WinTCharToUtf(nativePath, -1, &ds); -    cygwin_conv_to_posix_path(path, posixPath); -    result = (chdir(posixPath) == 0 ? 1 : 0); -    Tcl_DStringFree(&ds); -#else /* __CYGWIN__ */ +    if (!nativePath) { +	return -1; +    }      result = SetCurrentDirectory(nativePath); -#endif /* __CYGWIN__ */      if (result == 0) {  	TclWinConvertError(GetLastError()); @@ -1827,51 +1828,6 @@ TclpObjChdir(      return 0;  } -#ifdef __CYGWIN__ -/* - *--------------------------------------------------------------------------- - * - * TclpReadlink -- - * - *	This function replaces the library version of readlink(). - * - * Results: - *	The result is a pointer to a string specifying the contents of the - *	symbolic link given by 'path', or NULL if the symbolic link could not - *	be read. Storage for the result string is allocated in bufferPtr; the - *	caller must call Tcl_DStringFree() when the result is no longer - *	needed. - * - * Side effects: - *	See readlink() documentation. - * - *--------------------------------------------------------------------------- - */ - -char * -TclpReadlink( -    const char *path,		/* Path of file to readlink (UTF-8). */ -    Tcl_DString *linkPtr)	/* Uninitialized or free DString filled with -				 * contents of link (UTF-8). */ -{ -    char link[MAXPATHLEN]; -    int length; -    char *native; -    Tcl_DString ds; - -    native = Tcl_UtfToExternalDString(NULL, path, -1, &ds); -    length = readlink(native, link, sizeof(link));	/* INTL: Native. */ -    Tcl_DStringFree(&ds); - -    if (length < 0) { -	return NULL; -    } - -    Tcl_ExternalToUtfDString(NULL, link, length, linkPtr); -    return Tcl_DStringValue(linkPtr); -} -#endif /* __CYGWIN__ */ -  /*   *----------------------------------------------------------------------   * @@ -1907,8 +1863,9 @@ TclpGetCwd(      if (GetCurrentDirectory(MAX_PATH, buffer) == 0) {  	TclWinConvertError(GetLastError());  	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;      } @@ -2043,15 +2000,17 @@ NativeStat(  	if (GetFileAttributesEx(nativePath,  		GetFileExInfoStandard, &data) != TRUE) { -	    /* -	     * We might have just been denied access -	     */ - +	    HANDLE hFind;  	    WIN32_FIND_DATA ffd; -	    HANDLE hFind = FindFirstFile(nativePath, &ffd); +	    DWORD lasterror = GetLastError(); +	    if (lasterror != ERROR_SHARING_VIOLATION) { +		TclWinConvertError(lasterror); +		return -1; +		} +	    hFind = FindFirstFile(nativePath, &ffd);  	    if (hFind == INVALID_HANDLE_VALUE) { -		Tcl_SetErrno(ENOENT); +		TclWinConvertError(GetLastError());  		return -1;  	    }  	    memcpy(&data, &ffd, sizeof(data)); @@ -2117,7 +2076,7 @@ NativeDev(  	     * won't work.  	     */ -	    fullPath = Tcl_DStringAppend(&ds, "\\", 1); +	    fullPath = TclDStringAppendLiteral(&ds, "\\");  	    p = fullPath + Tcl_DStringLength(&ds);  	} else {  	    p++; @@ -2186,8 +2145,8 @@ NativeStatMode(       * positions.       */ -    mode |= (mode & 0x0700) >> 3; -    mode |= (mode & 0x0700) >> 6; +    mode |= (mode & (S_IREAD|S_IWRITE|S_IEXEC)) >> 3; +    mode |= (mode & (S_IREAD|S_IWRITE|S_IEXEC)) >> 6;      return (unsigned short) mode;  } @@ -2401,13 +2360,9 @@ TclpFilesystemPathType(  	return NULL;      } else {  	Tcl_DString ds; -	Tcl_Obj *objPtr;  	Tcl_WinTCharToUtf(volType, -1, &ds); -	objPtr = Tcl_NewStringObj(Tcl_DStringValue(&ds), -		Tcl_DStringLength(&ds)); -	Tcl_DStringFree(&ds); -	return objPtr; +	return TclDStringToObj(&ds);      }  #undef VOL_BUF_SIZE  } @@ -2463,361 +2418,213 @@ TclpObjNormalizePath(      Tcl_DStringInit(&dsNorm);      path = Tcl_GetString(pathPtr); -    if (TclWinGetPlatformId() == VER_PLATFORM_WIN32_WINDOWS) { -	/* -	 * We're on Win95, 98 or ME. There are two assumptions in this block -	 * of code. First that the native (NULL) encoding is basically ascii, -	 * and second that symbolic links are not possible. Both of these -	 * assumptions appear to be true of these operating systems. -	 * -	 * FIXME: This code branch may be derelict as those are not supported -	 * platforms any more. -	 */ - -	currentPathEndPosition = path + nextCheckpoint; -	if (*currentPathEndPosition == '/') { -	    currentPathEndPosition++; -	} - -	while (1) { -	    char cur = *currentPathEndPosition; +    currentPathEndPosition = path + nextCheckpoint; +    if (*currentPathEndPosition == '/') { +	currentPathEndPosition++; +    } +    while (1) { +	char cur = *currentPathEndPosition; -	    if ((cur=='/' || cur==0) && (path != currentPathEndPosition)) { -		/* -		 * Reached directory separator, or end of string. -		 */ +	if ((cur=='/' || cur==0) && (path != currentPathEndPosition)) { +	    /* +	     * Reached directory separator, or end of string. +	     */ -		const char *nativePath = Tcl_UtfToExternalDString(NULL, path, -			currentPathEndPosition - path, &ds); +	    WIN32_FILE_ATTRIBUTE_DATA data; +	    const TCHAR *nativePath = Tcl_WinUtfToTChar(path, +		    currentPathEndPosition - path, &ds); +	    if (GetFileAttributesEx(nativePath, +		    GetFileExInfoStandard, &data) != TRUE) {  		/* -		 * Now we convert the tail of the current path to its 'long -		 * form', and append it to 'dsNorm' which holds the current -		 * normalized path, if the file exists. +		 * File doesn't exist.  		 */  		if (isDrive) { -		    if (GetFileAttributesA(nativePath) -			    == INVALID_FILE_ATTRIBUTES) { -			/* -			 * File doesn't exist. -			 */ - -			if (isDrive) { -			    int len = WinIsReserved(path); - -			    if (len > 0) { -				/* -				 * Actually it does exist - COM1, etc. -				 */ - -				int i; - -				for (i=0 ; i<len ; i++) { -				    if (nativePath[i] >= 'a') { -					((char *) nativePath)[i] -= ('a'-'A'); -				    } -				} -				Tcl_DStringAppend(&dsNorm, nativePath, len); -				lastValidPathEnd = currentPathEndPosition; -			    } -			} -			Tcl_DStringFree(&ds); -			break; -		    } -		    if (nativePath[0] >= 'a') { -			((char *) nativePath)[0] -= ('a' - 'A'); -		    } -		    Tcl_DStringAppend(&dsNorm, nativePath, -			    Tcl_DStringLength(&ds)); -		} else { -		    char *checkDots = NULL; - -		    if (lastValidPathEnd[1] == '.') { -			checkDots = lastValidPathEnd + 1; -			while (checkDots < currentPathEndPosition) { -			    if (*checkDots != '.') { -				checkDots = NULL; -				break; -			    } -			    checkDots++; -			} -		    } -		    if (checkDots != NULL) { -			int dotLen = currentPathEndPosition-lastValidPathEnd; +		    int len = WinIsReserved(path); +		    if (len > 0) {  			/* -			 * Path is just dots. We shouldn't really ever see a -			 * path like that. However, to be nice we at least -			 * don't mangle the path - we just add the dots as a -			 * path segment and continue +			 * Actually it does exist - COM1, etc.  			 */ -			Tcl_DStringAppend(&dsNorm, (const char *) -				(nativePath + Tcl_DStringLength(&ds)-dotLen), -				dotLen); -		    } else { -			/* -			 * Normal path. -			 */ - -			WIN32_FIND_DATAA fData; -			HANDLE handle; - -			handle = FindFirstFileA(nativePath, &fData); -			if (handle == INVALID_HANDLE_VALUE) { -			    if (GetFileAttributesA(nativePath) -				    == INVALID_FILE_ATTRIBUTES) { -				/* -				 * File doesn't exist. -				 */ - -				Tcl_DStringFree(&ds); -				break; -			    } - -			    /* -			     * This is usually the '/' in 'c:/' at end of -			     * string. -			     */ +			int i; -			    Tcl_DStringAppend(&dsNorm,"/", 1); -			} else { -			    char *nativeName; +			for (i=0 ; i<len ; i++) { +			    WCHAR wc = ((WCHAR *) nativePath)[i]; -			    if (fData.cFileName[0] != '\0') { -				nativeName = fData.cFileName; -			    } else { -				nativeName = fData.cAlternateFileName; +			    if (wc >= L'a') { +				wc -= (L'a' - L'A'); +				((WCHAR *) nativePath)[i] = wc;  			    } -			    FindClose(handle); -			    Tcl_DStringAppend(&dsNorm,"/", 1); -			    Tcl_DStringAppend(&dsNorm,nativeName,-1);  			} +			Tcl_DStringAppend(&dsNorm, +				(const char *)nativePath, +				(int)(sizeof(WCHAR) * len)); +			lastValidPathEnd = currentPathEndPosition; +		    } else if (nextCheckpoint == 0) { +			/* Path starts with a drive designation +			 * that's not actually on the system. +			 * We still must normalize up past the +			 * first separator.  [Bug 3603434] */ +			currentPathEndPosition++;  		    }  		}  		Tcl_DStringFree(&ds); -		lastValidPathEnd = currentPathEndPosition; -		if (cur == 0) { -		    break; -		} - -		/* -		 * If we get here, we've got past one directory delimiter, so -		 * we know it is no longer a drive. -		 */ - -		isDrive = 0; +		break;  	    } -	    currentPathEndPosition++; -	} -    } else { -	/* -	 * We're on WinNT (or 2000 or XP; something with an NT core). -	 */ -	currentPathEndPosition = path + nextCheckpoint; -	if (*currentPathEndPosition == '/') { -	    currentPathEndPosition++; -	} -	while (1) { -	    char cur = *currentPathEndPosition; +	    /* +	     * File 'nativePath' does exist if we get here. We now want to +	     * check if it is a symlink and otherwise continue with the +	     * rest of the path. +	     */ -	    if ((cur=='/' || cur==0) && (path != currentPathEndPosition)) { -		/* -		 * Reached directory separator, or end of string. -		 */ +	    /* +	     * Check for symlinks, except at last component of path (we +	     * don't follow final symlinks). Also a drive (C:/) for +	     * example, may sometimes have the reparse flag set for some +	     * reason I don't understand. We therefore don't perform this +	     * check for drives. +	     */ -		WIN32_FILE_ATTRIBUTE_DATA data; -		const TCHAR *nativePath = Tcl_WinUtfToTChar(path, -			currentPathEndPosition - path, &ds); +	    if (cur != 0 && !isDrive && +		    data.dwFileAttributes & FILE_ATTRIBUTE_REPARSE_POINT){ +		Tcl_Obj *to = WinReadLinkDirectory(nativePath); -		if (GetFileAttributesEx(nativePath, -			GetFileExInfoStandard, &data) != TRUE) { +		if (to != NULL) {  		    /* -		     * File doesn't exist. +		     * Read the reparse point ok. Now, reparse points need +		     * not be normalized, otherwise we could use: +		     * +		     * Tcl_GetStringFromObj(to, &pathLen); +		     * nextCheckpoint = pathLen; +		     * +		     * So, instead we have to start from the beginning.  		     */ -		    if (isDrive) { -			int len = WinIsReserved(path); - -			if (len > 0) { -			    /* -			     * Actually it does exist - COM1, etc. -			     */ - -			    int i; +		    nextCheckpoint = 0; +		    Tcl_AppendToObj(to, currentPathEndPosition, -1); -			    for (i=0 ; i<len ; i++) { -				WCHAR wc = ((WCHAR *) nativePath)[i]; +		    /* +		     * Convert link to forward slashes. +		     */ -				if (wc >= L'a') { -				    wc -= (L'a' - L'A'); -				    ((WCHAR *) nativePath)[i] = wc; -				} -			    } -			    Tcl_DStringAppend(&dsNorm, -				    (const char *)nativePath, -				    (int)(sizeof(WCHAR) * len)); -			    lastValidPathEnd = currentPathEndPosition; +		    for (path = Tcl_GetString(to); *path != 0; path++) { +			if (*path == '\\') { +			    *path = '/';  			}  		    } -		    Tcl_DStringFree(&ds); -		    break; -		} - -		/* -		 * File 'nativePath' does exist if we get here. We now want to -		 * check if it is a symlink and otherwise continue with the -		 * rest of the path. -		 */ - -		/* -		 * Check for symlinks, except at last component of path (we -		 * don't follow final symlinks). Also a drive (C:/) for -		 * example, may sometimes have the reparse flag set for some -		 * reason I don't understand. We therefore don't perform this -		 * check for drives. -		 */ +		    path = Tcl_GetString(to); +		    currentPathEndPosition = path + nextCheckpoint; +		    if (temp != NULL) { +			Tcl_DecrRefCount(temp); +		    } +		    temp = to; -		if (cur != 0 && !isDrive && -			data.dwFileAttributes & FILE_ATTRIBUTE_REPARSE_POINT){ -		    Tcl_Obj *to = WinReadLinkDirectory(nativePath); +		    /* +		     * Reset variables so we can restart normalization. +		     */ -		    if (to != NULL) { -			/* -			 * Read the reparse point ok. Now, reparse points need -			 * not be normalized, otherwise we could use: -			 * -			 * Tcl_GetStringFromObj(to, &pathLen); -			 * nextCheckpoint = pathLen; -			 * -			 * So, instead we have to start from the beginning. -			 */ +		    isDrive = 1; +		    Tcl_DStringFree(&dsNorm); +		    Tcl_DStringFree(&ds); +		    continue; +		} +	    } -			nextCheckpoint = 0; -			Tcl_AppendToObj(to, currentPathEndPosition, -1); +#ifndef TclNORM_LONG_PATH +	    /* +	     * Now we convert the tail of the current path to its 'long +	     * form', and append it to 'dsNorm' which holds the current +	     * normalized path +	     */ -			/* -			 * Convert link to forward slashes. -			 */ +	    if (isDrive) { +		WCHAR drive = ((WCHAR *) nativePath)[0]; -			for (path = Tcl_GetString(to); *path != 0; path++) { -			    if (*path == '\\') { -				*path = '/'; -			    } -			} -			path = Tcl_GetString(to); -			currentPathEndPosition = path + nextCheckpoint; -			if (temp != NULL) { -			    Tcl_DecrRefCount(temp); +		if (drive >= L'a') { +		    drive -= (L'a' - L'A'); +		    ((WCHAR *) nativePath)[0] = drive; +		} +		Tcl_DStringAppend(&dsNorm, (const char *)nativePath, +			Tcl_DStringLength(&ds)); +	    } else { +		char *checkDots = NULL; + +		if (lastValidPathEnd[1] == '.') { +		    checkDots = lastValidPathEnd + 1; +		    while (checkDots < currentPathEndPosition) { +			if (*checkDots != '.') { +			    checkDots = NULL; +			    break;  			} -			temp = to; - -			/* -			 * Reset variables so we can restart normalization. -			 */ - -			isDrive = 1; -			Tcl_DStringFree(&dsNorm); -			Tcl_DStringFree(&ds); -			continue; +			checkDots++;  		    }  		} +		if (checkDots != NULL) { +		    int dotLen = currentPathEndPosition-lastValidPathEnd; -#ifndef TclNORM_LONG_PATH -		/* -		 * Now we convert the tail of the current path to its 'long -		 * form', and append it to 'dsNorm' which holds the current -		 * normalized path -		 */ - -		if (isDrive) { -		    WCHAR drive = ((WCHAR *) nativePath)[0]; +		    /* +		     * Path is just dots. We shouldn't really ever see a +		     * path like that. However, to be nice we at least +		     * don't mangle the path - we just add the dots as a +		     * path segment and continue. +		     */ -		    if (drive >= L'a') { -			drive -= (L'a' - L'A'); -			((WCHAR *) nativePath)[0] = drive; -		    } -		    Tcl_DStringAppend(&dsNorm, (const char *)nativePath, -			    Tcl_DStringLength(&ds)); +		    Tcl_DStringAppend(&dsNorm, ((const char *)nativePath) +			    + Tcl_DStringLength(&ds) +			    - (dotLen * sizeof(TCHAR)), +			    (int)(dotLen * sizeof(TCHAR)));  		} else { -		    char *checkDots = NULL; - -		    if (lastValidPathEnd[1] == '.') { -			checkDots = lastValidPathEnd + 1; -			while (checkDots < currentPathEndPosition) { -			    if (*checkDots != '.') { -				checkDots = NULL; -				break; -			    } -			    checkDots++; -			} -		    } -		    if (checkDots != NULL) { -			int dotLen = currentPathEndPosition-lastValidPathEnd; +		    /* +		     * Normal path. +		     */ -			/* -			 * Path is just dots. We shouldn't really ever see a -			 * path like that. However, to be nice we at least -			 * don't mangle the path - we just add the dots as a -			 * path segment and continue. -			 */ +		    WIN32_FIND_DATAW fData; +		    HANDLE handle; -			Tcl_DStringAppend(&dsNorm, ((const char *)nativePath) -				+ Tcl_DStringLength(&ds) -				- (dotLen * sizeof(TCHAR)), -				(int)(dotLen * sizeof(TCHAR))); -		    } else { +		    handle = FindFirstFileW((WCHAR *) nativePath, &fData); +		    if (handle == INVALID_HANDLE_VALUE) {  			/* -			 * Normal path. +			 * This is usually the '/' in 'c:/' at end of +			 * string.  			 */ -			WIN32_FIND_DATAW fData; -			HANDLE handle; - -			handle = FindFirstFileW((WCHAR *) nativePath, &fData); -			if (handle == INVALID_HANDLE_VALUE) { -			    /* -			     * This is usually the '/' in 'c:/' at end of -			     * string. -			     */ +			Tcl_DStringAppend(&dsNorm, (const char *) L"/", +				sizeof(WCHAR)); +		    } else { +			WCHAR *nativeName; -			    Tcl_DStringAppend(&dsNorm, (const char *) L"/", -				    sizeof(WCHAR)); +			if (fData.cFileName[0] != '\0') { +			    nativeName = fData.cFileName;  			} else { -			    WCHAR *nativeName; - -			    if (fData.cFileName[0] != '\0') { -				nativeName = fData.cFileName; -			    } else { -				nativeName = fData.cAlternateFileName; -			    } -			    FindClose(handle); -			    Tcl_DStringAppend(&dsNorm, (const char *) L"/", -				    sizeof(WCHAR)); -			    Tcl_DStringAppend(&dsNorm, -				    (const char *) nativeName, -				    (int) (wcslen(nativeName)*sizeof(WCHAR))); +			    nativeName = fData.cAlternateFileName;  			} +			FindClose(handle); +			Tcl_DStringAppend(&dsNorm, (const char *) L"/", +				sizeof(WCHAR)); +			Tcl_DStringAppend(&dsNorm, +				(const char *) nativeName, +				(int) (wcslen(nativeName)*sizeof(WCHAR)));  		    }  		} +	    }  #endif /* !TclNORM_LONG_PATH */ -		Tcl_DStringFree(&ds); -		lastValidPathEnd = currentPathEndPosition; -		if (cur == 0) { -		    break; -		} +	    Tcl_DStringFree(&ds); +	    lastValidPathEnd = currentPathEndPosition; +	    if (cur == 0) { +		break; +	    } -		/* -		 * If we get here, we've got past one directory delimiter, so -		 * we know it is no longer a drive. -		 */ +	    /* +	     * If we get here, we've got past one directory delimiter, so +	     * we know it is no longer a drive. +	     */ -		isDrive = 0; -	    } -	    currentPathEndPosition++; +	    isDrive = 0;  	} +	currentPathEndPosition++;  #ifdef TclNORM_LONG_PATH  	/* @@ -3090,10 +2897,11 @@ ClientData  TclNativeCreateNativeRep(      Tcl_Obj *pathPtr)  { -    char *nativePathPtr, *str; -    Tcl_DString ds; +    WCHAR *nativePathPtr; +    const char *str;      Tcl_Obj *validPathPtr;      int len; +    WCHAR *wp;      if (TclFSCwdIsNative()) {  	/* @@ -3119,22 +2927,72 @@ TclNativeCreateNativeRep(      }      str = Tcl_GetStringFromObj(validPathPtr, &len); -    if (str[0] == '/' && str[1] == '/' && str[2] == '?' && str[3] == '/') { -	char *p; -	for (p = str; p && *p; ++p) { -	    if (*p == '/') { -		*p = '\\'; -	    } +    if (strlen(str)!=len) { +	/* String contains NUL-bytes. This is invalid. */ +	return 0; +    } +    /* Let MultiByteToWideChar check for other invalid sequences, like +     * 0xC0 0x80 (== overlong NUL). See bug [3118489]: NUL in filenames */ +    len = MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, str, -1, 0, 0); +    if (len==0) { +	return 0; +    } +    /* Overallocate 6 chars, making some room for extended paths */ +    wp = nativePathPtr = ckalloc( (len+6) * sizeof(WCHAR) ); +    if (nativePathPtr==0) { +      return 0; +    } +    MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, str, -1, nativePathPtr, len); +    /* +    ** If path starts with "//?/" or "\\?\" (extended path), translate +    ** any slashes to backslashes but leave the '?' intact +    */ +    if ((str[0]=='\\' || str[0]=='/') && (str[1]=='\\' || str[1]=='/') +	    && str[2]=='?' && (str[3]=='\\' || str[3]=='/')) { +	wp[0] = wp[1] = wp[3] = '\\'; +	str += 4; +	wp += 4; +    } +    /* +    ** If there is no "\\?\" prefix but there is a drive or UNC +    ** path prefix and the path is larger than MAX_PATH chars, +    ** no Win32 API function can handle that unless it is +    ** prefixed with the extended path prefix. See: +    ** <http://msdn.microsoft.com/en-us/library/aa365247(VS.85).aspx#maxpath> +    **/ +    if (((str[0]>='A'&&str[0]<='Z') || (str[0]>='a'&&str[0]<='z')) +	    && str[1]==':' && (str[2]=='\\' || str[2]=='/')) { +	if (wp==nativePathPtr && len>MAX_PATH) { +	    memmove(wp+4, wp, len*sizeof(WCHAR)); +	    memcpy(wp, L"\\\\?\\", 4*sizeof(WCHAR)); +	    wp += 4;  	} +	/* +	 ** If (remainder of) path starts with "<drive>:/" or "<drive>:\", +	 ** leave the ':' intact but translate the backslash to a slash. +	 */ +	wp[2] = '\\'; +	wp += 3; +    } else if (wp==nativePathPtr && len>MAX_PATH +	    && (str[0]=='\\' || str[0]=='/') +	    && (str[1]=='\\' || str[1]=='/') && str[2]!='?') { +	memmove(wp+6, wp, len*sizeof(WCHAR)); +	memcpy(wp, L"\\\\?\\UNC", 7*sizeof(WCHAR)); +	wp += 7; +    } +    /* +    ** In the remainder of the path, translate invalid characters to +    ** characters in the Unicode private use area. +    */ +    while (*wp != '\0') { +	if ((*wp < ' ') || wcschr(L"\"*:<>?|", *wp)) { +	    *wp |= 0xF000; +	} else if (*wp == '/') { +	    *wp = '\\'; +	} +	++wp;      } -    Tcl_WinUtfToTChar(str, len, &ds); -    len = Tcl_DStringLength(&ds) + sizeof(WCHAR); -    Tcl_DecrRefCount(validPathPtr); -    nativePathPtr = ckalloc(len); -    memcpy(nativePathPtr, Tcl_DStringValue(&ds), (size_t) len); - -    Tcl_DStringFree(&ds);      return nativePathPtr;  } | 
