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.

2549 lines
95 KiB

  1. //____________________________________________________________________________
  2. //
  3. // (C) Copyright Seagate Software, Inc. 1994-1996
  4. // � 1998 Seagate Software, Inc. All rights reserved.
  5. //
  6. // All Rights Reserved Worldwide.
  7. //
  8. //____________________________________________________________________________
  9. //
  10. // FILE NAME : mtf_api.c
  11. //
  12. // DESCRIPTION : mtf api implementation
  13. //
  14. // CREATED: 6/20/95
  15. //
  16. //____________________________________________________________________________
  17. //
  18. // $Revision: 1.35 $
  19. // $Date: 02 Feb 1995 15:47:04 $
  20. // $Modtime: 02 Feb 1995 15:37:38 $
  21. //
  22. //____________________________________________________________________________
  23. // *****************************************************************************/
  24. #include <assert.h>
  25. #include <time.h>
  26. #include <string.h>
  27. #include <wchar.h>
  28. #include <stralign.h>
  29. #include "stdafx.h"
  30. #include "mtfapi.h"
  31. static USHORT iCountMTFa = 0; // Count of existing objects
  32. //
  33. // Constructor
  34. //
  35. CMTFApi::CMTFApi(void)
  36. {
  37. m_uAlignmentFactor = 0;
  38. m_iNumStrings = 0;
  39. for (int i = 0; i < iNUMSTRINGS; ++i){
  40. m_aszStrings[i] = 0;
  41. }
  42. iCountMTFa++;
  43. }
  44. //
  45. // Destructor
  46. //
  47. CMTFApi::~CMTFApi()
  48. {
  49. ClearStrings();
  50. iCountMTFa--;
  51. }
  52. /* ==================================================================================
  53. String Management
  54. When reading blocks, the strings are not null terminated -- we would like to
  55. pull them out and deliver them back in the ####_INFO structures in a civilized
  56. (null terminated) way. Thus, just set up an array of malloc'ec strings.
  57. Each call that uses strings should first call "ClearStrings" -- strings returned
  58. to the user will only be good up until the next call...
  59. ================================================================================= */
  60. // - returns the size of a wchar_t string
  61. // and returns zero for a null pointer
  62. size_t CMTFApi::wstrsize(wchar_t *s)
  63. {
  64. if (s)
  65. return wcslen(s) * sizeof(wchar_t);
  66. else
  67. return 0;
  68. }
  69. // - returns the size of a wchar_t unaligned string
  70. // and returns zero for a null pointer
  71. size_t CMTFApi::ua_wstrsize(wchar_t UNALIGNED *s)
  72. {
  73. if (s)
  74. return ua_wcslen(s) * sizeof(wchar_t);
  75. else
  76. return 0;
  77. }
  78. // - frees all allocated pointers in m_aszStrings and sets
  79. // m_iNumStrings to zero
  80. void CMTFApi::ClearStrings()
  81. {
  82. int i;
  83. for (i = 0; i < iNUMSTRINGS; ++i)
  84. {
  85. if (m_aszStrings[i])
  86. free(m_aszStrings[i]);
  87. m_aszStrings[i] = 0;
  88. }
  89. m_iNumStrings = 0;
  90. }
  91. // - allocates a string in m_aszStrings that is a copy of pString
  92. // (pString need not be null terminated)
  93. // (note -- iSize is the size of the string in bytes -- not the length!!!!!
  94. wchar_t * CMTFApi::MakeString(wchar_t UNALIGNED * pString, size_t iSize)
  95. {
  96. size_t i;
  97. if (m_iNumStrings >= iNUMSTRINGS) {
  98. return NULL;
  99. }
  100. m_aszStrings[m_iNumStrings] = (wchar_t *)malloc(iSize + sizeof(wchar_t));
  101. if (!m_aszStrings[m_iNumStrings])
  102. return NULL;
  103. for (i = 0; i < iSize / sizeof(wchar_t); ++i)
  104. m_aszStrings[m_iNumStrings][i] = pString[i];
  105. m_aszStrings[m_iNumStrings][i] = L'\0';
  106. return m_aszStrings[m_iNumStrings++];
  107. }
  108. /* ==================================================================================
  109. Other data structures
  110. ================================================================================= */
  111. #pragma pack(1)
  112. /***********************************************************************************
  113. ************************************************************************************
  114. ************************************************************************************
  115. **** MTF On Tape Structures
  116. ************************************************************************************
  117. ************************************************************************************
  118. ***********************************************************************************/
  119. /* ==================================================================================
  120. Common DBLK Header
  121. - The common dblk header exactly as it appears on tape in the head of the dblks
  122. ================================================================================== */
  123. typedef struct {
  124. UINT8 acBlockType[4]; /* 00h Unique identifier, see above */
  125. UINT32 uBlockAttributes; /* 04h Common attributes for this block */
  126. UINT16 uOffsetToFirstStream; /* 08h Offset to data associated with this */
  127. /* DBLK, or offset to next DBLK or */
  128. /* filemark if there is no associated */
  129. /* data. */
  130. UINT8 uOSID; /* 0Ah Machine/OS id where written, low byte */
  131. UINT8 uOSVersion; /* 0Bh Machine/OS id where written, high byte */
  132. UINT64 uDisplayableSize; /* 0Ch Displayable data size */
  133. UINT64 uFormatLogicalAddress; /* 14h Logical blk address relative to SSET */
  134. UINT16 uReservedForMBC; /* 1Ch Reserved for Media Based Catalog */
  135. UINT16 uSoftwareCompression; /* 1Eh Software Compression Algorithm ***/
  136. UINT8 acReserved1[4]; /* 20h reserved */
  137. UINT32 uControlBlockId; /* 24h Used for error recovery */
  138. UINT8 acReserved2[4]; /* 28h reserved */
  139. MTF_TAPE_ADDRESS sOSSpecificData; /* 2Ch Size and offset of OS specific stuff */
  140. UINT8 uStringType; /* 30h ASCII, Unicode, etc. */
  141. UINT8 uReserved3; /* 31h for alignment purposes */
  142. UINT16 uHeaderCheckSum; /* 32h Checksum of the block header. The */
  143. /* algorithm is: XOR each word preceeding */
  144. /* this one and store the result here. */
  145. /* (When the checksum is verified the */
  146. /* 'block_type' is also checked for a */
  147. /* non-zero value. */
  148. } MTF_DBLK_HDR;
  149. /* ==================================================================================
  150. DBLK TAPE Header
  151. - The TAPE DBLK, exactly as it appears on tape, including the common DBLK header (MTF_DBLK_HDR)
  152. ================================================================================== */
  153. typedef struct { /* MTF_DBLK_TAPE */
  154. MTF_DBLK_HDR sBlockHeader;
  155. UINT32 uTapeFamilyId;
  156. UINT32 uTapeAttributes;
  157. UINT16 uTapeSequenceNumber;
  158. UINT16 uPasswordEncryptionAlgorithm;
  159. UINT16 uSoftFilemarkBlockSize; /* Or ECC Algorithm */
  160. UINT16 uTapeCatalogType;
  161. MTF_TAPE_ADDRESS sTapeName;
  162. MTF_TAPE_ADDRESS sTapeDescription;
  163. MTF_TAPE_ADDRESS sTapePassword;
  164. MTF_TAPE_ADDRESS sSoftware_name;
  165. UINT16 uAlignmentFactor;
  166. UINT16 uSoftwareVendorId;
  167. MTF_DATE_TIME sTapeDate;
  168. UINT8 uMTFMajorVersion;
  169. } MTF_DBLK_TAPE;
  170. /* ==================================================================================
  171. Start of Set DBLK (SSET)
  172. - The SSET DBLK, exactly as it appears on tape, including the common DBLK header (MTF_DBLK_HDR)
  173. ================================================================================== */
  174. typedef struct {
  175. MTF_DBLK_HDR sBlockHeader;
  176. UINT32 uSSETAttributes;
  177. UINT16 uPasswordEncryptionAlgorithm;
  178. UINT16 uDataEncryptionAlgorithm; /* Or Software Compression Algorithm ***/
  179. UINT16 uSoftwareVendorId;
  180. UINT16 uDataSetNumber;
  181. MTF_TAPE_ADDRESS sDataSetName;
  182. MTF_TAPE_ADDRESS sDataSetDescription;
  183. MTF_TAPE_ADDRESS sDataSetPassword;
  184. MTF_TAPE_ADDRESS sUserName;
  185. UINT64 uPhysicalBlockAddress;
  186. MTF_DATE_TIME sMediaWriteDate;
  187. UINT8 uSoftwareVerMjr;
  188. UINT8 uSoftwareVerMnr;
  189. UINT8 uTimeZone;
  190. UINT8 uMTFMinorVer;
  191. UINT8 uTapeCatalogVersion;
  192. } MTF_DBLK_SSET;
  193. /* ==================================================================================
  194. Volume DBLK (VOLB)
  195. - The VOLB DBLK, exactly as it appears on tape, including the common DBLK header (MTF_DBLK_HDR)
  196. ================================================================================== */
  197. typedef struct {
  198. MTF_DBLK_HDR sBlockHeader;
  199. UINT32 uVolumeAttributes;
  200. MTF_TAPE_ADDRESS sDeviceName;
  201. MTF_TAPE_ADDRESS sVolumeName;
  202. MTF_TAPE_ADDRESS sMachineName;
  203. MTF_DATE_TIME sMediaWriteDate;
  204. } MTF_DBLK_VOLB;
  205. /* ==================================================================================
  206. Directory DBLK (DIRB)
  207. - The DIRB DBLK, exactly as it appears on tape, including the common DBLK header (MTF_DBLK_HDR)
  208. ================================================================================== */
  209. typedef struct {
  210. MTF_DBLK_HDR sBlockHeader;
  211. UINT32 uDirectoryAttributes;
  212. MTF_DATE_TIME sLastModificationDate;
  213. MTF_DATE_TIME sCreationDate;
  214. MTF_DATE_TIME sBackupDate;
  215. MTF_DATE_TIME sLastAccessDate;
  216. UINT32 uDirectoryId;
  217. MTF_TAPE_ADDRESS sDirectoryName;
  218. } MTF_DBLK_DIRB;
  219. /* ==================================================================================
  220. Directory DBLK (FILE)
  221. - The FILE DBLK, exactly as it appears on tape, including the common DBLK header (MTF_DBLK_HDR)
  222. ================================================================================== */
  223. typedef struct {
  224. MTF_DBLK_HDR sBlockHeader;
  225. UINT32 uFileAttributes;
  226. MTF_DATE_TIME sLastModificationDate;
  227. MTF_DATE_TIME sCreationDate;
  228. MTF_DATE_TIME sBackupDate;
  229. MTF_DATE_TIME sLastAccessDate;
  230. UINT32 uDirectoryId;
  231. UINT32 uFileId;
  232. MTF_TAPE_ADDRESS sFileName;
  233. } MTF_DBLK_FILE;
  234. #pragma pack()
  235. /* ==================================================================================
  236. Corrupt File DBLK (CFIL)
  237. - use MTF_DBLK_CFIL_INFO -- same structure
  238. ================================================================================== */
  239. typedef MTF_DBLK_CFIL_INFO MTF_DBLK_CFIL;
  240. /* ==================================================================================
  241. End of Set Pad Block (ESPB)
  242. ================================================================================== */
  243. // consists only of header
  244. /* ==================================================================================
  245. End of Set Block (ESET)
  246. - use MTF_DBLK_ESET_INFO -- same structure
  247. ================================================================================== */
  248. typedef MTF_DBLK_ESET_INFO MTF_DBLK_ESET;
  249. /* ==================================================================================
  250. End of Set Block (EOTM)
  251. - use MTF_DBLK_EOTM_INFO -- same structure
  252. ================================================================================== */
  253. typedef MTF_DBLK_EOTM_INFO MTF_DBLK_EOTM;
  254. /* ==================================================================================
  255. Soft Filemark (SFMB)
  256. - use MTF_DBLK_SFMB_INFO -- same structure
  257. ================================================================================== */
  258. typedef MTF_DBLK_SFMB_INFO MTF_DBLK_SFMB;
  259. /* ==================================================================================
  260. StreamHeader
  261. - use MTF_STREAM_INFO -- same structure
  262. ================================================================================== */
  263. typedef MTF_STREAM_INFO MTF_STREAM;
  264. /***********************************************************************************
  265. ************************************************************************************
  266. ************************************************************************************
  267. **** MTF Misc Data Types
  268. ************************************************************************************
  269. ************************************************************************************
  270. ***********************************************************************************/
  271. /* ==================================================================================
  272. Alignment Factor
  273. ================================================================================== */
  274. /***********************************************************************************
  275. * MTF_SetAlignmentFactor()
  276. * ** MTF API FUNCTION **
  277. ***********************************************************************************/
  278. void CMTFApi::MTF_SetAlignmentFactor(UINT16 uAF)
  279. {
  280. // store the user's alignment factor in a private member
  281. m_uAlignmentFactor = uAF;
  282. }
  283. /***********************************************************************************
  284. * MTF_GetAlignmentFactor()
  285. * ** MTF API FUNCTION **
  286. ***********************************************************************************/
  287. UINT16 CMTFApi::MTF_GetAlignmentFactor()
  288. {
  289. return m_uAlignmentFactor;
  290. }
  291. /***********************************************************************************
  292. * MTF_PadToNextAlignmentFactor()
  293. * ** MTF API FUNCTION **
  294. ***********************************************************************************/
  295. DWORD CMTFApi::MTF_PadToNextAlignmentFactor(
  296. BYTE *pBuffer,
  297. size_t nBufUsed,
  298. size_t nBufferSize,
  299. size_t *pnSizeUsed)
  300. {
  301. size_t i;
  302. size_t nAlignment;
  303. MTF_STREAM_INFO sStream;
  304. // figure out what the next alignment value is and then pad out the user's buffer
  305. // with an SPAD, making sure the buffer is big enough
  306. nAlignment = Align(nBufUsed + sizeof(MTF_STREAM_INFO), MTF_GetAlignmentFactor());
  307. *pnSizeUsed = nAlignment;
  308. if (nBufferSize < nAlignment)
  309. return MTF_ERROR_BUFFER_TOO_SMALL;
  310. MTF_SetSTREAMDefaults(&sStream, "SPAD");
  311. sStream.uStreamLength = MTF_CreateUINT64(nAlignment - nBufUsed - sizeof(MTF_STREAM_INFO), 0);
  312. MTF_WriteStreamHeader(&sStream,
  313. pBuffer + nBufUsed,
  314. nBufferSize - nBufUsed,
  315. 0);
  316. for (i = nBufUsed + sizeof(MTF_STREAM_INFO); i < nAlignment; ++i)
  317. pBuffer[i] = 0;
  318. return MTF_ERROR_NONE;
  319. }
  320. /***********************************************************************************
  321. * MTF_PadToNextPhysicalBlockBoundary() - (bmd)
  322. * ** MTF API FUNCTION **
  323. ***********************************************************************************/
  324. DWORD CMTFApi::MTF_PadToNextPhysicalBlockBoundary(
  325. BYTE *pBuffer,
  326. size_t nBlockSize,
  327. size_t nBufUsed,
  328. size_t nBufferSize,
  329. size_t *pnSizeUsed)
  330. {
  331. size_t i;
  332. size_t nAlignment;
  333. MTF_STREAM_INFO sStream;
  334. // figure out what the next alignment value is and then pad out the user's buffer
  335. // with an SPAD, making sure the buffer is big enough
  336. nAlignment = Align(nBufUsed + sizeof(MTF_STREAM_INFO), nBlockSize);
  337. *pnSizeUsed = nAlignment;
  338. if (nBufferSize < nAlignment)
  339. return MTF_ERROR_BUFFER_TOO_SMALL;
  340. MTF_SetSTREAMDefaults(&sStream, "SPAD");
  341. sStream.uStreamLength = MTF_CreateUINT64(nAlignment - nBufUsed - sizeof(MTF_STREAM_INFO), 0);
  342. MTF_WriteStreamHeader(&sStream, pBuffer + nBufUsed, nBufferSize - nBufUsed, 0);
  343. for (i = nBufUsed + sizeof(MTF_STREAM_INFO); i < nAlignment; ++i)
  344. pBuffer[i] = 0;
  345. return MTF_ERROR_NONE;
  346. }
  347. /***********************************************************************************
  348. * MTF_CreateUINT64()
  349. * ** MTF API FUNCTION **
  350. ***********************************************************************************/
  351. UINT64 CMTFApi::MTF_CreateUINT64(UINT32 uLSB, UINT32 uMSB)
  352. {
  353. UINT64 uRet;
  354. uRet = (UINT64) uMSB << 32;
  355. uRet += uLSB;
  356. return uRet;
  357. }
  358. /* ==================================================================================
  359. Compressed date structure for storing dates in minimal space on tape:
  360. BYTE 0 BYTE 1 BYTE 2 BYTE 3 BYTE 4
  361. 76543210 76543210 76543210 76543210 76543210
  362. yyyyyyyy yyyyyymm mmdddddh hhhhmmmm mmssssss
  363. 33333333 33222222 22221111 11111100 00000000
  364. 98765432 10987654 32109876 54321098 76543210
  365. ================================================================================== */
  366. /***********************************************************************************
  367. * MTF_CreateDateTime()
  368. * ** MTF API FUNCTION **
  369. ***********************************************************************************/
  370. MTF_DATE_TIME CMTFApi::MTF_CreateDateTime(
  371. int iYear,
  372. int iMonth,
  373. int iDay,
  374. int iHour,
  375. int iMinute,
  376. int iSecond
  377. )
  378. {
  379. MTF_DATE_TIME sDateTime = {0};
  380. UINT16 temp ;
  381. // pack the date time structure with the arguments as per the diagram above
  382. temp = (UINT16)iYear << 2 ;
  383. sDateTime.dt_field[0] = ((UINT8 *)&temp)[1] ;
  384. sDateTime.dt_field[1] = ((UINT8 *)&temp)[0] ;
  385. temp = (UINT16)iMonth << 6 ;
  386. sDateTime.dt_field[1] |= ((UINT8 *)&temp)[1] ;
  387. sDateTime.dt_field[2] = ((UINT8 *)&temp)[0] ;
  388. temp = (UINT16)iDay << 1 ;
  389. sDateTime.dt_field[2] |= ((UINT8 *)&temp)[0] ;
  390. temp = (UINT16)iHour << 4 ;
  391. sDateTime.dt_field[2] |= ((UINT8 *)&temp)[1] ;
  392. sDateTime.dt_field[3] = ((UINT8 *)&temp)[0] ;
  393. temp = (UINT16)iMinute << 6 ;
  394. sDateTime.dt_field[3] |= ((UINT8 *)&temp)[1] ;
  395. sDateTime.dt_field[4] = ((UINT8 *)&temp)[0] ;
  396. temp = (UINT16)iSecond ;
  397. sDateTime.dt_field[4] |= ((UINT8 *)&temp)[0] ;
  398. return sDateTime;
  399. }
  400. /***********************************************************************************
  401. * MTF_CreateDateTimeFromTM()
  402. * ** MTF API FUNCTION **
  403. ***********************************************************************************/
  404. MTF_DATE_TIME CMTFApi::MTF_CreateDateTimeFromTM(
  405. struct tm *pT
  406. )
  407. {
  408. // translate call to MTF_CreateDateTime
  409. return MTF_CreateDateTime(pT->tm_year + 1900, pT->tm_mon + 1, pT->tm_mday, pT->tm_hour, pT->tm_min, pT->tm_sec);
  410. }
  411. /***********************************************************************************
  412. * MTF_CreateDateTimeToTM()
  413. * ** MTF API FUNCTION **
  414. ***********************************************************************************/
  415. void CMTFApi::MTF_CreateDateTimeToTM(
  416. MTF_DATE_TIME *pDT,
  417. struct tm *pT
  418. )
  419. {
  420. UINT8 temp[2] ;
  421. // unpack the MTF_DATE_TIME structure and store the results
  422. temp[0] = pDT->dt_field[1] ;
  423. temp[1] = pDT->dt_field[0] ;
  424. pT->tm_year = *((UINT16 *)temp) >> 2 ;
  425. temp[0] = pDT->dt_field[2] ;
  426. temp[1] = pDT->dt_field[1] ;
  427. pT->tm_mon = (*((UINT16 *)temp) >> 6) & 0x000F ;
  428. pT->tm_mday = (*((UINT16 *)temp) >> 1) & 0x001F ;
  429. temp[0] = pDT->dt_field[3] ;
  430. temp[1] = pDT->dt_field[2] ;
  431. pT->tm_hour = (*((UINT16 *)temp) >> 4) & 0x001F ;
  432. temp[0] = pDT->dt_field[4] ;
  433. temp[1] = pDT->dt_field[3] ;
  434. pT->tm_min = (*((UINT16 *)temp) >> 6) & 0x003F ;
  435. pT->tm_sec = *((UINT16 *)temp) & 0x003F ;
  436. }
  437. /***********************************************************************************
  438. * MTF_CreateDateNull()
  439. * ** MTF API FUNCTION **
  440. ***********************************************************************************/
  441. MTF_DATE_TIME CMTFApi::MTF_CreateDateNull()
  442. {
  443. MTF_DATE_TIME sDateTime = {0};
  444. return sDateTime;
  445. }
  446. /***********************************************************************************
  447. * MTF_CreateDateTimeFromFileTime()
  448. * ** MTF API FUNCTION **
  449. ***********************************************************************************/
  450. MTF_DATE_TIME CMTFApi::MTF_CreateDateTimeFromFileTime(
  451. FILETIME sFileTime
  452. )
  453. {
  454. SYSTEMTIME sSystemTime;
  455. FileTimeToSystemTime(&sFileTime, &sSystemTime);
  456. return MTF_CreateDateTime(sSystemTime.wYear,
  457. sSystemTime.wMonth,
  458. sSystemTime.wDay,
  459. sSystemTime.wHour,
  460. sSystemTime.wMinute,
  461. sSystemTime.wSecond);
  462. }
  463. /***********************************************************************************
  464. ************************************************************************************
  465. **** MTF internal HELPER FUNCITONS
  466. ************************************************************************************
  467. ***********************************************************************************/
  468. /***********************************************************************************
  469. * StringToTapeAddress()
  470. *
  471. * Description: Used by the MTF_Write#### functions below. Given a Buffer, an
  472. * MTF_TAPE_ADDRESS struct and the current end of the string storage
  473. * area in the buffer, this function appends the string to the string
  474. * storage area, fills in the MTF_TAPE_ADDRESS struct indicating where
  475. * the string was stored and returns the new end of the string storage
  476. * area accounting for the added string.
  477. ***********************************************************************************/
  478. size_t CMTFApi::StringToTapeAddress(
  479. MTF_TAPE_ADDRESS *pAd, // the mtf tape address structure to fill
  480. BYTE *pBuffer, // the buffer that is being filled
  481. wchar_t *str, // the string to store MTF style in the buffer
  482. size_t uCurrentStorageOffset // the next available point in the buffer for string storage
  483. )
  484. {
  485. // if we have a string,
  486. // - put the size and offset in the MTF_TAPE_ADDRESS structure and then copy
  487. // the string to the pBuffer at the uCurrentStorageOffset'th byte
  488. // otherwise
  489. // - put a zero size and offset in the MTF_TAPE_ADDRESS struct.
  490. // return the new end of the string storage area
  491. if (str)
  492. {
  493. pAd->uSize = (UINT16)wstrsize(str);
  494. pAd->uOffset = (UINT16)uCurrentStorageOffset;
  495. memcpy(pBuffer + uCurrentStorageOffset, str, pAd->uSize);
  496. uCurrentStorageOffset += pAd->uSize;
  497. }
  498. else
  499. {
  500. pAd->uSize = 0;
  501. pAd->uOffset = 0;
  502. }
  503. return uCurrentStorageOffset;
  504. }
  505. /***********************************************************************************
  506. * Align()
  507. *
  508. * Description: Given uSize and an alignment factor, retuns the value
  509. * of the uSize+ pad, where pad is the value necesary to
  510. * get to the next alignment factor.
  511. *
  512. * Returns uSize + pad -- not just pad!
  513. ***********************************************************************************/
  514. size_t CMTFApi::Align(
  515. size_t uSize,
  516. size_t uAlignment)
  517. {
  518. if (uSize % uAlignment)
  519. return uSize - (uSize % uAlignment) + uAlignment;
  520. else
  521. return uSize;
  522. }
  523. /***********************************************************************************
  524. * CalcChecksum()
  525. *
  526. * Description: returns the 16bit XOR sum of the nNum bytes starting at the UINT16
  527. * pointed to by pStartPtr
  528. *
  529. ***********************************************************************************/
  530. UINT16 CMTFApi::CalcChecksum(
  531. BYTE * pStartPtr,
  532. int nNum )
  533. {
  534. UINT16 resultSoFar = 0;
  535. UINT16 UNALIGNED *pCur = (UINT16 *) pStartPtr;
  536. while( nNum-- )
  537. resultSoFar ^= *pCur++ ;
  538. return( resultSoFar ) ;
  539. }
  540. /***********************************************************************************
  541. * CalcChecksumOfStreamData() - (bmd)
  542. *
  543. * Description: returns the 32bit XOR sum of the nNum bytes starting at the UINT64
  544. * pointed to by pStartPtr
  545. *
  546. ***********************************************************************************/
  547. UINT32 CMTFApi::CalcChecksumOfStreamData(
  548. BYTE * pStartPtr,
  549. int nNum )
  550. {
  551. UINT32 resultSoFar = 0;
  552. UINT32 UNALIGNED *pCur = (UINT32 *) pStartPtr;
  553. while( nNum-- )
  554. resultSoFar ^= *pCur++ ;
  555. return( resultSoFar ) ;
  556. }
  557. /***********************************************************************************
  558. ************************************************************************************
  559. ************************************************************************************
  560. **** MTF API STRUCTURE FUNCTIONS
  561. ************************************************************************************
  562. ************************************************************************************
  563. ***********************************************************************************/
  564. /* ==================================================================================
  565. =====================================================================================
  566. Common DBLK: MTF_DBLK_HDR_INFO
  567. =====================================================================================
  568. ================================================================================== */
  569. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  570. size_t CMTFApi::MTF_DBLK_HDR_INFO_CalcAddDataSize(
  571. MTF_DBLK_HDR_INFO *pSTDInfo
  572. )
  573. {
  574. return pSTDInfo->uOSDataSize;
  575. }
  576. /***********************************************************************************
  577. * MTF_SetDblkHdrDefaults()
  578. * ** MTF API FUNCTION **
  579. ***********************************************************************************/
  580. void CMTFApi::MTF_SetDblkHdrDefaults(
  581. MTF_DBLK_HDR_INFO * pStdInfo
  582. )
  583. {
  584. int i;
  585. for (i = 0; i < 5; ++i)
  586. pStdInfo->acBlockType[i] = 0;
  587. pStdInfo->uBlockAttributes = 0;
  588. pStdInfo->uOSID = 0;
  589. pStdInfo->uOSVersion = 0;
  590. pStdInfo->uDisplayableSize = 0;
  591. pStdInfo->uFormatLogicalAddress = 0;
  592. pStdInfo->uReservedForMBC = 0;
  593. pStdInfo->uSoftwareCompression = MTF_COMPRESS_NONE;
  594. pStdInfo->uControlBlockId = 0;
  595. pStdInfo->pvOSData = 0;
  596. pStdInfo->uOSDataSize = 0;
  597. pStdInfo->uStringType = MTF_STRING_UNICODE_STR;
  598. }
  599. /***********************************************************************************
  600. * MTF_WriteDblkHdrToBuffer()
  601. *
  602. * Description: called by the MTF_Write#####() functions to format the common block
  603. * header to the buffer
  604. * - this also calculates the header check sum and fills it in
  605. *
  606. * Pre: - *puCurrentStorageOffset is the offset at where string and OS Data storage will
  607. * begin in the buffer
  608. * - the size of the buffer has been checked and can hold any info written to it
  609. *
  610. * Post: - *puCurrentStorageOffset is updated to reflect any added strings or storage
  611. *
  612. ***********************************************************************************/
  613. void CMTFApi::MTF_WriteDblkHdrToBuffer(
  614. UINT8 acID[4], // four byte header id to write
  615. UINT16 uOffsetToFirstStream, // the size of the DBLK for which this will be a header
  616. MTF_DBLK_HDR_INFO *psHdrInfo, // the header info struct to use (filled in by client)
  617. BYTE *pBuffer, // the buffer to format to
  618. size_t *puCurrentStorage) // the point in the buffer where string and os data stroage begins
  619. // (this will be updated upon return to reflect added data to storage)
  620. {
  621. MTF_DBLK_HDR *pHDR = 0;
  622. UINT16 uCurrentStorageOffset = 0;
  623. int i;
  624. // - if no *puCurrentStorage, we assume storage starts at
  625. // the end of the on tape MTF_DBLK_HDR structure
  626. if (puCurrentStorage)
  627. uCurrentStorageOffset = (UINT16)*puCurrentStorage;
  628. else
  629. uCurrentStorageOffset = (UINT16)sizeof(MTF_DBLK_HDR);
  630. pHDR = (MTF_DBLK_HDR *) pBuffer;
  631. // write in the four byte DBLK ID
  632. for (i = 0; i < 4; ++i)
  633. pHDR->acBlockType[i] = acID[i];
  634. pHDR->uBlockAttributes = psHdrInfo->uBlockAttributes;
  635. pHDR->uOffsetToFirstStream = uOffsetToFirstStream;
  636. pHDR->uOSID = psHdrInfo->uOSID;
  637. pHDR->uOSVersion = psHdrInfo->uOSVersion;
  638. pHDR->uDisplayableSize = psHdrInfo->uDisplayableSize;
  639. pHDR->uFormatLogicalAddress = psHdrInfo->uFormatLogicalAddress;
  640. pHDR->uReservedForMBC = 0; // must be zero in backup set
  641. pHDR->uSoftwareCompression = psHdrInfo->uSoftwareCompression;
  642. pHDR->uControlBlockId = psHdrInfo->uControlBlockId;
  643. pHDR->sOSSpecificData.uSize = psHdrInfo->uOSDataSize;
  644. // write out the os specific data at the current storage offset and update it
  645. if (psHdrInfo->uOSDataSize)
  646. {
  647. pHDR->sOSSpecificData.uOffset = uCurrentStorageOffset;
  648. memcpy(pBuffer + uCurrentStorageOffset, psHdrInfo->pvOSData, psHdrInfo->uOSDataSize);
  649. uCurrentStorageOffset = uCurrentStorageOffset + psHdrInfo->uOSDataSize;
  650. }
  651. else
  652. {
  653. pHDR->sOSSpecificData.uOffset = 0;
  654. pHDR->sOSSpecificData.uSize = 0;
  655. }
  656. pHDR->uStringType = psHdrInfo->uStringType;
  657. pHDR->uHeaderCheckSum = CalcChecksum(pBuffer, sizeof(MTF_DBLK_HDR) / sizeof(UINT16) - 1);
  658. if (puCurrentStorage)
  659. *puCurrentStorage = uCurrentStorageOffset;
  660. }
  661. void CMTFApi::MTF_DBLK_HDR_INFO_ReadFromBuffer(
  662. MTF_DBLK_HDR_INFO *psHdrInfo,
  663. BYTE *pBuffer)
  664. {
  665. MTF_DBLK_HDR *pHDR = 0;
  666. size_t uCurrentStorageOffset = 0;
  667. int i;
  668. pHDR = (MTF_DBLK_HDR *) pBuffer;
  669. for (i = 0; i < 4; ++i)
  670. psHdrInfo->acBlockType[i] = pHDR->acBlockType[i];
  671. psHdrInfo->acBlockType[4] = 0;
  672. psHdrInfo->uOffsetToFirstStream = pHDR->uOffsetToFirstStream;
  673. psHdrInfo->uBlockAttributes = pHDR->uBlockAttributes ;
  674. psHdrInfo->uOSID = pHDR->uOSID;
  675. psHdrInfo->uOSVersion = pHDR->uOSVersion;
  676. psHdrInfo->uDisplayableSize = pHDR->uDisplayableSize;
  677. psHdrInfo->uFormatLogicalAddress= pHDR->uFormatLogicalAddress;
  678. psHdrInfo->uSoftwareCompression = pHDR->uSoftwareCompression;
  679. psHdrInfo->uControlBlockId = pHDR->uControlBlockId;
  680. psHdrInfo->uOSDataSize = pHDR->sOSSpecificData.uSize;
  681. psHdrInfo->pvOSData = (pBuffer + pHDR->sOSSpecificData.uOffset);
  682. psHdrInfo->uStringType = pHDR->uStringType;
  683. psHdrInfo->uHeaderCheckSum = pHDR->uHeaderCheckSum;
  684. }
  685. /* ==================================================================================
  686. =====================================================================================
  687. TAPE DBLK: MTF_DBLK_TAPE_INFO
  688. =====================================================================================
  689. ================================================================================== */
  690. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  691. // **NOT INCLUDING THE COMMON DBLK HEADER additional info **
  692. size_t CMTFApi::MTF_DBLK_TAPE_INFO_CalcAddDataSize(
  693. MTF_DBLK_TAPE_INFO *pTapeInfo
  694. )
  695. {
  696. return wstrsize(pTapeInfo->szTapeName) +
  697. wstrsize(pTapeInfo->szTapeDescription) +
  698. wstrsize(pTapeInfo->szTapePassword) +
  699. wstrsize(pTapeInfo->szSoftwareName);
  700. }
  701. /***********************************************************************************
  702. * MTF_SetTAPEDefaults()
  703. * ** MTF API FUNCTION **
  704. ***********************************************************************************/
  705. void CMTFApi::MTF_SetTAPEDefaults(
  706. MTF_DBLK_TAPE_INFO *pTapeInfo
  707. )
  708. {
  709. time_t tTime;
  710. time(&tTime);
  711. pTapeInfo->uTapeFamilyId = 0;
  712. pTapeInfo->uTapeAttributes = 0;
  713. pTapeInfo->uTapeSequenceNumber = 0;
  714. pTapeInfo->uPasswordEncryptionAlgorithm = MTF_PW_ENCRYPT_NONE;
  715. pTapeInfo->uSoftFilemarkBlockSize = 0;
  716. pTapeInfo->uTapeCatalogType = MTF_OTC_NONE; // MTF_OTC_TYPE
  717. pTapeInfo->szTapeName = 0 ;
  718. pTapeInfo->szTapeDescription = 0 ;
  719. pTapeInfo->szTapePassword = 0;
  720. pTapeInfo->szSoftwareName = 0;
  721. pTapeInfo->uAlignmentFactor = MTF_GetAlignmentFactor();
  722. pTapeInfo->uSoftwareVendorId = 0;
  723. pTapeInfo->sTapeDate = MTF_CreateDateTimeFromTM(gmtime(&tTime));
  724. pTapeInfo->uMTFMajorVersion = MTF_FORMAT_VER_MAJOR;
  725. }
  726. /***********************************************************************************
  727. * MTF_WriteTAPEDblk()
  728. * ** MTF API FUNCTION **
  729. ***********************************************************************************/
  730. DWORD CMTFApi::MTF_WriteTAPEDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  731. MTF_DBLK_TAPE_INFO *psTapeInfo,
  732. BYTE *pBuffer,
  733. size_t nBufferSize,
  734. size_t *pnSizeUsed)
  735. {
  736. UINT16 uOffsetToFirstStream;
  737. //
  738. // Figure the size of the entire DBLK & make sure we have room
  739. //
  740. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_TAPE) +
  741. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo) +
  742. MTF_DBLK_TAPE_INFO_CalcAddDataSize(psTapeInfo));
  743. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  744. if (nBufferSize < uOffsetToFirstStream)
  745. {
  746. if (pnSizeUsed)
  747. *pnSizeUsed = uOffsetToFirstStream;
  748. return MTF_ERROR_BUFFER_TOO_SMALL;
  749. }
  750. memset(pBuffer, 0, uOffsetToFirstStream);
  751. //
  752. // write the header and then fill in the stuff from this info struct
  753. //
  754. {
  755. MTF_DBLK_TAPE *pTape = 0;
  756. size_t uCurrentStorageOffset = 0;
  757. uCurrentStorageOffset = sizeof(MTF_DBLK_TAPE);
  758. MTF_WriteDblkHdrToBuffer(
  759. (UINT8 *)(char *)MTF_ID_TAPE,
  760. uOffsetToFirstStream,
  761. psHdrInfo,
  762. pBuffer,
  763. &uCurrentStorageOffset);
  764. pTape = (MTF_DBLK_TAPE *) pBuffer;
  765. pTape->uTapeFamilyId = psTapeInfo->uTapeFamilyId;
  766. pTape->uTapeAttributes = psTapeInfo->uTapeAttributes;
  767. pTape->uTapeSequenceNumber = psTapeInfo->uTapeSequenceNumber;
  768. pTape->uPasswordEncryptionAlgorithm = psTapeInfo->uPasswordEncryptionAlgorithm;
  769. pTape->uSoftFilemarkBlockSize = psTapeInfo->uSoftFilemarkBlockSize;
  770. pTape->uTapeCatalogType = psTapeInfo->uTapeCatalogType;
  771. uCurrentStorageOffset = StringToTapeAddress(&pTape->sTapeName, pBuffer, psTapeInfo->szTapeName, uCurrentStorageOffset);
  772. uCurrentStorageOffset = StringToTapeAddress(&pTape->sTapeDescription, pBuffer, psTapeInfo->szTapeDescription, uCurrentStorageOffset);
  773. uCurrentStorageOffset = StringToTapeAddress(&pTape->sTapePassword, pBuffer, psTapeInfo->szTapePassword, uCurrentStorageOffset);
  774. uCurrentStorageOffset = StringToTapeAddress(&pTape->sSoftware_name, pBuffer, psTapeInfo->szSoftwareName, uCurrentStorageOffset);
  775. pTape->uAlignmentFactor = psTapeInfo->uAlignmentFactor;
  776. pTape->uSoftwareVendorId = psTapeInfo->uSoftwareVendorId;
  777. pTape->sTapeDate = psTapeInfo->sTapeDate;
  778. pTape->uMTFMajorVersion = psTapeInfo->uMTFMajorVersion;
  779. if (pnSizeUsed)
  780. *pnSizeUsed = uOffsetToFirstStream;
  781. }
  782. return MTF_ERROR_NONE;
  783. }
  784. /***********************************************************************************
  785. * MTF_ReadTAPEDblk()
  786. * ** MTF API FUNCTION **
  787. ***********************************************************************************/
  788. DWORD CMTFApi::MTF_ReadTAPEDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  789. MTF_DBLK_TAPE_INFO *psTapeInfo,
  790. BYTE *pBuffer)
  791. {
  792. MTF_DBLK_TAPE *pTape = 0;
  793. ClearStrings();
  794. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  795. pTape = (MTF_DBLK_TAPE *) pBuffer;
  796. psTapeInfo->uTapeFamilyId = pTape->uTapeFamilyId;
  797. psTapeInfo->uTapeAttributes = pTape->uTapeAttributes;
  798. psTapeInfo->uTapeSequenceNumber = pTape->uTapeSequenceNumber;
  799. psTapeInfo->uPasswordEncryptionAlgorithm = pTape->uPasswordEncryptionAlgorithm;
  800. psTapeInfo->uSoftFilemarkBlockSize = pTape->uSoftFilemarkBlockSize;
  801. psTapeInfo->uTapeCatalogType = pTape->uTapeCatalogType;
  802. psTapeInfo->uAlignmentFactor = pTape->uAlignmentFactor;
  803. psTapeInfo->uSoftwareVendorId = pTape->uSoftwareVendorId;
  804. psTapeInfo->sTapeDate = pTape->sTapeDate;
  805. psTapeInfo->uMTFMajorVersion = pTape->uMTFMajorVersion;
  806. psTapeInfo->szTapeName = MakeString((wchar_t UNALIGNED *) (pBuffer + pTape->sTapeName.uOffset), pTape->sTapeName.uSize);
  807. psTapeInfo->szTapeDescription = MakeString((wchar_t UNALIGNED *) (pBuffer + pTape->sTapeDescription.uOffset), pTape->sTapeDescription.uSize);
  808. psTapeInfo->szTapePassword = MakeString((wchar_t UNALIGNED *) (pBuffer + pTape->sTapePassword.uOffset), pTape->sTapePassword.uSize);
  809. psTapeInfo->szSoftwareName = MakeString((wchar_t UNALIGNED *) (pBuffer + pTape->sSoftware_name.uOffset), pTape->sSoftware_name.uSize);
  810. if ( !psTapeInfo->szTapeName || !psTapeInfo->szTapeDescription || !psTapeInfo->szTapePassword || !psTapeInfo->szSoftwareName)
  811. return MTF_OUT_OF_MEMORY;
  812. return MTF_ERROR_NONE;
  813. }
  814. /* ==================================================================================
  815. =====================================================================================
  816. SSET DBLK: MTF_DBLK_SSET_INFO
  817. =====================================================================================
  818. ================================================================================== */
  819. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  820. // **NOT INCLUDING THE COMMON DBLK HEADER additional info **
  821. size_t CMTFApi::MTF_DBLK_SSET_INFO_CalcAddDataSize(
  822. MTF_DBLK_SSET_INFO *pSSETInfo
  823. )
  824. {
  825. return wstrsize(pSSETInfo->szDataSetName)
  826. + wstrsize(pSSETInfo->szDataSetDescription)
  827. + wstrsize(pSSETInfo->szDataSetPassword)
  828. + wstrsize(pSSETInfo->szUserName);
  829. }
  830. /***********************************************************************************
  831. * MTF_SetSSETDefaults()
  832. * ** MTF API FUNCTION **
  833. ***********************************************************************************/
  834. void CMTFApi::MTF_SetSSETDefaults(
  835. MTF_DBLK_SSET_INFO *pSSETInfo
  836. )
  837. {
  838. time_t tTime;
  839. time(&tTime);
  840. pSSETInfo->uSSETAttributes = 0;
  841. pSSETInfo->uPasswordEncryptionAlgorithm = MTF_PW_ENCRYPT_NONE;
  842. pSSETInfo->uDataEncryptionAlgorithm = MTF_DATA_ENCRYPT_NONE;
  843. pSSETInfo->uSoftwareVendorId = 0;
  844. pSSETInfo->uDataSetNumber = 0;
  845. pSSETInfo->szDataSetName = 0 ;
  846. pSSETInfo->szDataSetDescription = 0 ;
  847. pSSETInfo->szDataSetPassword = 0 ;
  848. pSSETInfo->szUserName = 0 ;
  849. pSSETInfo->uPhysicalBlockAddress = 0;
  850. pSSETInfo->sMediaWriteDate = MTF_CreateDateTimeFromTM(gmtime(&tTime));
  851. pSSETInfo->uSoftwareVerMjr = 0;
  852. pSSETInfo->uSoftwareVerMnr = 0;
  853. pSSETInfo->uTimeZone = MTF_LOCAL_TZ;
  854. pSSETInfo->uMTFMinorVer = MTF_FORMAT_VER_MINOR;
  855. pSSETInfo->uTapeCatalogVersion = MTF_OTC_NONE; // MTF_OTC_VERSION
  856. }
  857. /***********************************************************************************
  858. * MTF_WriteSSETDblk()
  859. * ** MTF API FUNCTION **
  860. ***********************************************************************************/
  861. DWORD CMTFApi::MTF_WriteSSETDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  862. MTF_DBLK_SSET_INFO *psSSETInfo,
  863. BYTE *pBuffer,
  864. size_t nBufferSize,
  865. size_t *pnSizeUsed)
  866. {
  867. UINT16 uOffsetToFirstStream;
  868. //
  869. // Figure the size of the entire DBLK & make sure we have room
  870. //
  871. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_SSET) +
  872. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo) +
  873. MTF_DBLK_SSET_INFO_CalcAddDataSize(psSSETInfo));
  874. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  875. if (nBufferSize < uOffsetToFirstStream)
  876. {
  877. if (pnSizeUsed)
  878. *pnSizeUsed = uOffsetToFirstStream;
  879. return MTF_ERROR_BUFFER_TOO_SMALL;
  880. }
  881. memset(pBuffer, 0, uOffsetToFirstStream);
  882. {
  883. MTF_DBLK_SSET *psSSET = 0;
  884. size_t uCurrentStorageOffset = 0;
  885. uCurrentStorageOffset = sizeof(MTF_DBLK_SSET);
  886. MTF_WriteDblkHdrToBuffer(
  887. (UINT8 *)(char *)MTF_ID_SSET,
  888. uOffsetToFirstStream,
  889. psHdrInfo,
  890. pBuffer,
  891. &uCurrentStorageOffset);
  892. psSSET = (MTF_DBLK_SSET *) pBuffer;
  893. psSSET->uSSETAttributes = psSSETInfo->uSSETAttributes;
  894. psSSET->uPasswordEncryptionAlgorithm = psSSETInfo->uPasswordEncryptionAlgorithm;
  895. psSSET->uDataEncryptionAlgorithm = psSSETInfo->uDataEncryptionAlgorithm;
  896. psSSET->uSoftwareVendorId = psSSETInfo->uSoftwareVendorId;
  897. psSSET->uDataSetNumber = psSSETInfo->uDataSetNumber;
  898. uCurrentStorageOffset = StringToTapeAddress(&psSSET->sDataSetName, pBuffer, psSSETInfo->szDataSetName, uCurrentStorageOffset);
  899. uCurrentStorageOffset = StringToTapeAddress(&psSSET->sDataSetDescription, pBuffer, psSSETInfo->szDataSetDescription, uCurrentStorageOffset);
  900. uCurrentStorageOffset = StringToTapeAddress(&psSSET->sDataSetPassword, pBuffer, psSSETInfo->szDataSetPassword, uCurrentStorageOffset);
  901. uCurrentStorageOffset = StringToTapeAddress(&psSSET->sUserName, pBuffer, psSSETInfo->szUserName, uCurrentStorageOffset);
  902. psSSET->uPhysicalBlockAddress = psSSETInfo->uPhysicalBlockAddress;
  903. psSSET->sMediaWriteDate = psSSETInfo->sMediaWriteDate;
  904. psSSET->uSoftwareVerMjr = psSSETInfo->uSoftwareVerMjr;
  905. psSSET->uSoftwareVerMnr = psSSETInfo->uSoftwareVerMnr;
  906. psSSET->uTimeZone = psSSETInfo->uTimeZone;
  907. psSSET->uMTFMinorVer = psSSETInfo->uMTFMinorVer;
  908. psSSET->uTapeCatalogVersion = psSSETInfo->uTapeCatalogVersion;
  909. if (pnSizeUsed)
  910. *pnSizeUsed = uOffsetToFirstStream;
  911. }
  912. return MTF_ERROR_NONE;
  913. }
  914. /***********************************************************************************
  915. * MTF_ReadSSETDblk()
  916. * ** MTF API FUNCTION **
  917. ***********************************************************************************/
  918. DWORD CMTFApi::MTF_ReadSSETDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  919. MTF_DBLK_SSET_INFO *psSSETInfo,
  920. BYTE *pBuffer)
  921. {
  922. MTF_DBLK_SSET *psSSET = 0;
  923. ClearStrings();
  924. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  925. psSSET = (MTF_DBLK_SSET *) pBuffer;
  926. psSSETInfo->uSSETAttributes = psSSET->uSSETAttributes;
  927. psSSETInfo->uPasswordEncryptionAlgorithm = psSSET->uPasswordEncryptionAlgorithm;
  928. psSSETInfo->uDataEncryptionAlgorithm = psSSET->uDataEncryptionAlgorithm;
  929. psSSETInfo->uSoftwareVendorId = psSSET->uSoftwareVendorId;
  930. psSSETInfo->uDataSetNumber = psSSET->uDataSetNumber;
  931. psSSETInfo->uPhysicalBlockAddress = psSSET->uPhysicalBlockAddress;
  932. psSSETInfo->sMediaWriteDate = psSSET->sMediaWriteDate;
  933. psSSETInfo->uSoftwareVerMjr = psSSET->uSoftwareVerMjr;
  934. psSSETInfo->uSoftwareVerMnr = psSSET->uSoftwareVerMnr;
  935. psSSETInfo->uTimeZone = psSSET->uTimeZone;
  936. psSSETInfo->uMTFMinorVer = psSSET->uMTFMinorVer;
  937. psSSETInfo->uTapeCatalogVersion = psSSET->uTapeCatalogVersion;
  938. psSSETInfo->szDataSetName = MakeString((wchar_t UNALIGNED *) (pBuffer + psSSET->sDataSetName.uOffset), psSSET->sDataSetName.uSize);
  939. psSSETInfo->szDataSetDescription = MakeString((wchar_t UNALIGNED *) (pBuffer + psSSET->sDataSetDescription.uOffset), psSSET->sDataSetDescription.uSize);
  940. psSSETInfo->szDataSetPassword = MakeString((wchar_t UNALIGNED *) (pBuffer + psSSET->sDataSetPassword.uOffset), psSSET->sDataSetPassword.uSize);
  941. psSSETInfo->szUserName = MakeString((wchar_t UNALIGNED *) (pBuffer + psSSET->sUserName.uOffset), psSSET->sUserName.uSize);
  942. if ( !psSSETInfo->szDataSetName || !psSSETInfo->szDataSetDescription || !psSSETInfo->szDataSetPassword || !psSSETInfo->szUserName )
  943. return MTF_OUT_OF_MEMORY;
  944. return MTF_ERROR_NONE;
  945. }
  946. /* ==================================================================================
  947. =====================================================================================
  948. VOLB DBLK: MTF_DBLK_VOLB_INFO
  949. =====================================================================================
  950. ================================================================================== */
  951. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  952. // **NOT INCLUDING THE COMMON DBLK HEADER additional info **
  953. size_t CMTFApi::MTF_DBLK_VOLB_INFO_CalcAddDataSize(
  954. MTF_DBLK_VOLB_INFO *pVOLBInfo
  955. )
  956. {
  957. return wstrsize(pVOLBInfo->szDeviceName) +
  958. wstrsize(pVOLBInfo->szVolumeName) +
  959. wstrsize(pVOLBInfo->szMachineName);
  960. }
  961. /***********************************************************************************
  962. * MTF_SetVOLBDefaults()
  963. * ** MTF API FUNCTION **
  964. ***********************************************************************************/
  965. void CMTFApi::MTF_SetVOLBDefaults(MTF_DBLK_VOLB_INFO *pVOLBInfo)
  966. {
  967. time_t tTime;
  968. time(&tTime);
  969. pVOLBInfo->uVolumeAttributes = 0;
  970. pVOLBInfo->szDeviceName = 0 ;
  971. pVOLBInfo->szVolumeName = 0 ;
  972. pVOLBInfo->szMachineName = 0 ;
  973. pVOLBInfo->sMediaWriteDate = MTF_CreateDateTimeFromTM(gmtime(&tTime));;
  974. }
  975. /***********************************************************************************
  976. * MTF_SetVOLBForDevice()
  977. * ** MTF API FUNCTION **
  978. ***********************************************************************************/
  979. void CMTFApi::MTF_SetVOLBForDevice(MTF_DBLK_VOLB_INFO *pVOLBInfo, wchar_t *szDevice)
  980. {
  981. DWORD dwBufSize = MAX_COMPUTERNAME_LENGTH + 1;
  982. wchar_t tempDeviceName[MTF_STRLEN+4];
  983. wcscpy(m_szDeviceName, szDevice);
  984. MTF_SetVOLBDefaults(pVOLBInfo); // initialize
  985. // Determine the format and set the appropriate bit in the VOLB attributes.
  986. if (*(m_szDeviceName+1) == L':') {
  987. // drive letter w/colon format
  988. pVOLBInfo->uVolumeAttributes |= MTF_VOLB_DEV_DRIVE;
  989. }
  990. else if (0 == wcsncmp( m_szDeviceName, L"UNC", 3 )) {
  991. // UNC format
  992. pVOLBInfo->uVolumeAttributes |= MTF_VOLB_DEV_UNC;
  993. }
  994. else {
  995. // operating system specific format
  996. pVOLBInfo->uVolumeAttributes |= MTF_VOLB_DEV_OS_SPEC;
  997. }
  998. // need to prepend \\?\ for the GetVolumeInformation call
  999. wcscpy(tempDeviceName, L"\\\\?\\");
  1000. wcscat(tempDeviceName, m_szDeviceName);
  1001. GetVolumeInformationW(tempDeviceName, m_szVolumeName, MTF_STRLEN, 0, 0, 0, 0, 0);
  1002. GetComputerNameW(m_szMachineName, &dwBufSize);
  1003. pVOLBInfo->szDeviceName = m_szDeviceName;
  1004. pVOLBInfo->szVolumeName = m_szVolumeName;
  1005. pVOLBInfo->szMachineName = m_szMachineName;
  1006. }
  1007. /***********************************************************************************
  1008. * MTF_WriteVOLBDblk()
  1009. * ** MTF API FUNCTION **
  1010. ***********************************************************************************/
  1011. DWORD CMTFApi::MTF_WriteVOLBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1012. MTF_DBLK_VOLB_INFO *psVOLBInfo,
  1013. BYTE *pBuffer,
  1014. size_t nBufferSize,
  1015. size_t *pnSizeUsed)
  1016. {
  1017. UINT16 uOffsetToFirstStream;
  1018. //
  1019. // Figure the size of the entire DBLK & make sure we have room
  1020. //
  1021. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_VOLB) +
  1022. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo) +
  1023. MTF_DBLK_VOLB_INFO_CalcAddDataSize(psVOLBInfo));
  1024. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1025. if (nBufferSize < uOffsetToFirstStream)
  1026. {
  1027. if (pnSizeUsed)
  1028. *pnSizeUsed = uOffsetToFirstStream;
  1029. return MTF_ERROR_BUFFER_TOO_SMALL;
  1030. }
  1031. memset(pBuffer, 0, uOffsetToFirstStream);
  1032. {
  1033. MTF_DBLK_VOLB *psVOLB = 0;
  1034. size_t uCurrentStorageOffset = 0;
  1035. uCurrentStorageOffset = sizeof(MTF_DBLK_VOLB);
  1036. MTF_WriteDblkHdrToBuffer(
  1037. (UINT8 *)(char *)MTF_ID_VOLB,
  1038. uOffsetToFirstStream,
  1039. psHdrInfo,
  1040. pBuffer,
  1041. &uCurrentStorageOffset);
  1042. psVOLB = (MTF_DBLK_VOLB *) pBuffer;
  1043. psVOLB->uVolumeAttributes = psVOLBInfo->uVolumeAttributes;
  1044. uCurrentStorageOffset = StringToTapeAddress(&psVOLB->sDeviceName, pBuffer, psVOLBInfo->szDeviceName, uCurrentStorageOffset);
  1045. uCurrentStorageOffset = StringToTapeAddress(&psVOLB->sVolumeName, pBuffer, psVOLBInfo->szVolumeName, uCurrentStorageOffset);
  1046. uCurrentStorageOffset = StringToTapeAddress(&psVOLB->sMachineName, pBuffer, psVOLBInfo->szMachineName, uCurrentStorageOffset);
  1047. psVOLB->sMediaWriteDate = psVOLBInfo->sMediaWriteDate;
  1048. if (pnSizeUsed)
  1049. *pnSizeUsed = uOffsetToFirstStream;
  1050. }
  1051. return MTF_ERROR_NONE;
  1052. }
  1053. /***********************************************************************************
  1054. * MTF_ReadVOLBDblk()
  1055. * ** MTF API FUNCTION **
  1056. ***********************************************************************************/
  1057. DWORD CMTFApi::MTF_ReadVOLBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1058. MTF_DBLK_VOLB_INFO *psVOLBInfo,
  1059. BYTE *pBuffer)
  1060. {
  1061. MTF_DBLK_VOLB *psVOLB = 0;
  1062. ClearStrings();
  1063. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1064. psVOLB = (MTF_DBLK_VOLB *) pBuffer;
  1065. psVOLBInfo->uVolumeAttributes = psVOLB->uVolumeAttributes;
  1066. psVOLBInfo->sMediaWriteDate = psVOLB->sMediaWriteDate;
  1067. psVOLBInfo->szDeviceName = MakeString((wchar_t UNALIGNED *) (pBuffer + psVOLB->sDeviceName.uOffset), psVOLB->sDeviceName.uSize);
  1068. psVOLBInfo->szVolumeName = MakeString((wchar_t UNALIGNED *) (pBuffer + psVOLB->sVolumeName.uOffset), psVOLB->sVolumeName.uSize);
  1069. psVOLBInfo->szMachineName = MakeString((wchar_t UNALIGNED *) (pBuffer + psVOLB->sMachineName.uOffset), psVOLB->sMachineName.uSize);
  1070. if ( !psVOLBInfo->szDeviceName || !psVOLBInfo->szVolumeName || !psVOLBInfo->szMachineName )
  1071. return MTF_OUT_OF_MEMORY;
  1072. return MTF_ERROR_NONE;
  1073. }
  1074. /* ==================================================================================
  1075. =====================================================================================
  1076. DIRB DBLK: MTF_DBLK_DIRB_INFO
  1077. =====================================================================================
  1078. ================================================================================== */
  1079. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  1080. // **NOT INCLUDING THE COMMON DBLK HEADER additional info **
  1081. size_t CMTFApi::MTF_DBLK_DIRB_INFO_CalcAddDataSize(MTF_DBLK_DIRB_INFO *pDIRBInfo)
  1082. {
  1083. return wstrsize(pDIRBInfo->szDirectoryName);
  1084. }
  1085. /***********************************************************************************
  1086. * MTF_SetDIRBDefaults()
  1087. * ** MTF API FUNCTION **
  1088. ***********************************************************************************/
  1089. void CMTFApi::MTF_SetDIRBDefaults(
  1090. MTF_DBLK_DIRB_INFO *pDIRBInfo
  1091. )
  1092. {
  1093. pDIRBInfo->uDirectoryAttributes = 0;
  1094. pDIRBInfo->sLastModificationDate = MTF_CreateDateNull();
  1095. pDIRBInfo->sCreationDate = MTF_CreateDateNull();
  1096. pDIRBInfo->sBackupDate = MTF_CreateDateNull();
  1097. pDIRBInfo->sLastAccessDate = MTF_CreateDateNull();
  1098. pDIRBInfo->uDirectoryId = 0;
  1099. pDIRBInfo->szDirectoryName = 0 ;
  1100. }
  1101. /***********************************************************************************
  1102. * MTF_SetDIRBFromFindData()
  1103. * ** MTF API FUNCTION **
  1104. ***********************************************************************************/
  1105. void CMTFApi::MTF_SetDIRBFromFindData(
  1106. MTF_DBLK_DIRB_INFO *pDIRBInfo,
  1107. wchar_t *szDirectoryName,
  1108. WIN32_FIND_DATAW *pFindData
  1109. )
  1110. {
  1111. MTF_SetDIRBDefaults(pDIRBInfo); // initialize
  1112. if ( wcslen( szDirectoryName ) < MTF_STRLEN ) {
  1113. wcscpy(m_szDirectoryName, szDirectoryName);
  1114. pDIRBInfo->szDirectoryName = m_szDirectoryName;
  1115. }
  1116. else {
  1117. pDIRBInfo->uDirectoryAttributes |= MTF_DIRB_PATH_IN_STREAM;
  1118. pDIRBInfo->szDirectoryName = 0;
  1119. }
  1120. if (pFindData)
  1121. {
  1122. pDIRBInfo->uDirectoryAttributes |=
  1123. pFindData->dwFileAttributes & FILE_ATTRIBUTE_READONLY ? MTF_DIRB_READ_ONLY : 0
  1124. | pFindData->dwFileAttributes & FILE_ATTRIBUTE_HIDDEN ? MTF_DIRB_HIDDEN : 0
  1125. | pFindData->dwFileAttributes & FILE_ATTRIBUTE_SYSTEM ? MTF_DIRB_SYSTEM : 0
  1126. | pFindData->dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE ? MTF_DIRB_MODIFIED : 0;
  1127. pDIRBInfo->sLastModificationDate = MTF_CreateDateTimeFromFileTime(pFindData->ftLastWriteTime);
  1128. pDIRBInfo->sCreationDate = MTF_CreateDateTimeFromFileTime(pFindData->ftCreationTime);
  1129. pDIRBInfo->sLastAccessDate = MTF_CreateDateTimeFromFileTime(pFindData->ftLastAccessTime);
  1130. }
  1131. pDIRBInfo->uDirectoryId = 0;
  1132. }
  1133. /***********************************************************************************
  1134. * MTF_WriteDIRBDblk()
  1135. * ** MTF API FUNCTION **
  1136. ***********************************************************************************/
  1137. DWORD CMTFApi::MTF_WriteDIRBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1138. MTF_DBLK_DIRB_INFO *psDIRBInfo,
  1139. BYTE *pBuffer,
  1140. size_t nBufferSize,
  1141. size_t *pnSizeUsed)
  1142. {
  1143. UINT16 uOffsetToFirstStream;
  1144. //
  1145. // Figure the size of the entire DBLK & make sure we have room
  1146. //
  1147. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_DIRB) +
  1148. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo) +
  1149. MTF_DBLK_DIRB_INFO_CalcAddDataSize(psDIRBInfo));
  1150. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1151. if (nBufferSize < uOffsetToFirstStream)
  1152. {
  1153. if (pnSizeUsed)
  1154. *pnSizeUsed = uOffsetToFirstStream;
  1155. return MTF_ERROR_BUFFER_TOO_SMALL;
  1156. }
  1157. memset(pBuffer, 0, uOffsetToFirstStream);
  1158. {
  1159. MTF_DBLK_DIRB *psDIRB = 0;
  1160. size_t uCurrentStorageOffset = 0;
  1161. uCurrentStorageOffset = sizeof(MTF_DBLK_DIRB);
  1162. MTF_WriteDblkHdrToBuffer(
  1163. (UINT8 *)(char *)MTF_ID_DIRB,
  1164. uOffsetToFirstStream,
  1165. psHdrInfo,
  1166. pBuffer,
  1167. &uCurrentStorageOffset);
  1168. psDIRB = (MTF_DBLK_DIRB *) pBuffer;
  1169. psDIRB->uDirectoryAttributes = psDIRBInfo->uDirectoryAttributes;
  1170. psDIRB->sLastModificationDate = psDIRBInfo->sLastModificationDate;
  1171. psDIRB->sCreationDate = psDIRBInfo->sCreationDate;
  1172. psDIRB->sBackupDate = psDIRBInfo->sBackupDate;
  1173. psDIRB->sLastAccessDate = psDIRBInfo->sLastAccessDate;
  1174. psDIRB->uDirectoryId = psDIRBInfo->uDirectoryId;
  1175. //
  1176. // here, we need to turn the slashes (L'\\') to zeros (L'\0')in the directory name string...
  1177. //
  1178. {
  1179. int i, iLen;
  1180. wchar_t UNALIGNED *szDirectoryName = (wchar_t UNALIGNED *) (pBuffer + uCurrentStorageOffset);
  1181. uCurrentStorageOffset = StringToTapeAddress(&psDIRB->sDirectoryName, pBuffer, psDIRBInfo->szDirectoryName, uCurrentStorageOffset);
  1182. iLen = wstrsize(psDIRBInfo->szDirectoryName);
  1183. for (i = 0; i < iLen; ++i)
  1184. if (szDirectoryName[i] == L'\\')
  1185. szDirectoryName[i] = L'\0';
  1186. }
  1187. if (pnSizeUsed)
  1188. *pnSizeUsed = uOffsetToFirstStream;
  1189. }
  1190. return MTF_ERROR_NONE;
  1191. }
  1192. /***********************************************************************************
  1193. * MTF_ReadDIRBDblk()
  1194. * ** MTF API FUNCTION **
  1195. ***********************************************************************************/
  1196. DWORD CMTFApi::MTF_ReadDIRBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1197. MTF_DBLK_DIRB_INFO *psDIRBInfo,
  1198. BYTE *pBuffer)
  1199. {
  1200. MTF_DBLK_DIRB *psDIRB = 0;
  1201. ClearStrings();
  1202. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1203. psDIRB = (MTF_DBLK_DIRB *) pBuffer;
  1204. psDIRBInfo->uDirectoryAttributes = psDIRB->uDirectoryAttributes;
  1205. psDIRBInfo->sLastModificationDate = psDIRB->sLastModificationDate;
  1206. psDIRBInfo->sCreationDate = psDIRB->sCreationDate;
  1207. psDIRBInfo->sBackupDate = psDIRB->sBackupDate;
  1208. psDIRBInfo->sLastAccessDate = psDIRB->sLastAccessDate;
  1209. psDIRBInfo->uDirectoryId = psDIRB->uDirectoryId;
  1210. psDIRBInfo->szDirectoryName = NULL;
  1211. //
  1212. // we need to turn the zeros in the directory name back to slashes
  1213. // (there are no terminating \0's in the string -- all \0's are really \\'s
  1214. //
  1215. {
  1216. wchar_t *pTmpBuffer;
  1217. int i;
  1218. pTmpBuffer = (wchar_t *) malloc(psDIRB->sDirectoryName.uSize);
  1219. if (pTmpBuffer) {
  1220. memmove(pTmpBuffer, pBuffer + psDIRB->sDirectoryName.uOffset, psDIRB->sDirectoryName.uSize);
  1221. for (i = 0; i < psDIRB->sDirectoryName.uSize; ++i)
  1222. if (pTmpBuffer[i] == L'\0')
  1223. pTmpBuffer[i] = L'\\';
  1224. psDIRBInfo->szDirectoryName = MakeString(pTmpBuffer, psDIRB->sDirectoryName.uSize);
  1225. free(pTmpBuffer);
  1226. }
  1227. }
  1228. if ( !psDIRBInfo->szDirectoryName )
  1229. return MTF_OUT_OF_MEMORY;
  1230. return MTF_ERROR_NONE;
  1231. }
  1232. /* ==================================================================================
  1233. =====================================================================================
  1234. FILE DBLK: MTF_DBLK_FILE_INFO
  1235. =====================================================================================
  1236. ================================================================================== */
  1237. // Calculates the room that will be taken up in the DBLK by strings and OS specific data
  1238. // **NOT INCLUDING THE COMMON DBLK HEADER additional info **
  1239. size_t CMTFApi::MTF_DBLK_FILE_INFO_CalcAddDataSize(MTF_DBLK_FILE_INFO *pFILEInfo)
  1240. {
  1241. return wstrsize(pFILEInfo->szFileName);
  1242. }
  1243. /***********************************************************************************
  1244. * MTF_SetFILEDefaults()
  1245. * ** MTF API FUNCTION **
  1246. ***********************************************************************************/
  1247. void CMTFApi::MTF_SetFILEDefaults(MTF_DBLK_FILE_INFO *pFILEInfo)
  1248. {
  1249. pFILEInfo->uFileAttributes = 0;
  1250. pFILEInfo->sLastModificationDate = MTF_CreateDateNull();
  1251. pFILEInfo->sCreationDate = MTF_CreateDateNull();
  1252. pFILEInfo->sBackupDate = MTF_CreateDateNull();
  1253. pFILEInfo->sLastAccessDate = MTF_CreateDateNull();
  1254. pFILEInfo->uDirectoryId = 0;
  1255. pFILEInfo->uFileId = 0;
  1256. pFILEInfo->szFileName = 0;
  1257. }
  1258. /***********************************************************************************
  1259. * MTF_SetFILEFromFindData()
  1260. * ** MTF API FUNCTION **
  1261. ***********************************************************************************/
  1262. void CMTFApi::MTF_SetFILEFromFindData(MTF_DBLK_FILE_INFO *pFILEInfo, WIN32_FIND_DATAW *pFindData)
  1263. {
  1264. time_t tTime;
  1265. time(&tTime);
  1266. MTF_SetFILEDefaults(pFILEInfo); // initialize
  1267. wcscpy(m_szFileName, pFindData->cFileName);
  1268. pFILEInfo->uFileAttributes =
  1269. (pFindData->dwFileAttributes & FILE_ATTRIBUTE_READONLY ? MTF_FILE_READ_ONLY : 0)
  1270. | (pFindData->dwFileAttributes & FILE_ATTRIBUTE_HIDDEN ? MTF_FILE_HIDDEN : 0)
  1271. | (pFindData->dwFileAttributes & FILE_ATTRIBUTE_SYSTEM ? MTF_FILE_SYSTEM : 0)
  1272. | (pFindData->dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE ? MTF_FILE_MODIFIED : 0) ;
  1273. pFILEInfo->sLastModificationDate = MTF_CreateDateTimeFromFileTime(pFindData->ftLastWriteTime);
  1274. pFILEInfo->sCreationDate = MTF_CreateDateTimeFromFileTime(pFindData->ftCreationTime);
  1275. pFILEInfo->sLastAccessDate = MTF_CreateDateTimeFromFileTime(pFindData->ftLastAccessTime);
  1276. pFILEInfo->uDirectoryId = 0;
  1277. pFILEInfo->uFileId = 0;
  1278. pFILEInfo->szFileName = m_szFileName;
  1279. pFILEInfo->uDisplaySize = MTF_CreateUINT64(pFindData->nFileSizeLow, pFindData->nFileSizeHigh);
  1280. }
  1281. /***********************************************************************************
  1282. * MTF_WriteFILEDblk()
  1283. * ** MTF API FUNCTION **
  1284. ***********************************************************************************/
  1285. DWORD CMTFApi::MTF_WriteFILEDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1286. MTF_DBLK_FILE_INFO *psFILEInfo,
  1287. BYTE *pBuffer,
  1288. size_t nBufferSize,
  1289. size_t *pnSizeUsed)
  1290. {
  1291. UINT16 uOffsetToFirstStream;
  1292. //
  1293. // Figure the size of the entire DBLK & make sure we have room
  1294. //
  1295. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_FILE) +
  1296. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo) +
  1297. MTF_DBLK_FILE_INFO_CalcAddDataSize(psFILEInfo));
  1298. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1299. if (nBufferSize < uOffsetToFirstStream)
  1300. {
  1301. if (pnSizeUsed)
  1302. *pnSizeUsed = uOffsetToFirstStream;
  1303. return MTF_ERROR_BUFFER_TOO_SMALL;
  1304. }
  1305. memset(pBuffer, 0, uOffsetToFirstStream);
  1306. {
  1307. MTF_DBLK_FILE *psFILE = 0;
  1308. size_t uCurrentStorageOffset = 0;
  1309. uCurrentStorageOffset = sizeof(MTF_DBLK_FILE);
  1310. psHdrInfo->uDisplayableSize = psFILEInfo->uDisplaySize;
  1311. MTF_WriteDblkHdrToBuffer(
  1312. (UINT8 *)(char *)MTF_ID_FILE,
  1313. uOffsetToFirstStream,
  1314. psHdrInfo,
  1315. pBuffer,
  1316. &uCurrentStorageOffset);
  1317. psFILE = (MTF_DBLK_FILE *) pBuffer;
  1318. psFILE->uFileAttributes = psFILEInfo->uFileAttributes;
  1319. psFILE->sLastModificationDate = psFILEInfo->sLastModificationDate;
  1320. psFILE->sCreationDate = psFILEInfo->sCreationDate;
  1321. psFILE->sBackupDate = psFILEInfo->sBackupDate;
  1322. psFILE->sLastAccessDate = psFILEInfo->sLastAccessDate;
  1323. psFILE->uDirectoryId = psFILEInfo->uDirectoryId;
  1324. psFILE->uFileId = psFILEInfo->uFileId;
  1325. uCurrentStorageOffset = StringToTapeAddress(&psFILE->sFileName, pBuffer, psFILEInfo->szFileName, uCurrentStorageOffset);
  1326. if (pnSizeUsed)
  1327. *pnSizeUsed = uOffsetToFirstStream;
  1328. }
  1329. return MTF_ERROR_NONE;
  1330. }
  1331. /***********************************************************************************
  1332. * MTF_ReadFILEDblk()
  1333. * ** MTF API FUNCTION **
  1334. ***********************************************************************************/
  1335. DWORD CMTFApi::MTF_ReadFILEDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1336. MTF_DBLK_FILE_INFO *psFILEInfo,
  1337. BYTE *pBuffer)
  1338. {
  1339. MTF_DBLK_FILE *psFILE = 0;
  1340. ClearStrings();
  1341. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1342. psFILE = (MTF_DBLK_FILE *) pBuffer;
  1343. psFILEInfo->uFileAttributes = psFILE->uFileAttributes;
  1344. psFILEInfo->sLastModificationDate = psFILE->sLastModificationDate;
  1345. psFILEInfo->sCreationDate = psFILE->sCreationDate;
  1346. psFILEInfo->sBackupDate = psFILE->sBackupDate;
  1347. psFILEInfo->sLastAccessDate = psFILE->sLastAccessDate;
  1348. psFILEInfo->uDirectoryId = psFILE->uDirectoryId;
  1349. psFILEInfo->uFileId = psFILE->uFileId;
  1350. psFILEInfo->szFileName = MakeString((wchar_t UNALIGNED *) (pBuffer + psFILE->sFileName.uOffset), psFILE->sFileName.uSize);
  1351. if ( !psFILEInfo->szFileName )
  1352. return MTF_OUT_OF_MEMORY;
  1353. return MTF_ERROR_NONE;
  1354. }
  1355. /* ==================================================================================
  1356. =====================================================================================
  1357. CFIL DBLK: MTF_DBLK_CFIL_INFO
  1358. =====================================================================================
  1359. ================================================================================== */
  1360. /***********************************************************************************
  1361. * MTF_SetCFILDefaults()
  1362. * ** MTF API FUNCTION **
  1363. ***********************************************************************************/
  1364. void CMTFApi::MTF_SetCFILDefaults(
  1365. MTF_DBLK_CFIL_INFO *pCFILInfo
  1366. )
  1367. {
  1368. pCFILInfo->uCFileAttributes = MTF_CFIL_UNREADABLE_BLK;
  1369. pCFILInfo->uDirectoryId = 0;
  1370. pCFILInfo->uFileId = 0;
  1371. pCFILInfo->uStreamOffset = 0;
  1372. pCFILInfo->uCorruptStreamNumber = 0;
  1373. }
  1374. /***********************************************************************************
  1375. * MTF_WriteCFILDblk()
  1376. * ** MTF API FUNCTION **
  1377. ***********************************************************************************/
  1378. DWORD CMTFApi::MTF_WriteCFILDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1379. MTF_DBLK_CFIL_INFO *psCFILInfo,
  1380. BYTE *pBuffer,
  1381. size_t nBufferSize,
  1382. size_t *pnSizeUsed)
  1383. {
  1384. UINT16 uOffsetToFirstStream;
  1385. //
  1386. // Figure the size of the entire DBLK & make sure we have room
  1387. //
  1388. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_HDR) +
  1389. sizeof(MTF_DBLK_CFIL) +
  1390. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo));
  1391. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1392. if (nBufferSize < uOffsetToFirstStream)
  1393. {
  1394. if (pnSizeUsed)
  1395. *pnSizeUsed = uOffsetToFirstStream;
  1396. return MTF_ERROR_BUFFER_TOO_SMALL;
  1397. }
  1398. memset(pBuffer, 0, uOffsetToFirstStream);
  1399. {
  1400. MTF_DBLK_CFIL_INFO *psCFIL = 0;
  1401. size_t uCurrentStorageOffset = 0;
  1402. uCurrentStorageOffset = sizeof(MTF_DBLK_HDR) + sizeof(MTF_DBLK_CFIL);
  1403. MTF_WriteDblkHdrToBuffer(
  1404. (UINT8 *)(char *)MTF_ID_CFIL,
  1405. uOffsetToFirstStream,
  1406. psHdrInfo,
  1407. pBuffer,
  1408. &uCurrentStorageOffset);
  1409. psCFIL = (MTF_DBLK_CFIL_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1410. *psCFIL = *psCFILInfo;
  1411. if (pnSizeUsed)
  1412. *pnSizeUsed = uOffsetToFirstStream;
  1413. }
  1414. return MTF_ERROR_NONE;
  1415. }
  1416. /***********************************************************************************
  1417. * MTF_ReadCFILDblk()
  1418. * ** MTF API FUNCTION **
  1419. ***********************************************************************************/
  1420. DWORD CMTFApi::MTF_ReadCFILDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1421. MTF_DBLK_CFIL_INFO *psCFILInfo,
  1422. BYTE *pBuffer)
  1423. {
  1424. MTF_DBLK_CFIL *psCFIL = 0;
  1425. ClearStrings();
  1426. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1427. psCFIL = (MTF_DBLK_CFIL_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1428. *psCFILInfo = *psCFIL;
  1429. return MTF_ERROR_NONE;
  1430. }
  1431. /* ==================================================================================
  1432. =====================================================================================
  1433. ESPB DBLK
  1434. =====================================================================================
  1435. ================================================================================== */
  1436. /***********************************************************************************
  1437. * MTF_WriteESPBDblk()
  1438. * ** MTF API FUNCTION **
  1439. ***********************************************************************************/
  1440. DWORD CMTFApi::MTF_WriteESPBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1441. BYTE *pBuffer,
  1442. size_t nBufferSize,
  1443. size_t *pnSizeUsed)
  1444. {
  1445. UINT16 uOffsetToFirstStream;
  1446. //
  1447. // Figure the size of the entire DBLK & make sure we have room
  1448. //
  1449. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_HDR) +
  1450. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo));
  1451. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1452. if (nBufferSize < uOffsetToFirstStream) {
  1453. if (pnSizeUsed) {
  1454. *pnSizeUsed = uOffsetToFirstStream;
  1455. }
  1456. return MTF_ERROR_BUFFER_TOO_SMALL;
  1457. }
  1458. memset(pBuffer, 0, uOffsetToFirstStream);
  1459. MTF_WriteDblkHdrToBuffer(
  1460. (UINT8 *)(char *)MTF_ID_ESPB,
  1461. uOffsetToFirstStream,
  1462. psHdrInfo,
  1463. pBuffer,
  1464. 0);
  1465. if (pnSizeUsed)
  1466. *pnSizeUsed = uOffsetToFirstStream;
  1467. return MTF_ERROR_NONE;
  1468. }
  1469. /***********************************************************************************
  1470. * MTF_ReadESPBDblk()
  1471. * ** MTF API FUNCTION **
  1472. ***********************************************************************************/
  1473. DWORD CMTFApi::MTF_ReadESPBDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1474. BYTE *pBuffer)
  1475. {
  1476. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1477. return MTF_ERROR_NONE;
  1478. }
  1479. /* ==================================================================================
  1480. =====================================================================================
  1481. End of Set DBLK (ESET)
  1482. =====================================================================================
  1483. ================================================================================== */
  1484. /***********************************************************************************
  1485. * MTF_SetESETDefaults()
  1486. * ** MTF API FUNCTION **
  1487. ***********************************************************************************/
  1488. void CMTFApi::MTF_SetESETDefaults(MTF_DBLK_ESET_INFO *pESETInfo)
  1489. {
  1490. time_t tTime;
  1491. time(&tTime);
  1492. pESETInfo->uESETAttributes = 0;
  1493. pESETInfo->uNumberOfCorrupFiles = 0;
  1494. pESETInfo->uSetMapPBA = 0;
  1495. pESETInfo->uFileDetailPBA = 0;
  1496. pESETInfo->uFDDTapeSequenceNumber = 0;
  1497. pESETInfo->uDataSetNumber = 0;
  1498. pESETInfo->sMediaWriteDate = MTF_CreateDateTimeFromTM(gmtime(&tTime));
  1499. }
  1500. /***********************************************************************************
  1501. * MTF_WriteESETDblk()
  1502. * ** MTF API FUNCTION **
  1503. ***********************************************************************************/
  1504. DWORD CMTFApi::MTF_WriteESETDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1505. MTF_DBLK_ESET_INFO *psESETInfo,
  1506. BYTE *pBuffer,
  1507. size_t nBufferSize,
  1508. size_t *pnSizeUsed)
  1509. {
  1510. UINT16 uOffsetToFirstStream;
  1511. //
  1512. // Figure the size of the entire DBLK & make sure we have room
  1513. //
  1514. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_ESET) +
  1515. sizeof(MTF_DBLK_HDR) +
  1516. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo));
  1517. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1518. if (nBufferSize < uOffsetToFirstStream)
  1519. {
  1520. if (pnSizeUsed)
  1521. *pnSizeUsed = uOffsetToFirstStream;
  1522. return MTF_ERROR_BUFFER_TOO_SMALL;
  1523. }
  1524. memset(pBuffer, 0, uOffsetToFirstStream);
  1525. {
  1526. MTF_DBLK_ESET_INFO *psESET = 0;
  1527. size_t uCurrentStorageOffset = 0;
  1528. uCurrentStorageOffset = sizeof(MTF_DBLK_ESET) + sizeof(MTF_DBLK_HDR);
  1529. MTF_WriteDblkHdrToBuffer(
  1530. (UINT8 *)(char *)MTF_ID_ESET,
  1531. uOffsetToFirstStream,
  1532. psHdrInfo,
  1533. pBuffer,
  1534. &uCurrentStorageOffset);
  1535. psESET = (MTF_DBLK_ESET_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1536. *psESET = *psESETInfo;
  1537. if (pnSizeUsed)
  1538. *pnSizeUsed = uOffsetToFirstStream;
  1539. }
  1540. return MTF_ERROR_NONE;
  1541. }
  1542. /***********************************************************************************
  1543. * MTF_ReadESETDblk()
  1544. * ** MTF API FUNCTION **
  1545. ***********************************************************************************/
  1546. DWORD CMTFApi::MTF_ReadESETDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1547. MTF_DBLK_ESET_INFO *psESETInfo,
  1548. BYTE *pBuffer)
  1549. {
  1550. MTF_DBLK_ESET *psESET = 0;
  1551. ClearStrings();
  1552. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1553. psESET = (MTF_DBLK_ESET_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1554. *psESETInfo = *psESET;
  1555. return MTF_ERROR_NONE;
  1556. }
  1557. /* ==================================================================================
  1558. =====================================================================================
  1559. End of Set DBLK (EOTM)
  1560. =====================================================================================
  1561. ================================================================================== */
  1562. /***********************************************************************************
  1563. * MTF_SetEOTMDefaults()
  1564. * ** MTF API FUNCTION **
  1565. ***********************************************************************************/
  1566. void CMTFApi::MTF_SetEOTMDefaults(MTF_DBLK_EOTM_INFO *pEOTMInfo)
  1567. {
  1568. pEOTMInfo->uLastESETPBA = 0;
  1569. }
  1570. /***********************************************************************************
  1571. * MTF_WriteEOTMDblk()
  1572. * ** MTF API FUNCTION **
  1573. ***********************************************************************************/
  1574. DWORD CMTFApi::MTF_WriteEOTMDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1575. MTF_DBLK_EOTM_INFO *psEOTMInfo,
  1576. BYTE *pBuffer,
  1577. size_t nBufferSize,
  1578. size_t *pnSizeUsed)
  1579. {
  1580. UINT16 uOffsetToFirstStream;
  1581. //
  1582. // Figure the size of the entire DBLK & make sure we have room
  1583. //
  1584. uOffsetToFirstStream = (UINT16) (sizeof(MTF_DBLK_EOTM_INFO) +
  1585. sizeof(MTF_DBLK_HDR) +
  1586. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo));
  1587. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1588. if (nBufferSize < uOffsetToFirstStream)
  1589. {
  1590. if (pnSizeUsed)
  1591. *pnSizeUsed = uOffsetToFirstStream;
  1592. return MTF_ERROR_BUFFER_TOO_SMALL;
  1593. }
  1594. memset(pBuffer, 0, uOffsetToFirstStream);
  1595. {
  1596. MTF_DBLK_EOTM_INFO *psEOTM = 0;
  1597. size_t uCurrentStorageOffset = 0;
  1598. uCurrentStorageOffset = sizeof(MTF_DBLK_HDR) + sizeof(MTF_DBLK_EOTM);
  1599. MTF_WriteDblkHdrToBuffer(
  1600. (UINT8 *)(char *)MTF_ID_EOTM,
  1601. uOffsetToFirstStream,
  1602. psHdrInfo,
  1603. pBuffer,
  1604. &uCurrentStorageOffset);
  1605. psEOTM = (MTF_DBLK_EOTM_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1606. *psEOTM = *psEOTMInfo;
  1607. if (pnSizeUsed)
  1608. *pnSizeUsed = uOffsetToFirstStream;
  1609. }
  1610. return MTF_ERROR_NONE;
  1611. }
  1612. /***********************************************************************************
  1613. * MTF_ReadEOTMDblk()
  1614. * ** MTF API FUNCTION **
  1615. ***********************************************************************************/
  1616. DWORD CMTFApi::MTF_ReadEOTMDblk( MTF_DBLK_HDR_INFO *psHdrInfo,
  1617. MTF_DBLK_EOTM_INFO *psEOTMInfo,
  1618. BYTE *pBuffer)
  1619. {
  1620. MTF_DBLK_EOTM *psEOTM = 0;
  1621. ClearStrings();
  1622. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1623. psEOTM = (MTF_DBLK_EOTM_INFO *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1624. *psEOTMInfo = *psEOTM;
  1625. return MTF_ERROR_NONE;
  1626. }
  1627. /* ==================================================================================
  1628. =====================================================================================
  1629. Soft Filemark (SFMB)
  1630. =====================================================================================
  1631. ================================================================================== */
  1632. /***********************************************************************************
  1633. * MTF_CreateSFMB() - (bmd)
  1634. * ** MTF API FUNCTION **
  1635. ***********************************************************************************/
  1636. size_t CMTFApi::MTF_GetMaxSoftFilemarkEntries(size_t nBlockSize)
  1637. {
  1638. size_t n;
  1639. if (0 == nBlockSize || nBlockSize % 512) {
  1640. return 0;
  1641. }
  1642. // The SFMB fills the entire block.
  1643. // Calculate the total number of entries that fit within a block
  1644. // such that MTF_DBLK_HDR + MTF_DBLK_SFMB + (n-1 elements) < nBlockSize
  1645. n = (nBlockSize - sizeof(MTF_DBLK_HDR) - sizeof(MTF_DBLK_SFMB) + sizeof(UINT32))/sizeof(UINT32);
  1646. return n;
  1647. }
  1648. /***********************************************************************************
  1649. * MTF_InsertSoftFilemark() - (bmd)
  1650. * ** MTF API FUNCTION **
  1651. ***********************************************************************************/
  1652. void CMTFApi::MTF_InsertSoftFilemark(MTF_DBLK_SFMB_INFO *psSoftInfo,
  1653. UINT32 pba)
  1654. {
  1655. size_t n;
  1656. size_t bytesToShift;
  1657. // We insert a filemark entry by shifting all the entries down. The one closest BOM
  1658. // eventually drop out of the array.
  1659. if (psSoftInfo) {
  1660. n = psSoftInfo->uNumberOfFilemarkEntries;
  1661. bytesToShift = psSoftInfo->uFilemarkEntriesUsed * sizeof(UINT32);
  1662. // So we don't overwrite memory.
  1663. bytesToShift -= (psSoftInfo->uFilemarkEntriesUsed < psSoftInfo->uNumberOfFilemarkEntries) ? 0 : sizeof(UINT32);
  1664. memmove(&psSoftInfo->uFilemarkArray[1], &psSoftInfo->uFilemarkArray[0], bytesToShift);
  1665. psSoftInfo->uFilemarkArray[0] = pba;
  1666. if (psSoftInfo->uFilemarkEntriesUsed < psSoftInfo->uNumberOfFilemarkEntries) {
  1667. psSoftInfo->uFilemarkEntriesUsed++;
  1668. }
  1669. }
  1670. }
  1671. /***********************************************************************************
  1672. * MTF_WriteSFMBDblk() - (bmd)
  1673. * ** MTF API FUNCTION **
  1674. ***********************************************************************************/
  1675. DWORD CMTFApi::MTF_WriteSFMBDblk(MTF_DBLK_HDR_INFO *psHdrInfo,
  1676. MTF_DBLK_SFMB_INFO *psSoftInfo,
  1677. BYTE *pBuffer,
  1678. size_t nBufferSize,
  1679. size_t *pnSizeUsed)
  1680. {
  1681. UINT16 uOffsetToFirstStream;
  1682. size_t sizeOfSFMB;
  1683. if (NULL == psHdrInfo || NULL == psSoftInfo || NULL == pBuffer || NULL == pnSizeUsed || 0 == nBufferSize) {
  1684. return ERROR_INVALID_PARAMETER;
  1685. }
  1686. // Code assumes sizeof(MTF_DBLK_SFMB_INFO) == sizeof(MTF_DBLK_SFMB)
  1687. if (sizeof(MTF_DBLK_SFMB_INFO) != sizeof(MTF_DBLK_SFMB)) {
  1688. return ERROR_INVALID_FUNCTION;
  1689. }
  1690. //
  1691. // Figure the size of the entire DBLK & make sure we have room
  1692. //
  1693. sizeOfSFMB = sizeof(MTF_DBLK_SFMB) + (psSoftInfo->uNumberOfFilemarkEntries-1)*sizeof(UINT32);
  1694. uOffsetToFirstStream = (UINT16) (sizeOfSFMB +
  1695. sizeof(MTF_DBLK_HDR) +
  1696. MTF_DBLK_HDR_INFO_CalcAddDataSize(psHdrInfo));
  1697. uOffsetToFirstStream = (UINT16)Align(uOffsetToFirstStream, 4);
  1698. if (nBufferSize < uOffsetToFirstStream)
  1699. {
  1700. if (pnSizeUsed)
  1701. *pnSizeUsed = uOffsetToFirstStream;
  1702. return MTF_ERROR_BUFFER_TOO_SMALL;
  1703. }
  1704. memset(pBuffer, 0, uOffsetToFirstStream);
  1705. {
  1706. MTF_DBLK_SFMB_INFO *psSFMB = 0;
  1707. size_t uCurrentStorageOffset = 0;
  1708. uCurrentStorageOffset = sizeof(MTF_DBLK_HDR) + sizeOfSFMB;
  1709. MTF_WriteDblkHdrToBuffer(
  1710. (UINT8 *)(char *)MTF_ID_SFMB,
  1711. uOffsetToFirstStream,
  1712. psHdrInfo,
  1713. pBuffer,
  1714. &uCurrentStorageOffset);
  1715. psSFMB = (MTF_DBLK_SFMB *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1716. // Need a deep copy since MTF_DBLK_SFMB_INFO holds a placeholder for the array.
  1717. memcpy(psSFMB, psSoftInfo, sizeOfSFMB);
  1718. if (pnSizeUsed) {
  1719. *pnSizeUsed = uOffsetToFirstStream;
  1720. }
  1721. }
  1722. return MTF_ERROR_NONE;
  1723. }
  1724. /***********************************************************************************
  1725. * MTF_ReadSFMBDblk() - (bmd)
  1726. * ** MTF API FUNCTION **
  1727. ***********************************************************************************/
  1728. DWORD CMTFApi::MTF_ReadSFMBDblk(MTF_DBLK_HDR_INFO *psHdrInfo,
  1729. MTF_DBLK_SFMB_INFO *psSoftInfo,
  1730. BYTE *pBuffer)
  1731. {
  1732. MTF_DBLK_SFMB *psSFMB = 0;
  1733. size_t sizeOfSFMB;
  1734. if (NULL == psHdrInfo || NULL == psSoftInfo || NULL == pBuffer) {
  1735. return ERROR_INVALID_PARAMETER;
  1736. }
  1737. // Code assumes sizeof(MTF_DBLK_SFMB_INFO) == sizeof(MTF_DBLK_SFMB)
  1738. if (sizeof(MTF_DBLK_SFMB_INFO) != sizeof(MTF_DBLK_SFMB)) {
  1739. return ERROR_INVALID_FUNCTION;
  1740. }
  1741. ClearStrings();
  1742. MTF_DBLK_HDR_INFO_ReadFromBuffer(psHdrInfo, pBuffer);
  1743. psSFMB = (MTF_DBLK_SFMB *) (pBuffer + sizeof(MTF_DBLK_HDR));
  1744. // Need a deep copy since MTF_DBLK_SFMB_INFO holds a placeholder for the array.
  1745. sizeOfSFMB = sizeof(MTF_DBLK_SFMB) + (psSFMB->uNumberOfFilemarkEntries-1)*sizeof(UINT32);
  1746. memcpy(psSoftInfo, psSFMB, sizeOfSFMB);
  1747. return MTF_ERROR_NONE;
  1748. }
  1749. /* ==================================================================================
  1750. =====================================================================================
  1751. STREAM HEADER
  1752. =====================================================================================
  1753. ================================================================================== */
  1754. /***********************************************************************************
  1755. * MTF_SetSTREAMDefaults()
  1756. * ** MTF API FUNCTION **
  1757. ***********************************************************************************/
  1758. void CMTFApi::MTF_SetSTREAMDefaults(MTF_STREAM_INFO *pSTREAMInfo, char *szId)
  1759. {
  1760. memcpy(pSTREAMInfo->acStreamId, szId, 4);
  1761. pSTREAMInfo->uStreamFileSystemAttributes = 0;
  1762. pSTREAMInfo->uStreamTapeFormatAttributes = 0;
  1763. pSTREAMInfo->uStreamLength = 0;
  1764. pSTREAMInfo->uDataEncryptionAlgorithm = 0;
  1765. pSTREAMInfo->uDataCompressionAlgorithm = 0;
  1766. pSTREAMInfo->uCheckSum = 0;
  1767. }
  1768. /***********************************************************************************
  1769. * MTF_SetSTREAMFromStreamId()
  1770. * ** MTF API FUNCTION **
  1771. ***********************************************************************************/
  1772. void CMTFApi::MTF_SetSTREAMFromStreamId(
  1773. MTF_STREAM_INFO *pSTREAMInfo,
  1774. WIN32_STREAM_ID *pStreamId,
  1775. size_t nIDHeaderSize
  1776. )
  1777. {
  1778. // From Steve DeVos, Seagate:
  1779. // > BACKUP_INVALID and BACKUP_LINK will never be returned from BackupRead.
  1780. // >
  1781. // > -Steve
  1782. //
  1783. // TODO: MTF_NT_ENCRYPTED_STREAM "NTED"; These retrieved by NT Encyption APIs
  1784. // TODO: MTF_NT_QUOTA_STREAM "NTQU"; These retrieved by NT Quota APIs
  1785. MTF_SetSTREAMDefaults(pSTREAMInfo, "UNKN");
  1786. if (pStreamId->dwStreamId == BACKUP_DATA)
  1787. memcpy(pSTREAMInfo->acStreamId, "STAN", 4);
  1788. else if (pStreamId->dwStreamId == BACKUP_EA_DATA)
  1789. memcpy(pSTREAMInfo->acStreamId, "NTEA", 4);
  1790. else if (pStreamId->dwStreamId == BACKUP_SECURITY_DATA)
  1791. memcpy(pSTREAMInfo->acStreamId, "NACL", 4);
  1792. else if (pStreamId->dwStreamId == BACKUP_ALTERNATE_DATA)
  1793. memcpy(pSTREAMInfo->acStreamId, "ADAT", 4);
  1794. else if (pStreamId->dwStreamId == BACKUP_OBJECT_ID)
  1795. memcpy(pSTREAMInfo->acStreamId, "NTOI", 4);
  1796. else if (pStreamId->dwStreamId == BACKUP_REPARSE_DATA)
  1797. memcpy(pSTREAMInfo->acStreamId, "NTRP", 4);
  1798. else if (pStreamId->dwStreamId == BACKUP_SPARSE_BLOCK)
  1799. memcpy(pSTREAMInfo->acStreamId, "SPAR", 4);
  1800. else {
  1801. pSTREAMInfo->uStreamFileSystemAttributes |= MTF_STREAM_IS_NON_PORTABLE;
  1802. }
  1803. if (pStreamId->dwStreamAttributes & STREAM_MODIFIED_WHEN_READ)
  1804. pSTREAMInfo->uStreamFileSystemAttributes |= MTF_STREAM_MODIFIED_BY_READ;
  1805. if (pStreamId->dwStreamAttributes & STREAM_CONTAINS_SECURITY)
  1806. pSTREAMInfo->uStreamFileSystemAttributes |= MTF_STREAM_CONTAINS_SECURITY;
  1807. if (pStreamId->dwStreamAttributes & STREAM_SPARSE_ATTRIBUTE)
  1808. pSTREAMInfo->uStreamFileSystemAttributes |= MTF_STREAM_IS_SPARSE;
  1809. pSTREAMInfo->uStreamTapeFormatAttributes = 0;
  1810. pSTREAMInfo->uStreamLength = MTF_CreateUINT64(pStreamId->Size.LowPart, pStreamId->Size.HighPart) + nIDHeaderSize;
  1811. pSTREAMInfo->uDataEncryptionAlgorithm = 0;
  1812. pSTREAMInfo->uDataCompressionAlgorithm = 0;
  1813. pSTREAMInfo->uCheckSum = 0;
  1814. }
  1815. /***********************************************************************************
  1816. * MTF_SetStreamIdFromSTREAM() - (bmd)
  1817. * ** MTF API FUNCTION **
  1818. ***********************************************************************************/
  1819. void CMTFApi::MTF_SetStreamIdFromSTREAM(
  1820. WIN32_STREAM_ID *pStreamId,
  1821. MTF_STREAM_INFO *pSTREAMInfo,
  1822. size_t nIDHeaderSize
  1823. )
  1824. {
  1825. UNREFERENCED_PARAMETER(nIDHeaderSize);
  1826. memset( pStreamId, 0, sizeof( WIN32_STREAM_ID ) );
  1827. if (0 == memcmp(pSTREAMInfo->acStreamId, "STAN", 4))
  1828. pStreamId->dwStreamId = BACKUP_DATA;
  1829. else if (0 == memcmp(pSTREAMInfo->acStreamId, "NTEA", 4))
  1830. pStreamId->dwStreamId = BACKUP_EA_DATA;
  1831. else if (0 == memcmp(pSTREAMInfo->acStreamId, "NACL", 4))
  1832. pStreamId->dwStreamId = BACKUP_SECURITY_DATA;
  1833. else if (0 == memcmp(pSTREAMInfo->acStreamId, "ADAT", 4))
  1834. pStreamId->dwStreamId = BACKUP_ALTERNATE_DATA;
  1835. else if (0 == memcmp(pSTREAMInfo->acStreamId, "NTOI", 4))
  1836. pStreamId->dwStreamId = BACKUP_OBJECT_ID;
  1837. else if (0 == memcmp(pSTREAMInfo->acStreamId, "NTRP", 4))
  1838. pStreamId->dwStreamId = BACKUP_REPARSE_DATA;
  1839. else if (0 == memcmp(pSTREAMInfo->acStreamId, "SPAR", 4))
  1840. pStreamId->dwStreamId = BACKUP_SPARSE_BLOCK;
  1841. else {
  1842. pStreamId->dwStreamId = BACKUP_INVALID;
  1843. }
  1844. pStreamId->dwStreamAttributes = STREAM_NORMAL_ATTRIBUTE;
  1845. if (pSTREAMInfo->uStreamFileSystemAttributes & MTF_STREAM_MODIFIED_BY_READ)
  1846. pStreamId->dwStreamAttributes |= STREAM_MODIFIED_WHEN_READ;
  1847. if (pSTREAMInfo->uStreamFileSystemAttributes & MTF_STREAM_CONTAINS_SECURITY)
  1848. pStreamId->dwStreamAttributes |= STREAM_CONTAINS_SECURITY;
  1849. if (pSTREAMInfo->uStreamFileSystemAttributes & MTF_STREAM_IS_SPARSE)
  1850. pStreamId->dwStreamAttributes |= STREAM_SPARSE_ATTRIBUTE;
  1851. // TODO: Handle named data streams (size of name and in MTF stream)
  1852. // ? How do I know ?
  1853. pStreamId->Size.LowPart = (DWORD)((pSTREAMInfo->uStreamLength << 32) >>32);
  1854. pStreamId->Size.HighPart = (DWORD)(pSTREAMInfo->uStreamLength >> 32);
  1855. }
  1856. /***********************************************************************************
  1857. * MTF_WriteStreamHeader()
  1858. * ** MTF API FUNCTION **
  1859. ***********************************************************************************/
  1860. DWORD CMTFApi::MTF_WriteStreamHeader(MTF_STREAM_INFO *psStreamInfo,
  1861. BYTE *pBuffer,
  1862. size_t nBufferSize,
  1863. size_t *pnSizeUsed)
  1864. {
  1865. psStreamInfo->uCheckSum = CalcChecksum((BYTE *) psStreamInfo, sizeof(MTF_STREAM_INFO) / sizeof(UINT16) - 1);
  1866. if (nBufferSize < sizeof(MTF_STREAM_INFO))
  1867. {
  1868. if (pnSizeUsed)
  1869. *pnSizeUsed = sizeof(MTF_STREAM_INFO);
  1870. return MTF_ERROR_BUFFER_TOO_SMALL;
  1871. }
  1872. memset(pBuffer, 0, sizeof(MTF_STREAM_INFO));
  1873. *((MTF_STREAM_INFO *) pBuffer) = *psStreamInfo;
  1874. if (pnSizeUsed)
  1875. *pnSizeUsed = sizeof(MTF_STREAM_INFO);
  1876. return MTF_ERROR_NONE;
  1877. }
  1878. /***********************************************************************************
  1879. * MTF_WriteNameStream() - (bmd)
  1880. * ** MTF API FUNCTION **
  1881. ***********************************************************************************/
  1882. DWORD CMTFApi::MTF_WriteNameStream(
  1883. char *szType,
  1884. wchar_t *szName,
  1885. BYTE *pBuffer,
  1886. size_t nBufferSize,
  1887. size_t *pnSizeUsed)
  1888. {
  1889. MTF_STREAM_INFO sStream;
  1890. UINT16 uOffsetToCSUMStream;
  1891. UINT16 uOffsetToNextStream;
  1892. size_t nBufUsed;
  1893. UINT16 nameSize;
  1894. UINT32 nameChecksum;
  1895. //
  1896. // Figure the size of the entire Name stream including trailing CSUM and make sure we have room.
  1897. //
  1898. nameSize = (UINT16)wstrsize(szName); // including terminating '\0'
  1899. uOffsetToCSUMStream = sizeof(MTF_STREAM_INFO) + nameSize;
  1900. uOffsetToCSUMStream = (UINT16)Align(uOffsetToCSUMStream, 4);
  1901. uOffsetToNextStream = uOffsetToCSUMStream;
  1902. uOffsetToNextStream += sizeof(MTF_STREAM_INFO) + 4; // includes 4 byte CSUM data
  1903. uOffsetToNextStream = (UINT16)Align(uOffsetToNextStream, 4);
  1904. if (nBufferSize < uOffsetToNextStream) {
  1905. return MTF_ERROR_BUFFER_TOO_SMALL;
  1906. }
  1907. memset(pBuffer, 0, uOffsetToNextStream);
  1908. MTF_SetSTREAMDefaults(&sStream, szType);
  1909. sStream.uStreamLength = nameSize;
  1910. sStream.uStreamTapeFormatAttributes |= MTF_STREAM_CHECKSUMED;
  1911. MTF_WriteStreamHeader(&sStream, pBuffer, nBufferSize, &nBufUsed);
  1912. memcpy(pBuffer + nBufUsed, szName, nameSize);
  1913. if ( 0 == memcmp(sStream.acStreamId, "PNAM", 4) ) {
  1914. //
  1915. // here, we need to turn the slashes (L'\\') to zeros (L'\0')in the directory name string...
  1916. //
  1917. int i, iLen;
  1918. wchar_t UNALIGNED *szDirectoryName = (wchar_t UNALIGNED *) (pBuffer + nBufUsed);
  1919. iLen = ua_wstrsize(szDirectoryName);
  1920. for (i = 0; i < iLen; ++i)
  1921. if (szDirectoryName[i] == L'\\')
  1922. szDirectoryName[i] = L'\0';
  1923. }
  1924. // For Name streams, we always tack on a CSUM
  1925. nameChecksum = CalcChecksumOfStreamData(pBuffer + nBufUsed, nameSize / sizeof(UINT32) + 1);
  1926. MTF_SetSTREAMDefaults(&sStream, MTF_CHECKSUM_STREAM);
  1927. sStream.uStreamLength = sizeof(nameChecksum);
  1928. MTF_WriteStreamHeader(&sStream, pBuffer + uOffsetToCSUMStream, nBufferSize, &nBufUsed);
  1929. memcpy(pBuffer + uOffsetToCSUMStream + nBufUsed, &nameChecksum, sizeof(nameChecksum));
  1930. if (pnSizeUsed)
  1931. *pnSizeUsed = uOffsetToNextStream;
  1932. return MTF_ERROR_NONE;
  1933. }
  1934. /***********************************************************************************
  1935. * MTF_ReadStreamHeader()
  1936. * ** MTF API FUNCTION **
  1937. ***********************************************************************************/
  1938. DWORD CMTFApi::MTF_ReadStreamHeader(MTF_STREAM_INFO *psStreamInfo,
  1939. BYTE *pBuffer)
  1940. {
  1941. *psStreamInfo = *((MTF_STREAM_INFO *) pBuffer);
  1942. return MTF_ERROR_NONE;
  1943. }