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.

720 lines
15 KiB

  1. /*++
  2. Copyright (c) 1996 Microsoft Corporation
  3. Module Name:
  4. regmigrt.c
  5. Abstract:
  6. Registry migration routines
  7. Author:
  8. Ted Miller (tedm) 12-Apr-1996
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. typedef enum {
  14. AddQuotesNone,
  15. AddQuotesNormal,
  16. AddQuotesOpenNoClose,
  17. AddQuotesNoOpenOrClose,
  18. } AddQuotesOp;
  19. BOOL
  20. RetrieveMessageIntoBufferV(
  21. IN UINT MessageId,
  22. OUT PTSTR Buffer,
  23. IN UINT BufferSizeChars,
  24. IN va_list *arglist
  25. )
  26. {
  27. DWORD d;
  28. d = FormatMessage(
  29. FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_FROM_SYSTEM,
  30. hInst,
  31. MessageId,
  32. 0,
  33. Buffer,
  34. BufferSizeChars,
  35. arglist
  36. );
  37. if(!d) {
  38. d = FormatMessage(
  39. FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_ARGUMENT_ARRAY,
  40. hInst,
  41. MSG_NOT_FOUND,
  42. 0,
  43. Buffer,
  44. BufferSizeChars,
  45. (va_list *)&MessageId
  46. );
  47. if(!d) {
  48. return FALSE;
  49. }
  50. }
  51. return TRUE;
  52. }
  53. DWORD
  54. WriteText(
  55. IN HANDLE FileHandle,
  56. IN UINT MessageId,
  57. ...
  58. )
  59. {
  60. TCHAR Message[2048];
  61. CHAR message[4096];
  62. va_list arglist;
  63. DWORD Written;
  64. BOOL b;
  65. va_start(arglist,MessageId);
  66. b = RetrieveMessageIntoBufferV(
  67. MessageId,
  68. Message,
  69. sizeof(Message)/sizeof(Message[0]),
  70. &arglist
  71. );
  72. va_end(arglist);
  73. if(!b)
  74. return ERROR_INVALID_PARAMETER;
  75. #ifdef UNICODE
  76. Written = (DWORD)WideCharToMultiByte(
  77. CP_ACP,
  78. 0,
  79. Message,
  80. lstrlen(Message),
  81. message,
  82. sizeof(message),
  83. NULL,
  84. NULL
  85. );
  86. #else
  87. lstrcpyn(message,Message,sizeof(message));
  88. Written = lstrlen(message);
  89. #endif
  90. b = WriteFile(FileHandle,message,Written,&Written,NULL);
  91. return(b ? NO_ERROR : GetLastError());
  92. }
  93. DWORD
  94. FlushGenInfLineBuf(
  95. IN OUT PINFFILEGEN Context,
  96. IN HANDLE File
  97. )
  98. {
  99. CHAR TransBuf[INFLINEBUFLEN*2];
  100. DWORD rc;
  101. PVOID Buffer;
  102. DWORD Size;
  103. BOOL b;
  104. Buffer = TransBuf;
  105. #ifdef UNICODE
  106. Size = WideCharToMultiByte(
  107. CP_ACP,
  108. 0,
  109. Context->LineBuf,
  110. Context->LineBufUsed,
  111. TransBuf,
  112. sizeof(TransBuf),
  113. NULL,
  114. NULL
  115. );
  116. #else
  117. lstrcpyn(TransBuf,Context->LineBuf,sizeof(TransBuf));
  118. Size = Context->LineBufUsed;
  119. #endif
  120. if(WriteFile(File,Buffer,Size,&rc,NULL)) {
  121. rc = NO_ERROR;
  122. Context->LineBufUsed = 0;
  123. } else {
  124. rc = GetLastError();
  125. }
  126. return(rc);
  127. }
  128. DWORD
  129. __inline
  130. GenInfWriteChar(
  131. IN OUT PINFFILEGEN Context,
  132. IN HANDLE File,
  133. IN TCHAR Char
  134. )
  135. {
  136. DWORD rc;
  137. PVOID Buffer;
  138. Context->LineBuf[Context->LineBufUsed++] = Char;
  139. rc = (Context->LineBufUsed == INFLINEBUFLEN)
  140. ? FlushGenInfLineBuf(Context,File)
  141. : NO_ERROR;
  142. return(rc);
  143. }
  144. DWORD
  145. GenInfWriteString(
  146. IN OUT PINFFILEGEN Context,
  147. IN HANDLE File,
  148. IN LPCTSTR String,
  149. IN AddQuotesOp AddQuotes
  150. )
  151. {
  152. DWORD rc;
  153. TCHAR CONST *p;
  154. if((AddQuotes == AddQuotesNormal) || (AddQuotes == AddQuotesOpenNoClose)) {
  155. rc = GenInfWriteChar(Context,File,TEXT('\"'));
  156. if(rc != NO_ERROR) {
  157. return(rc);
  158. }
  159. }
  160. for(p=String; *p; p++) {
  161. rc = GenInfWriteChar(Context,File,*p);
  162. if(rc != NO_ERROR) {
  163. return(rc);
  164. }
  165. if((*p == TEXT('\"')) && (AddQuotes != AddQuotesNone)) {
  166. rc = GenInfWriteChar(Context,File,TEXT('\"'));
  167. if(rc != NO_ERROR) {
  168. return(rc);
  169. }
  170. }
  171. }
  172. if(AddQuotes == AddQuotesNormal) {
  173. rc = GenInfWriteChar(Context,File,TEXT('\"'));
  174. if(rc != NO_ERROR) {
  175. return(rc);
  176. }
  177. }
  178. return(NO_ERROR);
  179. }
  180. DWORD
  181. CreateAndOpenTempFile(
  182. IN LPCTSTR Path,
  183. IN LPCTSTR HeaderLine, OPTIONAL
  184. OUT HANDLE *Handle,
  185. OUT PTSTR Filename
  186. )
  187. {
  188. HANDLE h;
  189. DWORD rc;
  190. //
  191. // Note that this creates the file.
  192. //
  193. if(!GetTempFileName(Path,TEXT("$IF"),0,Filename)) {
  194. rc = GetLastError();
  195. goto c0;
  196. }
  197. h = CreateFile(
  198. Filename,
  199. GENERIC_WRITE,
  200. FILE_SHARE_READ,
  201. NULL,
  202. CREATE_ALWAYS,
  203. FILE_ATTRIBUTE_NORMAL,
  204. NULL
  205. );
  206. if(h == INVALID_HANDLE_VALUE) {
  207. rc = GetLastError();
  208. goto c1;
  209. }
  210. if(HeaderLine) {
  211. rc = WriteText(h,MSG_INF_SINGLELINE,HeaderLine);
  212. if(rc != NO_ERROR) {
  213. goto c2;
  214. }
  215. }
  216. *Handle = h;
  217. return(NO_ERROR);
  218. c2:
  219. CloseHandle(h);
  220. c1:
  221. DeleteFile(Filename);
  222. c0:
  223. return(rc);
  224. }
  225. DWORD
  226. InfCreateSection(
  227. IN LPCTSTR SectionName,
  228. IN OUT PINFFILEGEN *Context
  229. )
  230. {
  231. PTSTR Buffer;
  232. DWORD rc;
  233. Buffer = MALLOC( (lstrlen(SectionName) + 3)*sizeof(TCHAR) );
  234. if( !Buffer ) {
  235. return ERROR_NOT_ENOUGH_MEMORY;
  236. }
  237. lstrcpy( Buffer, TEXT("[") );
  238. lstrcat( Buffer, SectionName );
  239. lstrcat( Buffer, TEXT("]") );
  240. rc = WriteText((*Context)->FileHandle,MSG_INF_SINGLELINE,Buffer);
  241. FREE( Buffer );
  242. return(rc);
  243. }
  244. DWORD
  245. InfStart(
  246. IN LPCTSTR InfName,
  247. IN LPCTSTR Directory,
  248. OUT PINFFILEGEN *Context
  249. )
  250. {
  251. TCHAR InfFileName[MAX_PATH];
  252. DWORD d;
  253. DWORD rc;
  254. PINFFILEGEN context;
  255. UCHAR UnicodeMark[2];
  256. PTSTR p;
  257. DWORD BytesWritten = 0;
  258. //
  259. // Allocate some context.
  260. //
  261. context = MALLOC(sizeof(INFFILEGEN));
  262. if(!context) {
  263. rc = ERROR_NOT_ENOUGH_MEMORY;
  264. goto c0;
  265. }
  266. ZeroMemory(context,sizeof(INFFILEGEN));
  267. //
  268. // We'll create a unique inf name in the given directory
  269. // to use as the output inf. The directory itself will
  270. // become the oem file root.
  271. //
  272. if(!GetFullPathName(Directory,MAX_PATH,context->FileName,&p)) {
  273. rc = GetLastError();
  274. goto c1;
  275. }
  276. ConcatenatePaths(context->FileName,InfName,MAX_PATH);
  277. SetFileAttributes(context->FileName, FILE_ATTRIBUTE_NORMAL);
  278. DeleteFile(context->FileName);
  279. //
  280. // Create the output file.
  281. //
  282. context->FileHandle = CreateFile(
  283. context->FileName,
  284. GENERIC_WRITE,
  285. FILE_SHARE_READ,
  286. NULL,
  287. CREATE_ALWAYS,
  288. FILE_ATTRIBUTE_NORMAL,
  289. NULL
  290. );
  291. if(context->FileHandle == INVALID_HANDLE_VALUE) {
  292. rc = GetLastError();
  293. goto c1;
  294. }
  295. //
  296. // Write out header for inf file.
  297. //
  298. WriteFile(context->FileHandle,
  299. INF_FILE_HEADER,
  300. strlen(INF_FILE_HEADER),
  301. &BytesWritten,
  302. NULL);
  303. rc = GetLastError();
  304. if(rc != NO_ERROR) {
  305. goto c5;
  306. }
  307. *Context = context;
  308. return(NO_ERROR);
  309. c5:
  310. CloseHandle(context->FileHandle);
  311. DeleteFile(context->FileName);
  312. c1:
  313. FREE(context);
  314. c0:
  315. return(rc);
  316. }
  317. DWORD
  318. InfEnd(
  319. IN OUT PINFFILEGEN *Context
  320. )
  321. {
  322. PINFFILEGEN context;
  323. DWORD rc;
  324. HANDLE h;
  325. DWORD Size;
  326. context = *Context;
  327. *Context = NULL;
  328. h = context->FileHandle;
  329. rc = NO_ERROR;
  330. CloseHandle(h);
  331. if(rc != NO_ERROR) {
  332. DeleteFile(context->FileName);
  333. }
  334. FREE(context);
  335. return(rc);
  336. }
  337. DWORD
  338. pInfRegLineCommon(
  339. IN OUT PINFFILEGEN Context,
  340. IN HANDLE OutputFile,
  341. IN HKEY Key,
  342. IN LPCTSTR Subkey,
  343. IN LPCTSTR Value OPTIONAL
  344. )
  345. {
  346. LPCTSTR RootSpec;
  347. LPCTSTR SubkeySpec;
  348. DWORD rc;
  349. if(Subkey[0] == TEXT('\\')) {
  350. Subkey++;
  351. }
  352. //
  353. // Figure out the root key spec.
  354. //
  355. switch((ULONG_PTR)Key) {
  356. case (ULONG_PTR)HKEY_LOCAL_MACHINE:
  357. //
  358. // Check for HKEY_CLASSES_ROOT
  359. //
  360. if(_tcsnicmp(Subkey,TEXT("SOFTWARE\\Classes"),16)) {
  361. RootSpec = TEXT("HKLM");
  362. SubkeySpec = Subkey;
  363. } else {
  364. RootSpec = TEXT("HKCR");
  365. SubkeySpec = Subkey+16;
  366. if(*SubkeySpec == TEXT('\\')) {
  367. SubkeySpec++;
  368. }
  369. }
  370. break;
  371. case (ULONG_PTR)HKEY_CURRENT_USER:
  372. RootSpec = TEXT("HKCU");
  373. SubkeySpec = Subkey;
  374. break;
  375. case (ULONG_PTR)HKEY_CLASSES_ROOT:
  376. RootSpec = TEXT("HKCR");
  377. SubkeySpec = Subkey;
  378. break;
  379. default:
  380. //
  381. // Value we can't express via inf.
  382. // Use HKEY_ROOT but also write out a comment incidating
  383. // that there's a problem
  384. //
  385. RootSpec = TEXT("HKR");
  386. SubkeySpec = Subkey;
  387. Context->SawBogusOp = TRUE;
  388. rc = FlushGenInfLineBuf(Context,OutputFile);
  389. if(rc != NO_ERROR) {
  390. return(rc);
  391. }
  392. rc = WriteText(OutputFile,MSG_INF_BAD_REGSPEC_1);
  393. if(rc != NO_ERROR) {
  394. return(rc);
  395. }
  396. break;
  397. }
  398. rc = GenInfWriteString(Context,OutputFile,RootSpec,AddQuotesNone);
  399. if(rc == NO_ERROR) {
  400. rc = GenInfWriteChar(Context,OutputFile,TEXT(','));
  401. if(rc == NO_ERROR) {
  402. rc = GenInfWriteString(Context,OutputFile,SubkeySpec,AddQuotesNormal);
  403. if((rc == NO_ERROR) && Value) {
  404. rc = GenInfWriteChar(Context,OutputFile,TEXT(','));
  405. if(rc == NO_ERROR) {
  406. rc = GenInfWriteString(Context,OutputFile,Value,AddQuotesNormal);
  407. }
  408. }
  409. }
  410. }
  411. return(rc);
  412. }
  413. DWORD
  414. InfRecordAddReg(
  415. IN OUT PINFFILEGEN Context,
  416. IN HKEY Key,
  417. IN LPCTSTR Subkey,
  418. IN LPCTSTR Value, OPTIONAL
  419. IN DWORD DataType,
  420. IN PVOID Data,
  421. IN DWORD DataLength,
  422. IN BOOL SetNoClobberFlag
  423. )
  424. {
  425. DWORD rc;
  426. DWORD Flags;
  427. PTSTR p;
  428. DWORD d;
  429. int LineLen;
  430. TCHAR NumStr[24];
  431. //
  432. // Figure out flags based on data type.
  433. // The flags dword is built as two halves depending on whether
  434. // data is string or binary in nature.
  435. //
  436. // We do this before we write out the actual line
  437. // since that routine might also write a warning if a bogus root key
  438. // is specified.
  439. //
  440. switch(DataType) {
  441. case REG_SZ:
  442. Flags = FLG_ADDREG_TYPE_SZ;
  443. break;
  444. case REG_EXPAND_SZ:
  445. Flags = FLG_ADDREG_TYPE_EXPAND_SZ;
  446. break;
  447. case REG_MULTI_SZ:
  448. Flags = FLG_ADDREG_TYPE_MULTI_SZ;
  449. break;
  450. case REG_DWORD:
  451. Flags = FLG_ADDREG_TYPE_DWORD;
  452. break;
  453. //case REG_NONE:
  454. // Flags = FLG_ADDREG_TYPE_NONE;
  455. // break;
  456. case REG_NONE:
  457. Flags = FLG_ADDREG_KEYONLY;
  458. break;
  459. default:
  460. //
  461. // Arbitrary binary data. Better hope the data type doesn't overflow
  462. // 16 bits.
  463. //
  464. if(DataType > 0xffff) {
  465. Context->SawBogusOp = TRUE;
  466. rc = FlushGenInfLineBuf(Context,Context->FileHandle);
  467. if(rc != NO_ERROR) {
  468. return(rc);
  469. }
  470. rc = WriteText(Context->FileHandle,MSG_INF_BAD_REGSPEC_2);
  471. if(rc != NO_ERROR) {
  472. return(rc);
  473. }
  474. DataType = REG_BINARY;
  475. }
  476. Flags = FLG_ADDREG_BINVALUETYPE | (DataType << 16);
  477. break;
  478. }
  479. rc = pInfRegLineCommon(Context,Context->FileHandle,Key,Subkey,Value);
  480. if(rc != NO_ERROR) {
  481. return(rc);
  482. }
  483. if(Flags == FLG_ADDREG_KEYONLY) {
  484. rc = GenInfWriteChar(Context,Context->FileHandle,TEXT(','));
  485. if(rc != NO_ERROR) {
  486. return(rc);
  487. }
  488. }
  489. //
  490. // _stprintf(NumStr,TEXT(",%0#10lx"),Flags | 0x00000002); // Force NO_CLOBBER
  491. //
  492. // _stprintf(NumStr,TEXT(",%0#10lx"),Flags);
  493. wsprintf(NumStr,
  494. TEXT(",%#08lx"),
  495. SetNoClobberFlag? (Flags | 0x00000002) : Flags);
  496. rc = GenInfWriteString(Context,Context->FileHandle,NumStr,AddQuotesNone);
  497. if(rc != NO_ERROR) {
  498. return(rc);
  499. }
  500. //
  501. // Now we need to write out the data itself.
  502. // How we do this is dependent on the data type.
  503. //
  504. switch(DataType) {
  505. case REG_SZ:
  506. case REG_EXPAND_SZ:
  507. //
  508. // Single string. Ignore data length.
  509. //
  510. rc = GenInfWriteChar(Context,Context->FileHandle,TEXT(','));
  511. if(rc == NO_ERROR) {
  512. rc = GenInfWriteString(Context,Context->FileHandle,Data,AddQuotesNormal);
  513. }
  514. break;
  515. case REG_DWORD:
  516. //
  517. // Write out as a dword.
  518. //
  519. wsprintf(NumStr,TEXT(",%u"),*(DWORD UNALIGNED *)Data);
  520. rc = GenInfWriteString(Context,Context->FileHandle,NumStr,AddQuotesNone);
  521. break;
  522. case REG_MULTI_SZ:
  523. //
  524. // Write out each string.
  525. //
  526. for(p=Data; (rc==NO_ERROR) && *p; p+=lstrlen(p)+1) {
  527. rc = GenInfWriteChar(Context,Context->FileHandle,TEXT(','));
  528. if(rc == NO_ERROR) {
  529. rc = GenInfWriteString(Context,Context->FileHandle,p,AddQuotesNormal);
  530. }
  531. }
  532. break;
  533. case REG_NONE:
  534. //
  535. // Don't create a value entry
  536. //
  537. break;
  538. default:
  539. //
  540. // Treat as binary. If we have any data at all start a new line.
  541. //
  542. if(DataLength) {
  543. rc = GenInfWriteString(Context,Context->FileHandle,TEXT(",\\\r\n "),AddQuotesNone);
  544. }
  545. LineLen = 0;
  546. for(d=0; (rc==NO_ERROR) && (d<DataLength); d++) {
  547. if(LineLen == 25) {
  548. rc = GenInfWriteString(Context,Context->FileHandle,TEXT(",\\\r\n "),AddQuotesNone);
  549. LineLen = 0;
  550. }
  551. if(rc == NO_ERROR) {
  552. if(LineLen) {
  553. rc = GenInfWriteChar(Context,Context->FileHandle,TEXT(','));
  554. }
  555. if(rc == NO_ERROR) {
  556. wsprintf(NumStr,TEXT("%02x"),((PBYTE)Data)[d]);
  557. rc = GenInfWriteString(Context,Context->FileHandle,NumStr,AddQuotesNone);
  558. LineLen++;
  559. }
  560. }
  561. }
  562. break;
  563. }
  564. if(rc == NO_ERROR) {
  565. rc = GenInfWriteString(Context,Context->FileHandle,TEXT("\r\n"),AddQuotesNone);
  566. if(rc == NO_ERROR) {
  567. rc = FlushGenInfLineBuf(Context,Context->FileHandle);
  568. }
  569. }
  570. return(rc);
  571. }
  572. #if 0
  573. DWORD
  574. InfRecordDelReg(
  575. IN OUT PINFFILEGEN Context,
  576. IN HKEY Key,
  577. IN LPCTSTR Subkey,
  578. IN LPCTSTR Value OPTIONAL
  579. )
  580. {
  581. DWORD rc;
  582. rc = pInfRegLineCommon(Context,Context->DelRegFile,Key,Subkey,Value);
  583. if(rc == NO_ERROR) {
  584. rc = GenInfWriteString(Context,Context->DelRegFile,TEXT("\r\n"),AddQuotesNone);
  585. if(rc == NO_ERROR) {
  586. rc = FlushGenInfLineBuf(Context,Context->DelRegFile);
  587. }
  588. }
  589. return(rc);
  590. }
  591. #endif