• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libjingle
3  * Copyright 2004--2006, Google Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  *
8  *  1. Redistributions of source code must retain the above copyright notice,
9  *     this list of conditions and the following disclaimer.
10  *  2. Redistributions in binary form must reproduce the above copyright notice,
11  *     this list of conditions and the following disclaimer in the documentation
12  *     and/or other materials provided with the distribution.
13  *  3. The name of the author may not be used to endorse or promote products
14  *     derived from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 #include "talk/base/win32filesystem.h"
29 
30 #include "talk/base/win32.h"
31 #include <shellapi.h>
32 #include <shlobj.h>
33 #include <tchar.h>
34 
35 #include "talk/base/fileutils.h"
36 #include "talk/base/pathutils.h"
37 #include "talk/base/scoped_ptr.h"
38 #include "talk/base/stream.h"
39 #include "talk/base/stringutils.h"
40 
41 // In several places in this file, we test the integrity level of the process
42 // before calling GetLongPathName. We do this because calling GetLongPathName
43 // when running under protected mode IE (a low integrity process) can result in
44 // a virtualized path being returned, which is wrong if you only plan to read.
45 // TODO: Waiting to hear back from IE team on whether this is the
46 // best approach; IEIsProtectedModeProcess is another possible solution.
47 
48 namespace talk_base {
49 
CreateFolder(const Pathname & pathname)50 bool Win32Filesystem::CreateFolder(const Pathname &pathname) {
51   if (pathname.pathname().empty() || !pathname.filename().empty())
52     return false;
53 
54   std::wstring path16;
55   if (!Utf8ToWindowsFilename(pathname.pathname(), &path16))
56     return false;
57 
58   DWORD res = ::GetFileAttributes(path16.c_str());
59   if (res != INVALID_FILE_ATTRIBUTES) {
60     // Something exists at this location, check if it is a directory
61     return ((res & FILE_ATTRIBUTE_DIRECTORY) != 0);
62   } else if ((GetLastError() != ERROR_FILE_NOT_FOUND)
63               && (GetLastError() != ERROR_PATH_NOT_FOUND)) {
64     // Unexpected error
65     return false;
66   }
67 
68   // Directory doesn't exist, look up one directory level
69   if (!pathname.parent_folder().empty()) {
70     Pathname parent(pathname);
71     parent.SetFolder(pathname.parent_folder());
72     if (!CreateFolder(parent)) {
73       return false;
74     }
75   }
76 
77   return (::CreateDirectory(path16.c_str(), NULL) != 0);
78 }
79 
OpenFile(const Pathname & filename,const std::string & mode)80 FileStream *Win32Filesystem::OpenFile(const Pathname &filename,
81                                       const std::string &mode) {
82   FileStream *fs = new FileStream();
83   if (fs && !fs->Open(filename.pathname().c_str(), mode.c_str())) {
84     delete fs;
85     fs = NULL;
86   }
87   return fs;
88 }
89 
CreatePrivateFile(const Pathname & filename)90 bool Win32Filesystem::CreatePrivateFile(const Pathname &filename) {
91   // To make the file private to the current user, we first must construct a
92   // SECURITY_DESCRIPTOR specifying an ACL. This code is mostly based upon
93   // http://msdn.microsoft.com/en-us/library/ms707085%28VS.85%29.aspx
94 
95   // Get the current process token.
96   HANDLE process_token = INVALID_HANDLE_VALUE;
97   if (!::OpenProcessToken(GetCurrentProcess(),
98                           TOKEN_QUERY,
99                           &process_token)) {
100     LOG_ERR(LS_ERROR) << "OpenProcessToken() failed";
101     return false;
102   }
103 
104   // Get the size of its TOKEN_USER structure. Return value is not checked
105   // because we expect it to fail.
106   DWORD token_user_size = 0;
107   (void)::GetTokenInformation(process_token,
108                               TokenUser,
109                               NULL,
110                               0,
111                               &token_user_size);
112 
113   // Get the TOKEN_USER structure.
114   scoped_array<char> token_user_bytes(new char[token_user_size]);
115   PTOKEN_USER token_user = reinterpret_cast<PTOKEN_USER>(
116       token_user_bytes.get());
117   memset(token_user, 0, token_user_size);
118   BOOL success = ::GetTokenInformation(process_token,
119                                        TokenUser,
120                                        token_user,
121                                        token_user_size,
122                                        &token_user_size);
123   // We're now done with this.
124   ::CloseHandle(process_token);
125   if (!success) {
126     LOG_ERR(LS_ERROR) << "GetTokenInformation() failed";
127     return false;
128   }
129 
130   if (!IsValidSid(token_user->User.Sid)) {
131     LOG_ERR(LS_ERROR) << "Current process has invalid user SID";
132     return false;
133   }
134 
135   // Compute size needed for an ACL that allows access to just this user.
136   int acl_size = sizeof(ACL) + sizeof(ACCESS_ALLOWED_ACE) - sizeof(DWORD) +
137       GetLengthSid(token_user->User.Sid);
138 
139   // Allocate it.
140   scoped_array<char> acl_bytes(new char[acl_size]);
141   PACL acl = reinterpret_cast<PACL>(acl_bytes.get());
142   memset(acl, 0, acl_size);
143   if (!::InitializeAcl(acl, acl_size, ACL_REVISION)) {
144     LOG_ERR(LS_ERROR) << "InitializeAcl() failed";
145     return false;
146   }
147 
148   // Allow access to only the current user.
149   if (!::AddAccessAllowedAce(acl,
150                              ACL_REVISION,
151                              GENERIC_READ | GENERIC_WRITE | STANDARD_RIGHTS_ALL,
152                              token_user->User.Sid)) {
153     LOG_ERR(LS_ERROR) << "AddAccessAllowedAce() failed";
154     return false;
155   }
156 
157   // Now make the security descriptor.
158   SECURITY_DESCRIPTOR security_descriptor;
159   if (!::InitializeSecurityDescriptor(&security_descriptor,
160                                       SECURITY_DESCRIPTOR_REVISION)) {
161     LOG_ERR(LS_ERROR) << "InitializeSecurityDescriptor() failed";
162     return false;
163   }
164 
165   // Put the ACL in it.
166   if (!::SetSecurityDescriptorDacl(&security_descriptor,
167                                    TRUE,
168                                    acl,
169                                    FALSE)) {
170     LOG_ERR(LS_ERROR) << "SetSecurityDescriptorDacl() failed";
171     return false;
172   }
173 
174   // Finally create the file.
175   SECURITY_ATTRIBUTES security_attributes;
176   security_attributes.nLength = sizeof(security_attributes);
177   security_attributes.lpSecurityDescriptor = &security_descriptor;
178   security_attributes.bInheritHandle = FALSE;
179   HANDLE handle = ::CreateFile(
180       ToUtf16(filename.pathname()).c_str(),
181       GENERIC_READ | GENERIC_WRITE,
182       FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
183       &security_attributes,
184       CREATE_NEW,
185       0,
186       NULL);
187   if (INVALID_HANDLE_VALUE == handle) {
188     LOG_ERR(LS_ERROR) << "CreateFile() failed";
189     return false;
190   }
191   if (!::CloseHandle(handle)) {
192     LOG_ERR(LS_ERROR) << "CloseFile() failed";
193     // Continue.
194   }
195   return true;
196 }
197 
DeleteFile(const Pathname & filename)198 bool Win32Filesystem::DeleteFile(const Pathname &filename) {
199   LOG(LS_INFO) << "Deleting file " << filename.pathname();
200   if (!IsFile(filename)) {
201     ASSERT(IsFile(filename));
202     return false;
203   }
204   return ::DeleteFile(ToUtf16(filename.pathname()).c_str()) != 0;
205 }
206 
DeleteEmptyFolder(const Pathname & folder)207 bool Win32Filesystem::DeleteEmptyFolder(const Pathname &folder) {
208   LOG(LS_INFO) << "Deleting folder " << folder.pathname();
209 
210   std::string no_slash(folder.pathname(), 0, folder.pathname().length()-1);
211   return ::RemoveDirectory(ToUtf16(no_slash).c_str()) != 0;
212 }
213 
GetTemporaryFolder(Pathname & pathname,bool create,const std::string * append)214 bool Win32Filesystem::GetTemporaryFolder(Pathname &pathname, bool create,
215                                          const std::string *append) {
216   wchar_t buffer[MAX_PATH + 1];
217   if (!::GetTempPath(ARRAY_SIZE(buffer), buffer))
218     return false;
219   if (!IsCurrentProcessLowIntegrity() &&
220       !::GetLongPathName(buffer, buffer, ARRAY_SIZE(buffer)))
221     return false;
222   size_t len = strlen(buffer);
223   if ((len > 0) && (buffer[len-1] != '\\')) {
224     len += strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, L"\\");
225   }
226   if (len >= ARRAY_SIZE(buffer) - 1)
227     return false;
228   pathname.clear();
229   pathname.SetFolder(ToUtf8(buffer));
230   if (append != NULL) {
231     ASSERT(!append->empty());
232     pathname.AppendFolder(*append);
233   }
234   return !create || CreateFolder(pathname);
235 }
236 
TempFilename(const Pathname & dir,const std::string & prefix)237 std::string Win32Filesystem::TempFilename(const Pathname &dir,
238                                           const std::string &prefix) {
239   wchar_t filename[MAX_PATH];
240   if (::GetTempFileName(ToUtf16(dir.pathname()).c_str(),
241                         ToUtf16(prefix).c_str(), 0, filename) != 0)
242     return ToUtf8(filename);
243   ASSERT(false);
244   return "";
245 }
246 
MoveFile(const Pathname & old_path,const Pathname & new_path)247 bool Win32Filesystem::MoveFile(const Pathname &old_path,
248                                const Pathname &new_path) {
249   if (!IsFile(old_path)) {
250     ASSERT(IsFile(old_path));
251     return false;
252   }
253   LOG(LS_INFO) << "Moving " << old_path.pathname()
254                << " to " << new_path.pathname();
255   return ::MoveFile(ToUtf16(old_path.pathname()).c_str(),
256                     ToUtf16(new_path.pathname()).c_str()) != 0;
257 }
258 
MoveFolder(const Pathname & old_path,const Pathname & new_path)259 bool Win32Filesystem::MoveFolder(const Pathname &old_path,
260                                  const Pathname &new_path) {
261   if (!IsFolder(old_path)) {
262     ASSERT(IsFolder(old_path));
263     return false;
264   }
265   LOG(LS_INFO) << "Moving " << old_path.pathname()
266                << " to " << new_path.pathname();
267   if (::MoveFile(ToUtf16(old_path.pathname()).c_str(),
268                ToUtf16(new_path.pathname()).c_str()) == 0) {
269     if (::GetLastError() != ERROR_NOT_SAME_DEVICE) {
270       LOG_GLE(LS_ERROR) << "Failed to move file";
271       return false;
272     }
273     if (!CopyFolder(old_path, new_path))
274       return false;
275     if (!DeleteFolderAndContents(old_path))
276       return false;
277   }
278   return true;
279 }
280 
IsFolder(const Pathname & path)281 bool Win32Filesystem::IsFolder(const Pathname &path) {
282   WIN32_FILE_ATTRIBUTE_DATA data = {0};
283   if (0 == ::GetFileAttributesEx(ToUtf16(path.pathname()).c_str(),
284                                  GetFileExInfoStandard, &data))
285     return false;
286   return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ==
287       FILE_ATTRIBUTE_DIRECTORY;
288 }
289 
IsFile(const Pathname & path)290 bool Win32Filesystem::IsFile(const Pathname &path) {
291   WIN32_FILE_ATTRIBUTE_DATA data = {0};
292   if (0 == ::GetFileAttributesEx(ToUtf16(path.pathname()).c_str(),
293                                  GetFileExInfoStandard, &data))
294     return false;
295   return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == 0;
296 }
297 
IsAbsent(const Pathname & path)298 bool Win32Filesystem::IsAbsent(const Pathname& path) {
299   WIN32_FILE_ATTRIBUTE_DATA data = {0};
300   if (0 != ::GetFileAttributesEx(ToUtf16(path.pathname()).c_str(),
301                                  GetFileExInfoStandard, &data))
302     return false;
303   DWORD err = ::GetLastError();
304   return (ERROR_FILE_NOT_FOUND == err || ERROR_PATH_NOT_FOUND == err);
305 }
306 
CopyFile(const Pathname & old_path,const Pathname & new_path)307 bool Win32Filesystem::CopyFile(const Pathname &old_path,
308                                const Pathname &new_path) {
309   return ::CopyFile(ToUtf16(old_path.pathname()).c_str(),
310                     ToUtf16(new_path.pathname()).c_str(), TRUE) != 0;
311 }
312 
IsTemporaryPath(const Pathname & pathname)313 bool Win32Filesystem::IsTemporaryPath(const Pathname& pathname) {
314   TCHAR buffer[MAX_PATH + 1];
315   if (!::GetTempPath(ARRAY_SIZE(buffer), buffer))
316     return false;
317   if (!IsCurrentProcessLowIntegrity() &&
318       !::GetLongPathName(buffer, buffer, ARRAY_SIZE(buffer)))
319     return false;
320   return (::strnicmp(ToUtf16(pathname.pathname()).c_str(),
321                      buffer, strlen(buffer)) == 0);
322 }
323 
GetFileSize(const Pathname & pathname,size_t * size)324 bool Win32Filesystem::GetFileSize(const Pathname &pathname, size_t *size) {
325   WIN32_FILE_ATTRIBUTE_DATA data = {0};
326   if (::GetFileAttributesEx(ToUtf16(pathname.pathname()).c_str(),
327                             GetFileExInfoStandard, &data) == 0)
328   return false;
329   *size = data.nFileSizeLow;
330   return true;
331 }
332 
GetFileTime(const Pathname & path,FileTimeType which,time_t * time)333 bool Win32Filesystem::GetFileTime(const Pathname& path, FileTimeType which,
334                                   time_t* time) {
335   WIN32_FILE_ATTRIBUTE_DATA data = {0};
336   if (::GetFileAttributesEx(ToUtf16(path.pathname()).c_str(),
337                             GetFileExInfoStandard, &data) == 0)
338     return false;
339   switch (which) {
340   case FTT_CREATED:
341     FileTimeToUnixTime(data.ftCreationTime, time);
342     break;
343   case FTT_MODIFIED:
344     FileTimeToUnixTime(data.ftLastWriteTime, time);
345     break;
346   case FTT_ACCESSED:
347     FileTimeToUnixTime(data.ftLastAccessTime, time);
348     break;
349   default:
350     return false;
351   }
352   return true;
353 }
354 
GetAppPathname(Pathname * path)355 bool Win32Filesystem::GetAppPathname(Pathname* path) {
356   TCHAR buffer[MAX_PATH + 1];
357   if (0 == ::GetModuleFileName(NULL, buffer, ARRAY_SIZE(buffer)))
358     return false;
359   path->SetPathname(ToUtf8(buffer));
360   return true;
361 }
362 
GetAppDataFolder(Pathname * path,bool per_user)363 bool Win32Filesystem::GetAppDataFolder(Pathname* path, bool per_user) {
364   ASSERT(!organization_name_.empty());
365   ASSERT(!application_name_.empty());
366   TCHAR buffer[MAX_PATH + 1];
367   int csidl = per_user ? CSIDL_LOCAL_APPDATA : CSIDL_COMMON_APPDATA;
368   if (!::SHGetSpecialFolderPath(NULL, buffer, csidl, TRUE))
369     return false;
370   if (!IsCurrentProcessLowIntegrity() &&
371       !::GetLongPathName(buffer, buffer, ARRAY_SIZE(buffer)))
372     return false;
373   size_t len = strcatn(buffer, ARRAY_SIZE(buffer), __T("\\"));
374   len += strcpyn(buffer + len, ARRAY_SIZE(buffer) - len,
375                  ToUtf16(organization_name_).c_str());
376   if ((len > 0) && (buffer[len-1] != __T('\\'))) {
377     len += strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\"));
378   }
379   len += strcpyn(buffer + len, ARRAY_SIZE(buffer) - len,
380                  ToUtf16(application_name_).c_str());
381   if ((len > 0) && (buffer[len-1] != __T('\\'))) {
382     len += strcpyn(buffer + len, ARRAY_SIZE(buffer) - len, __T("\\"));
383   }
384   if (len >= ARRAY_SIZE(buffer) - 1)
385     return false;
386   path->clear();
387   path->SetFolder(ToUtf8(buffer));
388   return CreateFolder(*path);
389 }
390 
GetAppTempFolder(Pathname * path)391 bool Win32Filesystem::GetAppTempFolder(Pathname* path) {
392   if (!GetAppPathname(path))
393     return false;
394   std::string filename(path->filename());
395   return GetTemporaryFolder(*path, true, &filename);
396 }
397 
GetDiskFreeSpace(const Pathname & path,int64 * freebytes)398 bool Win32Filesystem::GetDiskFreeSpace(const Pathname& path, int64 *freebytes) {
399   if (!freebytes) {
400     return false;
401   }
402   char drive[4];
403   std::wstring drive16;
404   const wchar_t* target_drive = NULL;
405   if (path.GetDrive(drive, sizeof(drive))) {
406     drive16 = ToUtf16(drive);
407     target_drive = drive16.c_str();
408   } else if (path.folder().substr(0, 2) == "\\\\") {
409     // UNC path, fail.
410     // TODO: Handle UNC paths.
411     return false;
412   } else {
413     // The path is probably relative.  GetDriveType and GetDiskFreeSpaceEx
414     // use the current drive if NULL is passed as the drive name.
415     // TODO: Add method to Pathname to determine if the path is relative.
416     // TODO: Add method to Pathname to convert a path to absolute.
417   }
418   UINT driveType = ::GetDriveType(target_drive);
419   if ( (driveType & DRIVE_REMOTE) || (driveType & DRIVE_UNKNOWN) ) {
420     LOG(LS_VERBOSE) << " remove or unknown drive " << drive;
421     return false;
422   }
423 
424   int64 totalNumberOfBytes;  // receives the number of bytes on disk
425   int64 totalNumberOfFreeBytes;  // receives the free bytes on disk
426   // make sure things won't change in 64 bit machine
427   // TODO replace with compile time assert
428   ASSERT(sizeof(ULARGE_INTEGER) == sizeof(uint64));  //NOLINT
429   if (::GetDiskFreeSpaceEx(target_drive,
430                            (PULARGE_INTEGER)freebytes,
431                            (PULARGE_INTEGER)&totalNumberOfBytes,
432                            (PULARGE_INTEGER)&totalNumberOfFreeBytes)) {
433     return true;
434   } else {
435     LOG(LS_VERBOSE) << " GetDiskFreeSpaceEx returns error ";
436     return false;
437   }
438 }
439 
GetCurrentDirectory()440 Pathname Win32Filesystem::GetCurrentDirectory() {
441   Pathname cwd;
442   int path_len = 0;
443   scoped_array<wchar_t> path;
444   do {
445     int needed = ::GetCurrentDirectory(path_len, path.get());
446     if (needed == 0) {
447       // Error.
448       LOG_GLE(LS_ERROR) << "::GetCurrentDirectory() failed";
449       return cwd;  // returns empty pathname
450     }
451     if (needed <= path_len) {
452       // It wrote successfully.
453       break;
454     }
455     // Else need to re-alloc for "needed".
456     path.reset(new wchar_t[needed]);
457     path_len = needed;
458   } while (true);
459   cwd.SetFolder(ToUtf8(path.get()));
460   return cwd;
461 }
462 
463 // TODO: Consider overriding DeleteFolderAndContents for speed and potentially
464 // better OS integration (recycle bin?)
465 /*
466   std::wstring temp_path16 = ToUtf16(temp_path.pathname());
467   temp_path16.append(1, '*');
468   temp_path16.append(1, '\0');
469 
470   SHFILEOPSTRUCT file_op = { 0 };
471   file_op.wFunc = FO_DELETE;
472   file_op.pFrom = temp_path16.c_str();
473   file_op.fFlags = FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_SILENT;
474   return (0 == SHFileOperation(&file_op));
475 */
476 
477 }  // namespace talk_base
478