manifest.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675
  1. /*
  2. * Manifest parser for WUSA
  3. *
  4. * Copyright 2015 Michael Müller
  5. * Copyright 2015 Sebastian Lackner
  6. *
  7. * This library is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * This library is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with this library; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
  20. */
  21. #include <windows.h>
  22. #define COBJMACROS
  23. #include <initguid.h>
  24. #include <msxml.h>
  25. #include "wine/debug.h"
  26. #include "wine/list.h"
  27. #include "wusa.h"
  28. WINE_DEFAULT_DEBUG_CHANNEL(wusa);
  29. static struct dependency_entry *alloc_dependency(void)
  30. {
  31. struct dependency_entry *entry = calloc(1, sizeof(*entry));
  32. if (!entry) ERR("Failed to allocate memory for dependency\n");
  33. return entry;
  34. }
  35. static struct fileop_entry *alloc_fileop(void)
  36. {
  37. struct fileop_entry *entry = calloc(1, sizeof(*entry));
  38. if (!entry) ERR("Failed to allocate memory for fileop\n");
  39. return entry;
  40. }
  41. static struct registrykv_entry *alloc_registrykv(void)
  42. {
  43. struct registrykv_entry *entry = calloc(1, sizeof(*entry));
  44. if (!entry) ERR("Failed to allocate memory for registrykv\n");
  45. return entry;
  46. }
  47. static struct registryop_entry *alloc_registryop(void)
  48. {
  49. struct registryop_entry *entry = calloc(1, sizeof(*entry));
  50. if (!entry) ERR("Failed to allocate memory for registryop\n");
  51. else
  52. {
  53. list_init(&entry->keyvalues);
  54. }
  55. return entry;
  56. }
  57. static struct assembly_entry *alloc_assembly(void)
  58. {
  59. struct assembly_entry *entry = calloc(1, sizeof(*entry));
  60. if (!entry) ERR("Failed to allocate memory for assembly\n");
  61. else
  62. {
  63. list_init(&entry->dependencies);
  64. list_init(&entry->fileops);
  65. list_init(&entry->registryops);
  66. }
  67. return entry;
  68. }
  69. static void clear_identity(struct assembly_identity *entry)
  70. {
  71. free(entry->name);
  72. free(entry->version);
  73. free(entry->architecture);
  74. free(entry->language);
  75. free(entry->pubkey_token);
  76. }
  77. void free_dependency(struct dependency_entry *entry)
  78. {
  79. clear_identity(&entry->identity);
  80. free(entry);
  81. }
  82. static void free_fileop(struct fileop_entry *entry)
  83. {
  84. free(entry->source);
  85. free(entry->target);
  86. free(entry);
  87. }
  88. static void free_registrykv(struct registrykv_entry *entry)
  89. {
  90. free(entry->name);
  91. free(entry->value_type);
  92. free(entry->value);
  93. free(entry);
  94. }
  95. static void free_registryop(struct registryop_entry *entry)
  96. {
  97. struct registrykv_entry *keyvalue, *keyvalue2;
  98. free(entry->key);
  99. LIST_FOR_EACH_ENTRY_SAFE(keyvalue, keyvalue2, &entry->keyvalues, struct registrykv_entry, entry)
  100. {
  101. list_remove(&keyvalue->entry);
  102. free_registrykv(keyvalue);
  103. }
  104. free(entry);
  105. }
  106. void free_assembly(struct assembly_entry *entry)
  107. {
  108. struct dependency_entry *dependency, *dependency2;
  109. struct fileop_entry *fileop, *fileop2;
  110. struct registryop_entry *registryop, *registryop2;
  111. free(entry->filename);
  112. free(entry->displayname);
  113. clear_identity(&entry->identity);
  114. LIST_FOR_EACH_ENTRY_SAFE(dependency, dependency2, &entry->dependencies, struct dependency_entry, entry)
  115. {
  116. list_remove(&dependency->entry);
  117. free_dependency(dependency);
  118. }
  119. LIST_FOR_EACH_ENTRY_SAFE(fileop, fileop2, &entry->fileops, struct fileop_entry, entry)
  120. {
  121. list_remove(&fileop->entry);
  122. free_fileop(fileop);
  123. }
  124. LIST_FOR_EACH_ENTRY_SAFE(registryop, registryop2, &entry->registryops, struct registryop_entry, entry)
  125. {
  126. list_remove(&registryop->entry);
  127. free_registryop(registryop);
  128. }
  129. free(entry);
  130. }
  131. static WCHAR *get_xml_attribute(IXMLDOMElement *root, const WCHAR *name)
  132. {
  133. WCHAR *ret = NULL;
  134. VARIANT var;
  135. BSTR bstr;
  136. if ((bstr = SysAllocString(name)))
  137. {
  138. VariantInit(&var);
  139. if (SUCCEEDED(IXMLDOMElement_getAttribute(root, bstr, &var)))
  140. {
  141. ret = (V_VT(&var) == VT_BSTR) ? wcsdup(V_BSTR(&var)) : NULL;
  142. VariantClear(&var);
  143. }
  144. SysFreeString(bstr);
  145. }
  146. return ret;
  147. }
  148. static BOOL check_xml_tagname(IXMLDOMElement *root, const WCHAR *tagname)
  149. {
  150. BOOL ret = FALSE;
  151. BSTR bstr;
  152. if (SUCCEEDED(IXMLDOMElement_get_tagName(root, &bstr)))
  153. {
  154. ret = !wcscmp(bstr, tagname);
  155. SysFreeString(bstr);
  156. }
  157. return ret;
  158. }
  159. static IXMLDOMElement *select_xml_node(IXMLDOMElement *root, const WCHAR *name)
  160. {
  161. IXMLDOMElement *ret = NULL;
  162. IXMLDOMNode *node;
  163. BSTR bstr;
  164. if ((bstr = SysAllocString(name)))
  165. {
  166. if (SUCCEEDED(IXMLDOMElement_selectSingleNode(root, bstr, &node)))
  167. {
  168. if (FAILED(IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void **)&ret)))
  169. ret = NULL;
  170. IXMLDOMNode_Release(node);
  171. }
  172. SysFreeString(bstr);
  173. }
  174. return ret;
  175. }
  176. static BOOL call_xml_callbacks(IXMLDOMElement *root, BOOL (*func)(IXMLDOMElement *child, WCHAR *tagname, void *context), void *context)
  177. {
  178. IXMLDOMNodeList *children;
  179. IXMLDOMElement *child;
  180. IXMLDOMNode *node;
  181. BSTR tagname;
  182. BOOL ret = TRUE;
  183. if (FAILED(IXMLDOMElement_get_childNodes(root, &children)))
  184. return FALSE;
  185. while (ret && IXMLDOMNodeList_nextNode(children, &node) == S_OK)
  186. {
  187. if (SUCCEEDED(IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void **)&child)))
  188. {
  189. if (SUCCEEDED(IXMLDOMElement_get_tagName(child, &tagname)))
  190. {
  191. ret = func(child, tagname, context);
  192. SysFreeString(tagname);
  193. }
  194. IXMLDOMElement_Release(child);
  195. }
  196. IXMLDOMNode_Release(node);
  197. }
  198. IXMLDOMNodeList_Release(children);
  199. return ret;
  200. }
  201. static IXMLDOMElement *load_xml(const WCHAR *filename)
  202. {
  203. IXMLDOMDocument *document = NULL;
  204. IXMLDOMElement *root = NULL;
  205. VARIANT_BOOL success;
  206. VARIANT variant;
  207. BSTR bstr;
  208. TRACE("Loading XML from %s\n", debugstr_w(filename));
  209. if (!(bstr = SysAllocString(filename)))
  210. return FALSE;
  211. if (SUCCEEDED(CoCreateInstance(&CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void **)&document)))
  212. {
  213. VariantInit(&variant);
  214. V_VT(&variant) = VT_BSTR;
  215. V_BSTR(&variant) = bstr;
  216. if (SUCCEEDED(IXMLDOMDocument_load(document, variant, &success)) && success)
  217. {
  218. if (FAILED(IXMLDOMDocument_get_documentElement(document, &root)))
  219. root = NULL;
  220. }
  221. IXMLDOMDocument_Release(document);
  222. }
  223. SysFreeString(bstr);
  224. return root;
  225. }
  226. static BOOL read_identity(IXMLDOMElement *root, struct assembly_identity *identity)
  227. {
  228. memset(identity, 0, sizeof(*identity));
  229. if (!(identity->name = get_xml_attribute(root, L"name"))) goto error;
  230. if (!(identity->version = get_xml_attribute(root, L"version"))) goto error;
  231. if (!(identity->architecture = get_xml_attribute(root, L"processorArchitecture"))) goto error;
  232. if (!(identity->language = get_xml_attribute(root, L"language"))) goto error;
  233. if (!(identity->pubkey_token = get_xml_attribute(root, L"publicKeyToken"))) goto error;
  234. return TRUE;
  235. error:
  236. clear_identity(identity);
  237. memset(identity, 0, sizeof(*identity));
  238. return FALSE;
  239. }
  240. /* <assembly><dependency><dependentAssembly> */
  241. static BOOL read_dependent_assembly(IXMLDOMElement *root, struct assembly_identity *identity)
  242. {
  243. IXMLDOMElement *child = NULL;
  244. WCHAR *dependency_type;
  245. BOOL ret = FALSE;
  246. if (!(dependency_type = get_xml_attribute(root, L"dependencyType")))
  247. {
  248. WARN("Failed to get dependency type, assuming install\n");
  249. }
  250. if (dependency_type && wcscmp(dependency_type, L"install") && wcscmp(dependency_type, L"prerequisite"))
  251. {
  252. FIXME("Unimplemented dependency type %s\n", debugstr_w(dependency_type));
  253. goto error;
  254. }
  255. if (!(child = select_xml_node(root, L".//assemblyIdentity")))
  256. {
  257. FIXME("Failed to find assemblyIdentity child node\n");
  258. goto error;
  259. }
  260. ret = read_identity(child, identity);
  261. error:
  262. if (child) IXMLDOMElement_Release(child);
  263. free(dependency_type);
  264. return ret;
  265. }
  266. /* <assembly><dependency> */
  267. static BOOL read_dependency(IXMLDOMElement *child, WCHAR *tagname, void *context)
  268. {
  269. struct assembly_entry *assembly = context;
  270. struct dependency_entry *entry;
  271. if (wcscmp(tagname, L"dependentAssembly"))
  272. {
  273. FIXME("Don't know how to handle dependency tag %s\n", debugstr_w(tagname));
  274. return FALSE;
  275. }
  276. if ((entry = alloc_dependency()))
  277. {
  278. if (read_dependent_assembly(child, &entry->identity))
  279. {
  280. TRACE("Found dependency %s\n", debugstr_w(entry->identity.name));
  281. list_add_tail(&assembly->dependencies, &entry->entry);
  282. return TRUE;
  283. }
  284. free_dependency(entry);
  285. }
  286. return FALSE;
  287. }
  288. static BOOL iter_dependency(IXMLDOMElement *root, struct assembly_entry *assembly)
  289. {
  290. return call_xml_callbacks(root, read_dependency, assembly);
  291. }
  292. /* <assembly><package><update><component> */
  293. /* <assembly><package><update><package> */
  294. static BOOL read_components(IXMLDOMElement *child, WCHAR *tagname, void *context)
  295. {
  296. struct assembly_entry *assembly = context;
  297. struct dependency_entry *entry;
  298. if (wcscmp(tagname, L"assemblyIdentity"))
  299. {
  300. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  301. return TRUE;
  302. }
  303. if ((entry = alloc_dependency()))
  304. {
  305. if (read_identity(child, &entry->identity))
  306. {
  307. TRACE("Found identity %s\n", debugstr_w(entry->identity.name));
  308. list_add_tail(&assembly->dependencies, &entry->entry);
  309. return TRUE;
  310. }
  311. free_dependency(entry);
  312. }
  313. return FALSE;
  314. }
  315. static BOOL iter_components(IXMLDOMElement *root, struct assembly_entry *assembly)
  316. {
  317. return call_xml_callbacks(root, read_components, assembly);
  318. }
  319. /* <assembly><package><update> */
  320. static BOOL read_update(IXMLDOMElement *child, WCHAR *tagname, void *context)
  321. {
  322. struct assembly_entry *assembly = context;
  323. if (!wcscmp(tagname, L"component"))
  324. return iter_components(child, assembly);
  325. if (!wcscmp(tagname, L"package"))
  326. return iter_components(child, assembly);
  327. if (!wcscmp(tagname, L"applicable"))
  328. return TRUE;
  329. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  330. return FALSE;
  331. }
  332. static BOOL iter_update(IXMLDOMElement *root, struct assembly_entry *assembly)
  333. {
  334. return call_xml_callbacks(root, read_update, assembly);
  335. }
  336. /* <assembly><package> */
  337. static BOOL read_package(IXMLDOMElement *child, WCHAR *tagname, void *context)
  338. {
  339. struct assembly_entry *assembly = context;
  340. if (!wcscmp(tagname, L"update"))
  341. return iter_update(child, assembly);
  342. if (!wcscmp(tagname, L"parent"))
  343. return TRUE;
  344. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  345. return TRUE;
  346. }
  347. static BOOL iter_package(IXMLDOMElement *root, struct assembly_entry *assembly)
  348. {
  349. return call_xml_callbacks(root, read_package, assembly);
  350. }
  351. /* <assembly><file> */
  352. static BOOL read_file(IXMLDOMElement *root, struct assembly_entry *assembly)
  353. {
  354. struct fileop_entry *entry;
  355. if (!(entry = alloc_fileop()))
  356. return FALSE;
  357. if (!(entry->source = get_xml_attribute(root, L"sourceName"))) goto error;
  358. if (!(entry->target = get_xml_attribute(root, L"destinationPath"))) goto error;
  359. TRACE("Found fileop %s -> %s\n", debugstr_w(entry->source), debugstr_w(entry->target));
  360. list_add_tail(&assembly->fileops, &entry->entry);
  361. return TRUE;
  362. error:
  363. free_fileop(entry);
  364. return FALSE;
  365. }
  366. /* <assembly><registryKeys><registryKey> */
  367. static BOOL read_registry_key(IXMLDOMElement *child, WCHAR *tagname, void *context)
  368. {
  369. struct registryop_entry *registryop = context;
  370. struct registrykv_entry *entry;
  371. if (!wcscmp(tagname, L"securityDescriptor")) return TRUE;
  372. if (!wcscmp(tagname, L"systemProtection")) return TRUE;
  373. if (wcscmp(tagname, L"registryValue"))
  374. {
  375. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  376. return TRUE;
  377. }
  378. if (!(entry = alloc_registrykv()))
  379. return FALSE;
  380. if (!(entry->value_type = get_xml_attribute(child, L"valueType"))) goto error;
  381. entry->name = get_xml_attribute(child, L"name"); /* optional */
  382. entry->value = get_xml_attribute(child, L"value"); /* optional */
  383. TRACE("Found registry %s -> %s\n", debugstr_w(entry->name), debugstr_w(entry->value));
  384. list_add_tail(&registryop->keyvalues, &entry->entry);
  385. return TRUE;
  386. error:
  387. free_registrykv(entry);
  388. return FALSE;
  389. }
  390. static BOOL iter_registry_key(IXMLDOMElement *root, struct registryop_entry *registryop)
  391. {
  392. return call_xml_callbacks(root, read_registry_key, registryop);
  393. }
  394. /* <assembly><registryKeys> */
  395. static BOOL read_registry_keys(IXMLDOMElement *child, WCHAR *tagname, void *context)
  396. {
  397. struct assembly_entry *assembly = context;
  398. struct registryop_entry *entry;
  399. WCHAR *keyname;
  400. if (wcscmp(tagname, L"registryKey"))
  401. {
  402. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  403. return TRUE;
  404. }
  405. if (!(keyname = get_xml_attribute(child, L"keyName")))
  406. {
  407. FIXME("RegistryKey tag doesn't specify keyName\n");
  408. return FALSE;
  409. }
  410. if ((entry = alloc_registryop()))
  411. {
  412. list_init(&entry->keyvalues);
  413. if (iter_registry_key(child, entry))
  414. {
  415. entry->key = keyname;
  416. TRACE("Found registryop %s\n", debugstr_w(entry->key));
  417. list_add_tail(&assembly->registryops, &entry->entry);
  418. return TRUE;
  419. }
  420. free_registryop(entry);
  421. }
  422. free(keyname);
  423. return FALSE;
  424. }
  425. static BOOL iter_registry_keys(IXMLDOMElement *root, struct assembly_entry *assembly)
  426. {
  427. return call_xml_callbacks(root, read_registry_keys, assembly);
  428. }
  429. /* <assembly> */
  430. static BOOL read_assembly(IXMLDOMElement *child, WCHAR *tagname, void *context)
  431. {
  432. struct assembly_entry *assembly = context;
  433. if (!wcscmp(tagname, L"assemblyIdentity") && !assembly->identity.name)
  434. return read_identity(child, &assembly->identity);
  435. if (!wcscmp(tagname, L"dependency"))
  436. return iter_dependency(child, assembly);
  437. if (!wcscmp(tagname, L"package"))
  438. return iter_package(child, assembly);
  439. if (!wcscmp(tagname, L"file"))
  440. return read_file(child, assembly);
  441. if (!wcscmp(tagname, L"registryKeys"))
  442. return iter_registry_keys(child, assembly);
  443. if (!wcscmp(tagname, L"trustInfo"))
  444. return TRUE;
  445. if (!wcscmp(tagname, L"configuration"))
  446. return TRUE;
  447. if (!wcscmp(tagname, L"deployment"))
  448. return TRUE;
  449. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  450. return TRUE;
  451. }
  452. static BOOL iter_assembly(IXMLDOMElement *root, struct assembly_entry *assembly)
  453. {
  454. return call_xml_callbacks(root, read_assembly, assembly);
  455. }
  456. struct assembly_entry *load_manifest(const WCHAR *filename)
  457. {
  458. struct assembly_entry *entry = NULL;
  459. IXMLDOMElement *root = NULL;
  460. TRACE("Loading manifest %s\n", debugstr_w(filename));
  461. if (!(root = load_xml(filename))) return NULL;
  462. if (!check_xml_tagname(root, L"assembly"))
  463. {
  464. FIXME("Didn't find assembly root node?\n");
  465. goto done;
  466. }
  467. if ((entry = alloc_assembly()))
  468. {
  469. entry->filename = wcsdup(filename);
  470. entry->displayname = get_xml_attribute(root, L"displayName");
  471. if (iter_assembly(root, entry)) goto done;
  472. free_assembly(entry);
  473. entry = NULL;
  474. }
  475. done:
  476. IXMLDOMElement_Release(root);
  477. return entry;
  478. }
  479. /* <unattend><servicing><package> */
  480. static BOOL read_update_package(IXMLDOMElement *child, WCHAR *tagname, void *context)
  481. {
  482. struct dependency_entry *entry;
  483. struct list *update_list = context;
  484. if (!wcscmp(tagname, L"source")) return TRUE;
  485. if (wcscmp(tagname, L"assemblyIdentity"))
  486. {
  487. TRACE("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  488. return TRUE;
  489. }
  490. if ((entry = alloc_dependency()))
  491. {
  492. if (read_identity(child, &entry->identity))
  493. {
  494. TRACE("Found update %s\n", debugstr_w(entry->identity.name));
  495. list_add_tail(update_list, &entry->entry);
  496. return TRUE;
  497. }
  498. free(entry);
  499. }
  500. return FALSE;
  501. }
  502. static BOOL iter_update_package(IXMLDOMElement *root, struct list *update_list)
  503. {
  504. return call_xml_callbacks(root, read_update_package, update_list);
  505. }
  506. /* <unattend><servicing> */
  507. static BOOL read_servicing(IXMLDOMElement *child, WCHAR *tagname, void *context)
  508. {
  509. struct list *update_list = context;
  510. WCHAR *action;
  511. BOOL ret = TRUE;
  512. if (wcscmp(tagname, L"package"))
  513. {
  514. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  515. return TRUE;
  516. }
  517. if (!(action = get_xml_attribute(child, L"action")))
  518. {
  519. FIXME("Servicing tag doesn't specify action\n");
  520. return FALSE;
  521. }
  522. if (!wcscmp(action, L"install"))
  523. ret = iter_update_package(child, update_list);
  524. else
  525. FIXME("action %s not supported\n", debugstr_w(action));
  526. free(action);
  527. return ret;
  528. }
  529. static BOOL iter_servicing(IXMLDOMElement *root, struct list *update_list)
  530. {
  531. return call_xml_callbacks(root, read_servicing, update_list);
  532. }
  533. /* <unattend> */
  534. static BOOL read_unattend(IXMLDOMElement *child, WCHAR *tagname, void *context)
  535. {
  536. struct list *update_list = context;
  537. if (wcscmp(tagname, L"servicing"))
  538. {
  539. FIXME("Ignoring unexpected tag %s\n", debugstr_w(tagname));
  540. return TRUE;
  541. }
  542. return iter_servicing(child, update_list);
  543. }
  544. static BOOL iter_unattend(IXMLDOMElement *root, struct list *update_list)
  545. {
  546. return call_xml_callbacks(root, read_unattend, update_list);
  547. }
  548. BOOL load_update(const WCHAR *filename, struct list *update_list)
  549. {
  550. IXMLDOMElement *root = NULL;
  551. BOOL ret = FALSE;
  552. TRACE("Reading update %s\n", debugstr_w(filename));
  553. if (!(root = load_xml(filename))) return FALSE;
  554. if (!check_xml_tagname(root, L"unattend"))
  555. {
  556. FIXME("Didn't find unattend root node?\n");
  557. goto done;
  558. }
  559. ret = iter_unattend(root, update_list);
  560. done:
  561. IXMLDOMElement_Release(root);
  562. return ret;
  563. }