Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

573 lines
15 KiB

  1. //+----------------------------------------------------------------------------
  2. //
  3. // File: cmfdi.cpp
  4. //
  5. // Module: CMDL32.EXE
  6. //
  7. // Synopsis: CFdi class implementations
  8. //
  9. // Copyright (c) 1996-1999 Microsoft Corporation
  10. //
  11. // Author: nickball Created 04/08/98
  12. //
  13. //+----------------------------------------------------------------------------
  14. #include "cmmaster.h"
  15. #include <fcntl.h>
  16. #include <sys\stat.h>
  17. const TCHAR* const c_pszExeFile = TEXT("PBUPDATE.EXE"); // for detecting itExeInCab
  18. const TCHAR* const c_pszInfFile = TEXT("PBUPDATE.INF"); // for detecting itInfInCab
  19. const TCHAR* const c_pszVerFile = TEXT("PBUPDATE.VER"); // version string in .CAB
  20. const TCHAR* const c_pszPbr = TEXT("PBR"); // regions file extension
  21. //
  22. // CFDIFile Implementation
  23. //
  24. CFDIFile::~CFDIFile()
  25. {
  26. // nothing
  27. }
  28. DWORD CFDIFile::Read(LPVOID pv, DWORD cb)
  29. {
  30. return 0;
  31. }
  32. DWORD CFDIFile::Write(LPVOID pv, DWORD cb)
  33. {
  34. return 0;
  35. }
  36. long CFDIFile::Seek(long dist, int seektype)
  37. {
  38. return 0;
  39. }
  40. int CFDIFile::Close()
  41. {
  42. return 0;
  43. }
  44. HANDLE CFDIFile::GetHandle()
  45. {
  46. return (INVALID_HANDLE_VALUE);
  47. }
  48. //
  49. // CFDIFileFile Implementation
  50. //
  51. CFDIFileFile::CFDIFileFile()
  52. {
  53. m_hFile = INVALID_HANDLE_VALUE;
  54. }
  55. CFDIFileFile::~CFDIFileFile()
  56. {
  57. if (m_hFile != INVALID_HANDLE_VALUE) {
  58. MYDBG(("CFDIFileFile::~CFDIFileFile() destructor called while file still open."));
  59. Close();
  60. }
  61. }
  62. BOOL CFDIFileFile::CreateFile(LPCTSTR pszFile,
  63. DWORD dwDesiredAccess,
  64. DWORD dwShareMode,
  65. DWORD dwCreationDistribution,
  66. DWORD dwFlagsAndAttributes,
  67. DWORD dwFileSize)
  68. {
  69. // Make sure the files isn't in use
  70. if (m_hFile != INVALID_HANDLE_VALUE)
  71. {
  72. MYDBG(("CFDIFileFile::CreateFile() file is already open."));
  73. SetLastError(ERROR_OUT_OF_STRUCTURES);
  74. return (FALSE);
  75. }
  76. // Open Create/Open the file
  77. m_hFile = ::CreateFile(pszFile,dwDesiredAccess,dwShareMode,NULL,dwCreationDistribution,dwFlagsAndAttributes,NULL);
  78. if (m_hFile == INVALID_HANDLE_VALUE)
  79. {
  80. MYDBG(("CFDIFileFile::CreateFile() CreateFile(pszFile=%s,dwDesiredAccess=%u,dwShareMode=%u,dwCreationDistribution=%u,dwFlagsAndAttributes=%u) failed, GLE=%u.",
  81. pszFile,dwDesiredAccess,dwShareMode,dwCreationDistribution,dwFlagsAndAttributes,GetLastError()));
  82. return (FALSE);
  83. }
  84. // If dwFileSize is specified, move the pointer by dwFileSize bytes
  85. if (dwFileSize)
  86. {
  87. BOOL bRes;
  88. DWORD dwRes;
  89. dwRes = SetFilePointer(m_hFile,dwFileSize,NULL,FILE_BEGIN);
  90. MYDBGTST(dwRes==INVALID_SET_FILE_POINTER ,("CFDIFileFile::CreateFile() SetFilePointer() failed, GLE=%u.",GetLastError()));
  91. // If that worked, set the end of file at the file pointer position
  92. if (dwRes != INVALID_SET_FILE_POINTER)
  93. {
  94. bRes = SetEndOfFile(m_hFile);
  95. MYDBGTST(!bRes,("CFDIFileFile::CreateFile() SetEndOfFile() failed, GLE=%u.",GetLastError()));
  96. }
  97. // Reset the file pointer to the beginning
  98. if ((dwRes != INVALID_SET_FILE_POINTER ) && bRes)
  99. {
  100. dwRes = SetFilePointer(m_hFile,0,NULL,FILE_BEGIN);
  101. MYDBGTST(dwRes==INVALID_SET_FILE_POINTER ,("CFDIFileFile::CreateFile() SetFilePointer() failed, GLE=%u.",GetLastError()));
  102. }
  103. // Close the file and bail if we failed the above
  104. if ((dwRes == INVALID_SET_FILE_POINTER ) || !bRes)
  105. {
  106. bRes = CloseHandle(m_hFile);
  107. MYDBGTST(!bRes,("CFDIFileFile::CreateFile() CloseHandle() failed, GLE=%u.",GetLastError()));
  108. m_hFile = INVALID_HANDLE_VALUE;
  109. return (-1);
  110. }
  111. }
  112. return (TRUE);
  113. }
  114. BOOL CFDIFileFile::CreateUniqueFile(LPTSTR pszFile, DWORD dwDesiredAccess, DWORD dwShareMode, DWORD dwFlagsAndAttributes)
  115. {
  116. DWORD dwIdx;
  117. TCHAR szFile[MAX_PATH+1];
  118. if (m_hFile != INVALID_HANDLE_VALUE) {
  119. MYDBG(("CFDIFileFile::CreateUniqueFile() file is already open."));
  120. SetLastError(ERROR_OUT_OF_STRUCTURES);
  121. return (FALSE);
  122. }
  123. dwIdx = 0;
  124. while (1) {
  125. wsprintf(szFile,"%08u.tmp",dwIdx);
  126. m_hFile = ::CreateFile(szFile,dwDesiredAccess,dwShareMode,NULL,CREATE_NEW,dwFlagsAndAttributes,NULL);
  127. if (m_hFile != INVALID_HANDLE_VALUE) {
  128. break;
  129. }
  130. if (IsErrorForUnique(GetLastError(), szFile)) {
  131. MYDBG(("CFDIFileFile::CreateUniqueFile() CreateFile() failed, GLE=%u.",GetLastError()));
  132. return (FALSE);
  133. }
  134. dwIdx++;
  135. }
  136. lstrcpy(pszFile,szFile);
  137. return (TRUE);
  138. }
  139. DWORD CFDIFileFile::Read(LPVOID pv, DWORD cb)
  140. {
  141. BOOL bRes;
  142. DWORD dwRes;
  143. bRes = ReadFile(m_hFile,pv,cb,&dwRes,NULL);
  144. if (!bRes) {
  145. MYDBG(("CFDIFileFile::Read() ReadFile() failed, GLE=%u.",GetLastError()));
  146. return ((UINT) -1);
  147. }
  148. return (dwRes);
  149. }
  150. DWORD CFDIFileFile::Write(LPVOID pv, DWORD cb)
  151. {
  152. BOOL bRes;
  153. DWORD dwRes;
  154. bRes = WriteFile(m_hFile,pv,cb,&dwRes,NULL);
  155. if (!bRes) {
  156. MYDBG(("CFDIFileFile::Write() WriteFile() failed, GLE=%u.",GetLastError()));
  157. return ((UINT) -1);
  158. }
  159. return (dwRes);
  160. }
  161. long CFDIFileFile::Seek(long dist, int seektype)
  162. {
  163. DWORD dwRes;
  164. dwRes = SetFilePointer(m_hFile,dist,NULL,seektype);
  165. if (dwRes == INVALID_SET_FILE_POINTER) {
  166. MYDBG(("CFDIFileFile::Seek() SetFilePointer() failed, GLE=%u.",GetLastError()));
  167. return (-1);
  168. }
  169. return ((long) dwRes);
  170. }
  171. int CFDIFileFile::Close()
  172. {
  173. BOOL bRes;
  174. bRes = CloseHandle(m_hFile);
  175. if (!bRes) {
  176. MYDBG(("CFDIFileFile::Close() CloseHandle() failed, GLE=%u.",GetLastError()));
  177. return (-1);
  178. }
  179. m_hFile = INVALID_HANDLE_VALUE;
  180. return (0);
  181. }
  182. HANDLE CFDIFileFile::GetHandle()
  183. {
  184. return (m_hFile);
  185. }
  186. //
  187. // FDI wrapper routines
  188. //
  189. void HUGE * FAR DIAMONDAPI fdi_alloc(ULONG cb)
  190. {
  191. return (CmMalloc(cb));
  192. }
  193. void FAR DIAMONDAPI fdi_free(void HUGE *pv)
  194. {
  195. CmFree(pv);
  196. }
  197. INT_PTR FAR DIAMONDAPI fdi_open(char FAR *pszFile, int oflag, int pmode)
  198. {
  199. TCHAR szTempFileName[MAX_PATH+1];
  200. DWORD dwDesiredAccess;
  201. DWORD dwShareMode = FILE_SHARE_READ;
  202. DWORD dwCreationDistribution;
  203. DWORD dwFlagsAndAttributes = FILE_ATTRIBUTE_NORMAL;
  204. DWORD dwFileSize = 0;
  205. DWORD dwRes;
  206. BOOL bRes;
  207. CFDIFileFile *pfff = NULL;
  208. pfff = new CFDIFileFile;
  209. if (!pfff) {
  210. MYDBG(("fdi_open() new CFDIFileFile failed."));
  211. return (-1);
  212. }
  213. switch (oflag & (_O_RDONLY|_O_WRONLY|_O_RDWR)) {
  214. case _O_RDONLY:
  215. dwDesiredAccess = GENERIC_READ;
  216. break;
  217. case _O_WRONLY:
  218. dwDesiredAccess = GENERIC_WRITE;
  219. break;
  220. case _O_RDWR:
  221. dwDesiredAccess = GENERIC_READ | GENERIC_WRITE;
  222. break;
  223. default:
  224. MYDBG(("fdi_open() invalid read/write flags, oflag=%u.",oflag));
  225. return (-1);
  226. }
  227. if (oflag & _O_CREAT) {
  228. if (!(pmode & _S_IWRITE)) {
  229. dwFlagsAndAttributes |= FILE_ATTRIBUTE_READONLY;
  230. }
  231. }
  232. if (oflag & (_O_CREAT | _O_EXCL)) {
  233. dwCreationDistribution = CREATE_NEW;
  234. } else if (oflag & (_O_CREAT | _O_TRUNC)) {
  235. dwCreationDistribution = CREATE_ALWAYS;
  236. } else if (oflag & _O_CREAT) {
  237. dwCreationDistribution = OPEN_ALWAYS;
  238. } else if (oflag & _O_TRUNC) {
  239. dwCreationDistribution = TRUNCATE_EXISTING;
  240. } else {
  241. dwCreationDistribution = OPEN_EXISTING;
  242. }
  243. if (*pszFile == '*') {
  244. PFDISPILLFILE pfsf = (PFDISPILLFILE) pszFile;
  245. TCHAR szTempPath[MAX_PATH+1];
  246. ZeroMemory(szTempPath,sizeof(szTempPath));
  247. ZeroMemory(szTempFileName,sizeof(szTempFileName));
  248. dwRes = GetTempPath(sizeof(szTempPath)/sizeof(TCHAR)-1,szTempPath);
  249. MYDBGTST(!dwRes,("fdi_open() GetTempPath() failed, GLE=%u.",GetLastError()));
  250. dwRes = GetTempFileName(szTempPath,TEXT("ctf"),0,szTempFileName);
  251. MYDBGTST(!dwRes,("fdi_open() GetTempFileName() failed, GLE=%u.",GetLastError()));
  252. MYDBGTST(!dwRes,("fdi_open() GetTempFileName() failed, GLE=%u.",GetLastError()));
  253. pszFile = szTempFileName;
  254. dwFileSize = pfsf->cbFile;
  255. dwFlagsAndAttributes |= FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE;
  256. }
  257. bRes = pfff->CreateFile(pszFile,dwDesiredAccess,dwShareMode,dwCreationDistribution,dwFlagsAndAttributes,dwFileSize);
  258. if (!bRes) {
  259. delete pfff;
  260. return (-1);
  261. }
  262. return ((INT_PTR)pfff);
  263. }
  264. UINT FAR DIAMONDAPI fdi_read(INT_PTR hf, void FAR *pv, UINT cb)
  265. {
  266. return (((CFDIFile *) hf)->Read(pv,cb));
  267. }
  268. UINT FAR DIAMONDAPI fdi_write(INT_PTR hf, void FAR *pv, UINT cb)
  269. {
  270. return (((CFDIFile *) hf)->Write(pv,cb));
  271. }
  272. long FAR DIAMONDAPI fdi_seek(INT_PTR hf, long dist, int seektype)
  273. {
  274. return (((CFDIFile *) hf)->Seek(dist,seektype));
  275. }
  276. int FAR DIAMONDAPI fdi_close(INT_PTR hf)
  277. {
  278. int nRes;
  279. CFDIFile *pff = (CFDIFile *) hf;
  280. nRes = pff->Close();
  281. delete pff;
  282. return (nRes);
  283. }
  284. INT_PTR FAR DIAMONDAPI fdi_notify(FDINOTIFICATIONTYPE fdint, PFDINOTIFICATION pfdin)
  285. {
  286. NotifyArgs *pnaArgs = (NotifyArgs *) pfdin->pv;
  287. BOOL bRes;
  288. switch (fdint)
  289. {
  290. case fdintCOPY_FILE:
  291. {
  292. InstallType itType;
  293. TCHAR szTmp[MAX_PATH+1];
  294. LPTSTR pszExt; // file extension
  295. PFILEPROCESSINFO pFPI;
  296. MYDBG(("fdi_notify() fdint=fdintCOPY_FILE, psz1=%s, cb=%u.",pfdin->psz1,pfdin->cb));
  297. if (lstrlen(pnaArgs->pdaArgs->szCabDir)+lstrlen(pfdin->psz1)+1>sizeof(szTmp)/sizeof(TCHAR)-1)
  298. {
  299. MYDBG(("fdi_notify() szCabDir=%s+pszFile=%s exceeds MAX_PATH.",pnaArgs->pdaArgs->szCabDir,pfdin->psz1));
  300. return (-1);
  301. }
  302. lstrcpy(szTmp,pnaArgs->pdaArgs->szCabDir);
  303. if (szTmp[0] && (GetLastChar(szTmp) != '\\'))
  304. {
  305. lstrcat(szTmp,TEXT("\\"));
  306. }
  307. lstrcat(szTmp,pfdin->psz1);
  308. if (!(pnaArgs->dwAppFlags & AF_NO_EXEINCAB) && (lstrcmpi(pfdin->psz1,c_pszExeFile) == 0))
  309. {
  310. //
  311. // Its a .EXE, note the fact for later processing
  312. //
  313. #ifdef EXTENDED_CAB_CONTENTS
  314. itType = itExeInCab;
  315. #else
  316. itType = itInvalid; // currently an invalid type
  317. #endif // EXTENDED_CAB_CONTENTS
  318. pnaArgs->pdaArgs->fContainsExeOrInf = TRUE;
  319. }
  320. else if (!(pnaArgs->dwAppFlags & AF_NO_INFINCAB) && (lstrcmpi(pfdin->psz1,c_pszInfFile) == 0))
  321. {
  322. //
  323. // Its a .INF, note the fact for later processing
  324. //
  325. #ifdef EXTENDED_CAB_CONTENTS
  326. itType = itInfInCab;
  327. #else
  328. itType = itInvalid; // currently an invalid type
  329. #endif // EXTENDED_CAB_CONTENTS
  330. pnaArgs->pdaArgs->fContainsExeOrInf = TRUE;
  331. }
  332. else if (!(pnaArgs->dwAppFlags & AF_NO_PBDINCAB) && (lstrcmpi(pfdin->psz1,c_pszPbdFile) == 0))
  333. {
  334. //
  335. // Its a .PBD, note the fact for later processing
  336. //
  337. itType = itPbdInCab;
  338. }
  339. else if ((pszExt = CmStrchr(pfdin->psz1, TEXT('.'))) && (lstrcmpi(pszExt+1, c_pszPbk) == 0))
  340. {
  341. *pszExt = TEXT('\0');
  342. // if the PBK is not for this service, we don't use it.
  343. if (lstrcmpi(pfdin->psz1, pnaArgs->pdaArgs->pszPhoneBookName) != 0)
  344. {
  345. itType = itInvalid;
  346. }
  347. else
  348. {
  349. itType = itPbkInCab;
  350. }
  351. // restore the filename
  352. *pszExt = TEXT('.');
  353. }
  354. else if ((pszExt = CmStrchr(pfdin->psz1, TEXT('.'))) && (lstrcmpi(pszExt+1, c_pszPbr) == 0))
  355. {
  356. *pszExt = TEXT('\0');
  357. // if the PBR is not for this service, we don't use it.
  358. if (lstrcmpi(pfdin->psz1, pnaArgs->pdaArgs->pszPhoneBookName) != 0)
  359. {
  360. itType = itInvalid;
  361. }
  362. else
  363. {
  364. itType = itPbrInCab;
  365. }
  366. // restore the filename
  367. *pszExt = TEXT('.');
  368. // save the name in pdaArgs
  369. // if (!(pnaArgs->pdaArgs->pszNewPbrFile = CmStrCpyAlloc(pfdin->psz1)))
  370. // {
  371. // MYDBG((TEXT("fdi_notify(): CmStrCpyAlloc for pszNewPbrFile failed.")));
  372. // return -1;
  373. // }
  374. }
  375. else if (lstrcmpi(pfdin->psz1, c_pszVerFile) == 0)
  376. {
  377. // a version file - we don't process it. We'll read the version in fdintCLOSE_FILE_INFO
  378. itType = itInvalid;
  379. }
  380. #ifdef EXTENDED_CAB_CONTENTS
  381. else if (!(pnaArgs->dwAppFlags & AF_NO_SHLINCAB))
  382. {
  383. itType = itShlInCab;
  384. pnaArgs->pdaArgs->fContainsShl = TRUE;
  385. }
  386. #endif // EXTENDED_CAB_CONTENTS
  387. else
  388. {
  389. itType = itInvalid;
  390. }
  391. // create a file process info. add one to the existing list.
  392. if (itType != itInvalid)
  393. {
  394. if (!pnaArgs->pdaArgs->rgfpiFileProcessInfo)
  395. pFPI = (PFILEPROCESSINFO)CmMalloc(sizeof(FILEPROCESSINFO));
  396. else
  397. pFPI = (PFILEPROCESSINFO)CmRealloc(pnaArgs->pdaArgs->rgfpiFileProcessInfo,
  398. (pnaArgs->pdaArgs->dwNumFilesToProcess+1)*sizeof(FILEPROCESSINFO));
  399. if (!pFPI)
  400. {
  401. MYDBG((TEXT("fdi_notify(): Malloc(FILEPROCESSINFO) failed.")));
  402. return -1;
  403. }
  404. pnaArgs->pdaArgs->rgfpiFileProcessInfo = pFPI;
  405. pnaArgs->pdaArgs->dwNumFilesToProcess++;
  406. pFPI[pnaArgs->pdaArgs->dwNumFilesToProcess-1].itType = itType;
  407. pFPI[pnaArgs->pdaArgs->dwNumFilesToProcess-1].pszFile = CmStrCpyAlloc(pfdin->psz1);
  408. }
  409. // Do standard fdintCOPY_FILE processing, create the file and return the handle
  410. CFDIFileFile *pfff;
  411. BOOL bRes;
  412. pfff = new CFDIFileFile;
  413. if (!pfff)
  414. {
  415. MYDBG(("fdi_notify() new CFDIFileFile failed."));
  416. return (-1);
  417. }
  418. bRes = pfff->CreateFile(szTmp,GENERIC_WRITE,FILE_SHARE_READ,CREATE_NEW,FILE_ATTRIBUTE_NORMAL,0);
  419. if (!bRes)
  420. {
  421. delete pfff;
  422. return (-1);
  423. }
  424. return ((INT_PTR)pfff);
  425. }
  426. case fdintCLOSE_FILE_INFO:
  427. {
  428. int iRes;
  429. TCHAR szTmp[MAX_PATH+1];
  430. // Append file name to cab dir
  431. lstrcpy(szTmp,pnaArgs->pdaArgs->szCabDir);
  432. if (szTmp[0] && (GetLastChar(szTmp) != '\\'))
  433. {
  434. lstrcat(szTmp,TEXT("\\"));
  435. }
  436. lstrcat(szTmp,pfdin->psz1);
  437. // Set the date and time to the original file time not the current time
  438. FILETIME ftTmp;
  439. FILETIME ftTime;
  440. bRes = DosDateTimeToFileTime(pfdin->date,pfdin->time,&ftTmp);
  441. MYDBGTST(!bRes,("fdi_notify() DosDateTimeToFileTime(%u,%u) failed, GLE=%u.",pfdin->date,pfdin->time,GetLastError()));
  442. bRes = LocalFileTimeToFileTime(&ftTmp,&ftTime);
  443. MYDBGTST(!bRes,("fdi_notify() LocalFileTimeToFileTime() failed, GLE=%u.",GetLastError()));
  444. bRes = SetFileTime(((CFDIFile *) (pfdin->hf))->GetHandle(),&ftTime,&ftTime,&ftTime);
  445. MYDBGTST(!bRes,("fdi_notify() SetFileTime() failed, GLE=%u.",GetLastError()));
  446. iRes = fdi_close(pfdin->hf);
  447. // If this is the version file, get the version number
  448. if (lstrcmpi(pfdin->psz1,c_pszVerFile) == 0)
  449. {
  450. pnaArgs->pdaArgs->pszVerNew = GetVersionFromFile(szTmp);
  451. }
  452. // Set file attributes according to original file attribs
  453. bRes = SetFileAttributes(szTmp,pfdin->attribs);
  454. MYDBGTST(!bRes,("fdi_notify() SetFileAttributes(%s,%u) failed, GLE=%u.",szTmp,pfdin->attribs,GetLastError()));
  455. return ((iRes==0)?TRUE:FALSE);
  456. }
  457. case fdintNEXT_CABINET:
  458. MYDBG(("fdi_notify_scan() spanning cabinets is not supported."));
  459. return (-1);
  460. default:
  461. MYDBG(("fdi_notify_scan() fdint=%u.",fdint));
  462. break;
  463. }
  464. return (0);
  465. }