forked from mirrors/gecko-dev
		
	This removes an unnecessary level of indirection by replacing all nsStringGlue.h instances with just nsString.h. --HG-- extra : rebase_source : 340989240af4018f3ebfd92826ae11b0cb46d019
		
			
				
	
	
		
			144 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| #ifndef mozilla_FileUtilsWin_h
 | |
| #define mozilla_FileUtilsWin_h
 | |
| 
 | |
| #include <windows.h>
 | |
| 
 | |
| #include "mozilla/Scoped.h"
 | |
| #include "nsString.h"
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| inline bool
 | |
| EnsureLongPath(nsAString& aDosPath)
 | |
| {
 | |
|   uint32_t aDosPathOriginalLen = aDosPath.Length();
 | |
|   auto inputPath = PromiseFlatString(aDosPath);
 | |
|   // Try to get the long path, or else get the required length of the long path
 | |
|   DWORD longPathLen = GetLongPathNameW(inputPath.get(),
 | |
|                                        reinterpret_cast<wchar_t*>(aDosPath.BeginWriting()),
 | |
|                                        aDosPathOriginalLen);
 | |
|   if (longPathLen == 0) {
 | |
|     return false;
 | |
|   }
 | |
|   aDosPath.SetLength(longPathLen);
 | |
|   if (longPathLen <= aDosPathOriginalLen) {
 | |
|     // Our string happened to be long enough for the first call to succeed.
 | |
|     return true;
 | |
|   }
 | |
|   // Now we have a large enough buffer, get the actual string
 | |
|   longPathLen = GetLongPathNameW(inputPath.get(),
 | |
|                                  reinterpret_cast<wchar_t*>(aDosPath.BeginWriting()), aDosPath.Length());
 | |
|   if (longPathLen == 0) {
 | |
|     return false;
 | |
|   }
 | |
|   // This success check should always be less-than because longPathLen excludes
 | |
|   // the null terminator on success, but includes it in the first call that
 | |
|   // returned the required size.
 | |
|   if (longPathLen < aDosPath.Length()) {
 | |
|     aDosPath.SetLength(longPathLen);
 | |
|     return true;
 | |
|   }
 | |
|   // We shouldn't reach this, but if we do then it's a failure!
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| inline bool
 | |
| NtPathToDosPath(const nsAString& aNtPath, nsAString& aDosPath)
 | |
| {
 | |
|   aDosPath.Truncate();
 | |
|   if (aNtPath.IsEmpty()) {
 | |
|     return true;
 | |
|   }
 | |
|   NS_NAMED_LITERAL_STRING(symLinkPrefix, "\\??\\");
 | |
|   uint32_t ntPathLen = aNtPath.Length();
 | |
|   uint32_t symLinkPrefixLen = symLinkPrefix.Length();
 | |
|   if (ntPathLen >= 6 && aNtPath.CharAt(5) == L':' &&
 | |
|       ntPathLen >= symLinkPrefixLen &&
 | |
|       Substring(aNtPath, 0, symLinkPrefixLen).Equals(symLinkPrefix)) {
 | |
|     // Symbolic link for DOS device. Just strip off the prefix.
 | |
|     aDosPath = aNtPath;
 | |
|     aDosPath.Cut(0, 4);
 | |
|     return true;
 | |
|   }
 | |
|   nsAutoString logicalDrives;
 | |
|   DWORD len = 0;
 | |
|   while (true) {
 | |
|     len = GetLogicalDriveStringsW(
 | |
|       len, reinterpret_cast<wchar_t*>(logicalDrives.BeginWriting()));
 | |
|     if (!len) {
 | |
|       return false;
 | |
|     } else if (len > logicalDrives.Length()) {
 | |
|       logicalDrives.SetLength(len);
 | |
|     } else {
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   const char16_t* cur = logicalDrives.BeginReading();
 | |
|   const char16_t* end = logicalDrives.EndReading();
 | |
|   nsString targetPath;
 | |
|   targetPath.SetLength(MAX_PATH);
 | |
|   wchar_t driveTemplate[] = L" :";
 | |
|   do {
 | |
|     // Unfortunately QueryDosDevice doesn't support the idiom for querying the
 | |
|     // output buffer size, so it may require retries.
 | |
|     driveTemplate[0] = *cur;
 | |
|     DWORD targetPathLen = 0;
 | |
|     SetLastError(ERROR_SUCCESS);
 | |
|     while (true) {
 | |
|       targetPathLen = QueryDosDeviceW(driveTemplate,
 | |
|                                       reinterpret_cast<wchar_t*>(targetPath.BeginWriting()),
 | |
|                                       targetPath.Length());
 | |
|       if (targetPathLen || GetLastError() != ERROR_INSUFFICIENT_BUFFER) {
 | |
|         break;
 | |
|       }
 | |
|       targetPath.SetLength(targetPath.Length() * 2);
 | |
|     }
 | |
|     if (targetPathLen) {
 | |
|       // Need to use wcslen here because targetPath contains embedded NULL chars
 | |
|       size_t firstTargetPathLen = wcslen(targetPath.get());
 | |
|       const char16_t* pathComponent = aNtPath.BeginReading() +
 | |
|                                       firstTargetPathLen;
 | |
|       bool found = _wcsnicmp(char16ptr_t(aNtPath.BeginReading()), targetPath.get(),
 | |
|                              firstTargetPathLen) == 0 &&
 | |
|                    *pathComponent == L'\\';
 | |
|       if (found) {
 | |
|         aDosPath = driveTemplate;
 | |
|         aDosPath += pathComponent;
 | |
|         return EnsureLongPath(aDosPath);
 | |
|       }
 | |
|     }
 | |
|     // Advance to the next NUL character in logicalDrives
 | |
|     while (*cur++);
 | |
|   } while (cur != end);
 | |
|   // Try to handle UNC paths. NB: This must happen after we've checked drive
 | |
|   // mappings in case a UNC path is mapped to a drive!
 | |
|   NS_NAMED_LITERAL_STRING(uncPrefix, "\\\\");
 | |
|   NS_NAMED_LITERAL_STRING(deviceMupPrefix, "\\Device\\Mup\\");
 | |
|   if (StringBeginsWith(aNtPath, deviceMupPrefix)) {
 | |
|     aDosPath = uncPrefix;
 | |
|     aDosPath += Substring(aNtPath, deviceMupPrefix.Length());
 | |
|     return true;
 | |
|   }
 | |
|   NS_NAMED_LITERAL_STRING(deviceLanmanRedirectorPrefix,
 | |
|                           "\\Device\\LanmanRedirector\\");
 | |
|   if (StringBeginsWith(aNtPath, deviceLanmanRedirectorPrefix)) {
 | |
|     aDosPath = uncPrefix;
 | |
|     aDosPath += Substring(aNtPath, deviceLanmanRedirectorPrefix.Length());
 | |
|     return true;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool
 | |
| HandleToFilename(HANDLE aHandle, const LARGE_INTEGER& aOffset,
 | |
|                  nsAString& aFilename);
 | |
| 
 | |
| } // namespace mozilla
 | |
| 
 | |
| #endif // mozilla_FileUtilsWin_h
 |