file_access_win32.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. /*
  2. * Copyright (C) 2008-2011 Teluu Inc. (http://www.teluu.com)
  3. * Copyright (C) 2003-2008 Benny Prijono <benny@prijono.org>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation; either version 2 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program; if not, write to the Free Software
  17. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  18. */
  19. #include <pj/file_access.h>
  20. #include <pj/unicode.h>
  21. #include <pj/assert.h>
  22. #include <pj/errno.h>
  23. #include <pj/string.h>
  24. #include <pj/os.h>
  25. #include <windows.h>
  26. #include <time.h>
  27. #if defined(PJ_WIN32_WINCE) && PJ_WIN32_WINCE!=0
  28. /* WinCE lacks READ_CONTROL so we must use GENERIC_READ */
  29. # define CONTROL_ACCESS GENERIC_READ
  30. #else
  31. # define CONTROL_ACCESS READ_CONTROL
  32. #endif
  33. static pj_status_t get_file_size(HANDLE hFile, pj_off_t *size)
  34. {
  35. #if defined(PJ_WIN32_WINPHONE8) && PJ_WIN32_WINPHONE8
  36. FILE_COMPRESSION_INFO fileInfo;
  37. if (GetFileInformationByHandleEx(hFile, FileCompressionInfo, &fileInfo,
  38. sizeof(FILE_COMPRESSION_INFO)))
  39. {
  40. *size = fileInfo.CompressedFileSize.QuadPart;
  41. }
  42. else {
  43. *size = -1;
  44. return PJ_RETURN_OS_ERROR(GetLastError());
  45. }
  46. #else
  47. DWORD sizeLo, sizeHi;
  48. sizeLo = GetFileSize(hFile, &sizeHi);
  49. if (sizeLo == INVALID_FILE_SIZE) {
  50. DWORD dwStatus = GetLastError();
  51. if (dwStatus != NO_ERROR) {
  52. *size = -1;
  53. return PJ_RETURN_OS_ERROR(dwStatus);
  54. }
  55. }
  56. *size = sizeHi;
  57. *size = ((*size) << 32) + sizeLo;
  58. #endif
  59. return PJ_SUCCESS;
  60. }
  61. static HANDLE WINAPI create_file(LPCTSTR filename, DWORD desired_access,
  62. DWORD share_mode,
  63. LPSECURITY_ATTRIBUTES security_attributes,
  64. DWORD creation_disposition,
  65. DWORD flags_and_attributes,
  66. HANDLE template_file)
  67. {
  68. #if defined(PJ_WIN32_WINPHONE8) && PJ_WIN32_WINPHONE8
  69. PJ_UNUSED_ARG(security_attributes);
  70. PJ_UNUSED_ARG(flags_and_attributes);
  71. PJ_UNUSED_ARG(template_file);
  72. return CreateFile2(filename, desired_access, share_mode,
  73. creation_disposition, NULL);
  74. #else
  75. return CreateFile(filename, desired_access, share_mode,
  76. security_attributes, creation_disposition,
  77. flags_and_attributes, template_file);
  78. #endif
  79. }
  80. /*
  81. * pj_file_exists()
  82. */
  83. PJ_DEF(pj_bool_t) pj_file_exists(const char *filename)
  84. {
  85. PJ_DECL_UNICODE_TEMP_BUF(wfilename,256)
  86. HANDLE hFile;
  87. PJ_ASSERT_RETURN(filename != NULL, 0);
  88. hFile = create_file(PJ_STRING_TO_NATIVE(filename,
  89. wfilename, sizeof(wfilename)),
  90. CONTROL_ACCESS,
  91. FILE_SHARE_READ, NULL,
  92. OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
  93. if (hFile == INVALID_HANDLE_VALUE)
  94. return 0;
  95. CloseHandle(hFile);
  96. return PJ_TRUE;
  97. }
  98. /*
  99. * pj_file_size()
  100. */
  101. PJ_DEF(pj_off_t) pj_file_size(const char *filename)
  102. {
  103. PJ_DECL_UNICODE_TEMP_BUF(wfilename,256)
  104. HANDLE hFile;
  105. pj_off_t size;
  106. PJ_ASSERT_RETURN(filename != NULL, -1);
  107. hFile = create_file(PJ_STRING_TO_NATIVE(filename,
  108. wfilename, sizeof(wfilename)),
  109. CONTROL_ACCESS,
  110. FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
  111. OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
  112. if (hFile == INVALID_HANDLE_VALUE)
  113. return -1;
  114. get_file_size(hFile, &size);
  115. CloseHandle(hFile);
  116. return size;
  117. }
  118. /*
  119. * pj_file_delete()
  120. */
  121. PJ_DEF(pj_status_t) pj_file_delete(const char *filename)
  122. {
  123. PJ_DECL_UNICODE_TEMP_BUF(wfilename,256)
  124. PJ_ASSERT_RETURN(filename != NULL, PJ_EINVAL);
  125. if (DeleteFile(PJ_STRING_TO_NATIVE(filename,wfilename,sizeof(wfilename))) == FALSE)
  126. return PJ_RETURN_OS_ERROR(GetLastError());
  127. return PJ_SUCCESS;
  128. }
  129. /*
  130. * pj_file_move()
  131. */
  132. PJ_DEF(pj_status_t) pj_file_move( const char *oldname, const char *newname)
  133. {
  134. PJ_DECL_UNICODE_TEMP_BUF(woldname,256)
  135. PJ_DECL_UNICODE_TEMP_BUF(wnewname,256)
  136. BOOL rc;
  137. PJ_ASSERT_RETURN(oldname!=NULL && newname!=NULL, PJ_EINVAL);
  138. #if PJ_WIN32_WINNT >= 0x0400
  139. rc = MoveFileEx(PJ_STRING_TO_NATIVE(oldname,woldname,sizeof(woldname)),
  140. PJ_STRING_TO_NATIVE(newname,wnewname,sizeof(wnewname)),
  141. MOVEFILE_COPY_ALLOWED|MOVEFILE_REPLACE_EXISTING);
  142. #else
  143. rc = MoveFile(PJ_STRING_TO_NATIVE(oldname,woldname,sizeof(woldname)),
  144. PJ_STRING_TO_NATIVE(newname,wnewname,sizeof(wnewname)));
  145. #endif
  146. if (!rc)
  147. return PJ_RETURN_OS_ERROR(GetLastError());
  148. return PJ_SUCCESS;
  149. }
  150. static pj_status_t file_time_to_time_val(const FILETIME *file_time,
  151. pj_time_val *time_val)
  152. {
  153. #if !defined(PJ_WIN32_WINPHONE8) || !PJ_WIN32_WINPHONE8
  154. FILETIME local_file_time;
  155. #endif
  156. SYSTEMTIME localTime;
  157. pj_parsed_time pt;
  158. #if !defined(PJ_WIN32_WINPHONE8) || !PJ_WIN32_WINPHONE8
  159. if (!FileTimeToLocalFileTime(file_time, &local_file_time))
  160. return PJ_RETURN_OS_ERROR(GetLastError());
  161. #endif
  162. if (!FileTimeToSystemTime(file_time, &localTime))
  163. return PJ_RETURN_OS_ERROR(GetLastError());
  164. //if (!SystemTimeToTzSpecificLocalTime(NULL, &systemTime, &localTime))
  165. // return PJ_RETURN_OS_ERROR(GetLastError());
  166. pj_bzero(&pt, sizeof(pt));
  167. pt.year = localTime.wYear;
  168. pt.mon = localTime.wMonth-1;
  169. pt.day = localTime.wDay;
  170. pt.wday = localTime.wDayOfWeek;
  171. pt.hour = localTime.wHour;
  172. pt.min = localTime.wMinute;
  173. pt.sec = localTime.wSecond;
  174. pt.msec = localTime.wMilliseconds;
  175. return pj_time_encode(&pt, time_val);
  176. }
  177. /*
  178. * pj_file_getstat()
  179. */
  180. PJ_DEF(pj_status_t) pj_file_getstat(const char *filename, pj_file_stat *stat)
  181. {
  182. PJ_DECL_UNICODE_TEMP_BUF(wfilename,256)
  183. HANDLE hFile;
  184. FILETIME creationTime, accessTime, writeTime;
  185. #if defined(PJ_WIN32_WINPHONE8) && PJ_WIN32_WINPHONE8
  186. FILE_BASIC_INFO fileInfo;
  187. #endif
  188. PJ_ASSERT_RETURN(filename!=NULL && stat!=NULL, PJ_EINVAL);
  189. hFile = create_file(PJ_STRING_TO_NATIVE(filename,
  190. wfilename, sizeof(wfilename)),
  191. CONTROL_ACCESS,
  192. FILE_SHARE_READ, NULL,
  193. OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
  194. if (hFile == INVALID_HANDLE_VALUE)
  195. return PJ_RETURN_OS_ERROR(GetLastError());
  196. if (get_file_size(hFile, &stat->size) != PJ_SUCCESS) {
  197. CloseHandle(hFile);
  198. return PJ_RETURN_OS_ERROR(GetLastError());
  199. }
  200. #if defined(PJ_WIN32_WINPHONE8) && PJ_WIN32_WINPHONE8
  201. if (GetFileInformationByHandleEx(hFile, FileBasicInfo, &fileInfo,
  202. sizeof(FILE_BASIC_INFO)))
  203. {
  204. creationTime.dwLowDateTime = fileInfo.CreationTime.LowPart;
  205. creationTime.dwHighDateTime = fileInfo.CreationTime.HighPart;
  206. accessTime.dwLowDateTime = fileInfo.LastAccessTime.LowPart;
  207. accessTime.dwHighDateTime = fileInfo.LastAccessTime.HighPart;
  208. writeTime.dwLowDateTime = fileInfo.LastWriteTime.LowPart;
  209. writeTime.dwHighDateTime = fileInfo.LastWriteTime.HighPart;
  210. }
  211. else {
  212. CloseHandle(hFile);
  213. return PJ_RETURN_OS_ERROR(GetLastError());
  214. }
  215. #else
  216. if (GetFileTime(hFile, &creationTime, &accessTime, &writeTime) == FALSE) {
  217. DWORD dwStatus = GetLastError();
  218. CloseHandle(hFile);
  219. return PJ_RETURN_OS_ERROR(dwStatus);
  220. }
  221. #endif
  222. CloseHandle(hFile);
  223. if (file_time_to_time_val(&creationTime, &stat->ctime) != PJ_SUCCESS)
  224. return PJ_RETURN_OS_ERROR(GetLastError());
  225. file_time_to_time_val(&accessTime, &stat->atime);
  226. file_time_to_time_val(&writeTime, &stat->mtime);
  227. return PJ_SUCCESS;
  228. }