xref: /reactos/dll/win32/msi/msipriv.h (revision 02e84521)
1 /*
2  * Implementation of the Microsoft Installer (msi.dll)
3  *
4  * Copyright 2002-2005 Mike McCormack for CodeWeavers
5  * Copyright 2005 Aric Stewart for CodeWeavers
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 
22 #ifndef __WINE_MSI_PRIVATE__
23 #define __WINE_MSI_PRIVATE__
24 
25 #include <stdarg.h>
26 
27 #include "windef.h"
28 #include "winbase.h"
29 #include "fdi.h"
30 #include "msi.h"
31 #include "msiquery.h"
32 #include "msidefs.h"
33 #include "objbase.h"
34 #include "objidl.h"
35 #include "fusion.h"
36 #include "winnls.h"
37 #include "winver.h"
38 #include "wine/list.h"
39 #include "wine/debug.h"
40 
41 static const BOOL is_64bit = sizeof(void *) > sizeof(int);
42 BOOL is_wow64 DECLSPEC_HIDDEN;
43 
44 #define MSI_DATASIZEMASK 0x00ff
45 #define MSITYPE_VALID    0x0100
46 #define MSITYPE_LOCALIZABLE 0x200
47 #define MSITYPE_STRING   0x0800
48 #define MSITYPE_NULLABLE 0x1000
49 #define MSITYPE_KEY      0x2000
50 #define MSITYPE_TEMPORARY 0x4000
51 #define MSITYPE_UNKNOWN   0x8000
52 
53 #define MAX_STREAM_NAME_LEN     62
54 #define LONG_STR_BYTES  3
55 
56 /* Install UI level mask for AND operation to exclude flags */
57 #define INSTALLUILEVEL_MASK             0x0007
58 
59 #define MSITYPE_IS_BINARY(type) (((type) & ~MSITYPE_NULLABLE) == (MSITYPE_STRING|MSITYPE_VALID))
60 
61 struct tagMSITABLE;
62 typedef struct tagMSITABLE MSITABLE;
63 
64 struct string_table;
65 typedef struct string_table string_table;
66 
67 struct tagMSIOBJECTHDR;
68 typedef struct tagMSIOBJECTHDR MSIOBJECTHDR;
69 
70 typedef VOID (*msihandledestructor)( MSIOBJECTHDR * );
71 
72 struct tagMSIOBJECTHDR
73 {
74     UINT magic;
75     UINT type;
76     LONG refcount;
77     msihandledestructor destructor;
78 };
79 
80 #define MSI_INITIAL_MEDIA_TRANSFORM_OFFSET 10000
81 #define MSI_INITIAL_MEDIA_TRANSFORM_DISKID 30000
82 
83 typedef struct tagMSISTREAM
84 {
85     UINT str_index;
86     IStream *stream;
87 } MSISTREAM;
88 
89 typedef struct tagMSITRANSFORM
90 {
91     struct list entry;
92     IStorage *stg;
93 } MSITRANSFORM;
94 
95 typedef struct tagMSIDATABASE
96 {
97     MSIOBJECTHDR hdr;
98     IStorage *storage;
99     string_table *strings;
100     UINT bytes_per_strref;
101     LPWSTR path;
102     LPWSTR deletefile;
103     LPWSTR tempfolder;
104     LPCWSTR mode;
105     UINT media_transform_offset;
106     UINT media_transform_disk_id;
107     struct list tables;
108     struct list transforms;
109     MSISTREAM *streams;
110     UINT num_streams;
111     UINT num_streams_allocated;
112 } MSIDATABASE;
113 
114 typedef struct tagMSIVIEW MSIVIEW;
115 
116 typedef struct tagMSIQUERY
117 {
118     MSIOBJECTHDR hdr;
119     MSIVIEW *view;
120     UINT row;
121     MSIDATABASE *db;
122     struct list mem;
123 } MSIQUERY;
124 
125 /* maybe we can use a Variant instead of doing it ourselves? */
126 typedef struct tagMSIFIELD
127 {
128     UINT type;
129     union
130     {
131         INT iVal;
132         INT_PTR pVal;
133         LPWSTR szwVal;
134         IStream *stream;
135     } u;
136     int len;
137 } MSIFIELD;
138 
139 typedef struct tagMSIRECORD
140 {
141     MSIOBJECTHDR hdr;
142     UINT count;       /* as passed to MsiCreateRecord */
143     MSIFIELD fields[1]; /* nb. array size is count+1 */
144 } MSIRECORD;
145 
146 typedef struct tagMSISOURCELISTINFO
147 {
148     struct list entry;
149     DWORD context;
150     DWORD options;
151     LPCWSTR property;
152     LPWSTR value;
153 } MSISOURCELISTINFO;
154 
155 typedef struct tagMSIMEDIADISK
156 {
157     struct list entry;
158     DWORD context;
159     DWORD options;
160     DWORD disk_id;
161     LPWSTR volume_label;
162     LPWSTR disk_prompt;
163 } MSIMEDIADISK;
164 
165 typedef struct tagMSIMEDIAINFO
166 {
167     UINT disk_id;
168     UINT type;
169     UINT last_sequence;
170     LPWSTR disk_prompt;
171     LPWSTR cabinet;
172     LPWSTR volume_label;
173     BOOL is_continuous;
174     BOOL is_extracted;
175     WCHAR sourcedir[MAX_PATH];
176 } MSIMEDIAINFO;
177 
178 typedef struct tagMSICABINETSTREAM
179 {
180     struct list entry;
181     UINT disk_id;
182     IStorage *storage;
183     WCHAR *stream;
184 } MSICABINETSTREAM;
185 
186 typedef struct tagMSIPATCHINFO
187 {
188     struct list entry;
189     LPWSTR patchcode;
190     LPWSTR products;
191     LPWSTR transforms;
192     LPWSTR filename;
193     LPWSTR localfile;
194     MSIPATCHSTATE state;
195     DWORD uninstallable;
196     BOOL delete_on_close;
197     BOOL registered;
198     UINT disk_id;
199 } MSIPATCHINFO;
200 
201 typedef struct tagMSIBINARY
202 {
203     struct list entry;
204     WCHAR *source;
205     WCHAR *tmpfile;
206     HMODULE module;
207 } MSIBINARY;
208 
209 typedef struct _column_info
210 {
211     LPCWSTR table;
212     LPCWSTR column;
213     INT   type;
214     BOOL   temporary;
215     struct expr *val;
216     struct _column_info *next;
217 } column_info;
218 
219 typedef const struct tagMSICOLUMNHASHENTRY *MSIITERHANDLE;
220 
221 typedef struct tagMSIVIEWOPS
222 {
223     /*
224      * fetch_int - reads one integer from {row,col} in the table
225      *
226      *  This function should be called after the execute method.
227      *  Data returned by the function should not change until
228      *   close or delete is called.
229      *  To get a string value, query the database's string table with
230      *   the integer value returned from this function.
231      */
232     UINT (*fetch_int)( struct tagMSIVIEW *view, UINT row, UINT col, UINT *val );
233 
234     /*
235      * fetch_stream - gets a stream from {row,col} in the table
236      *
237      *  This function is similar to fetch_int, except fetches a
238      *    stream instead of an integer.
239      */
240     UINT (*fetch_stream)( struct tagMSIVIEW *view, UINT row, UINT col, IStream **stm );
241 
242     /*
243      * get_row - gets values from a row
244      *
245      */
246     UINT (*get_row)( struct tagMSIVIEW *view, UINT row, MSIRECORD **rec );
247 
248     /*
249      * set_row - sets values in a row as specified by mask
250      *
251      *  Similar semantics to fetch_int
252      */
253     UINT (*set_row)( struct tagMSIVIEW *view, UINT row, MSIRECORD *rec, UINT mask );
254 
255     /*
256      * Inserts a new row into the database from the records contents
257      */
258     UINT (*insert_row)( struct tagMSIVIEW *view, MSIRECORD *record, UINT row, BOOL temporary );
259 
260     /*
261      * Deletes a row from the database
262      */
263     UINT (*delete_row)( struct tagMSIVIEW *view, UINT row );
264 
265     /*
266      * execute - loads the underlying data into memory so it can be read
267      */
268     UINT (*execute)( struct tagMSIVIEW *view, MSIRECORD *record );
269 
270     /*
271      * close - clears the data read by execute from memory
272      */
273     UINT (*close)( struct tagMSIVIEW *view );
274 
275     /*
276      * get_dimensions - returns the number of rows or columns in a table.
277      *
278      *  The number of rows can only be queried after the execute method
279      *   is called. The number of columns can be queried at any time.
280      */
281     UINT (*get_dimensions)( struct tagMSIVIEW *view, UINT *rows, UINT *cols );
282 
283     /*
284      * get_column_info - returns the name and type of a specific column
285      *
286      *  The column information can be queried at any time.
287      */
288     UINT (*get_column_info)( struct tagMSIVIEW *view, UINT n, LPCWSTR *name, UINT *type,
289                              BOOL *temporary, LPCWSTR *table_name );
290 
291     /*
292      * modify - not yet implemented properly
293      */
294     UINT (*modify)( struct tagMSIVIEW *view, MSIMODIFY eModifyMode, MSIRECORD *record, UINT row );
295 
296     /*
297      * delete - destroys the structure completely
298      */
299     UINT (*delete)( struct tagMSIVIEW * );
300 
301     /*
302      * find_matching_rows - iterates through rows that match a value
303      *
304      * If the column type is a string then a string ID should be passed in.
305      *  If the value to be looked up is an integer then no transformation of
306      *  the input value is required, except if the column is a string, in which
307      *  case a string ID should be passed in.
308      * The handle is an input/output parameter that keeps track of the current
309      *  position in the iteration. It must be initialised to zero before the
310      *  first call and continued to be passed in to subsequent calls.
311      */
312     UINT (*find_matching_rows)( struct tagMSIVIEW *view, UINT col, UINT val, UINT *row, MSIITERHANDLE *handle );
313 
314     /*
315      * add_ref - increases the reference count of the table
316      */
317     UINT (*add_ref)( struct tagMSIVIEW *view );
318 
319     /*
320      * release - decreases the reference count of the table
321      */
322     UINT (*release)( struct tagMSIVIEW *view );
323 
324     /*
325      * add_column - adds a column to the table
326      */
327     UINT (*add_column)( struct tagMSIVIEW *view, LPCWSTR table, UINT number, LPCWSTR column, UINT type, BOOL hold );
328 
329     /*
330      * remove_column - removes the column represented by table name and column number from the table
331      */
332     UINT (*remove_column)( struct tagMSIVIEW *view, LPCWSTR table, UINT number );
333 
334     /*
335      * sort - orders the table by columns
336      */
337     UINT (*sort)( struct tagMSIVIEW *view, column_info *columns );
338 
339     /*
340      * drop - drops the table from the database
341      */
342     UINT (*drop)( struct tagMSIVIEW *view );
343 } MSIVIEWOPS;
344 
345 struct tagMSIVIEW
346 {
347     MSIOBJECTHDR hdr;
348     const MSIVIEWOPS *ops;
349     MSIDBERROR error;
350     const WCHAR *error_column;
351 };
352 
353 struct msi_dialog_tag;
354 typedef struct msi_dialog_tag msi_dialog;
355 
356 enum platform
357 {
358     PLATFORM_UNKNOWN,
359     PLATFORM_INTEL,
360     PLATFORM_INTEL64,
361     PLATFORM_X64,
362     PLATFORM_ARM
363 };
364 
365 enum clr_version
366 {
367     CLR_VERSION_V10,
368     CLR_VERSION_V11,
369     CLR_VERSION_V20,
370     CLR_VERSION_V40,
371     CLR_VERSION_MAX
372 };
373 
374 enum script
375 {
376     SCRIPT_NONE     = -1,
377     SCRIPT_INSTALL  = 0,
378     SCRIPT_COMMIT   = 1,
379     SCRIPT_ROLLBACK = 2,
380     SCRIPT_MAX      = 3
381 };
382 
383 typedef struct tagMSIPACKAGE
384 {
385     MSIOBJECTHDR hdr;
386     MSIDATABASE *db;
387     INT version;
388     enum platform platform;
389     UINT num_langids;
390     LANGID *langids;
391     struct list patches;
392     struct list components;
393     struct list features;
394     struct list files;
395     struct list filepatches;
396     struct list tempfiles;
397     struct list folders;
398     struct list binaries;
399     struct list cabinet_streams;
400     LPWSTR ActionFormat;
401     LPWSTR LastAction;
402     LPWSTR LastActionTemplate;
403     UINT   LastActionResult;
404     UINT   action_progress_increment;
405     HANDLE log_file;
406     IAssemblyCache *cache_net[CLR_VERSION_MAX];
407     IAssemblyCache *cache_sxs;
408 
409     struct list classes;
410     struct list extensions;
411     struct list progids;
412     struct list mimes;
413     struct list appids;
414 
415     LPWSTR *script_actions[SCRIPT_MAX];
416     int    script_actions_count[SCRIPT_MAX];
417     LPWSTR *unique_actions;
418     int    unique_actions_count;
419     BOOL   ExecuteSequenceRun;
420     UINT   InWhatSequence;
421 
422     struct list RunningActions;
423 
424     LPWSTR BaseURL;
425     LPWSTR PackagePath;
426     LPWSTR ProductCode;
427     LPWSTR localfile;
428     BOOL delete_on_close;
429 
430     INSTALLUILEVEL ui_level;
431     msi_dialog *dialog;
432     LPWSTR next_dialog;
433     float center_x;
434     float center_y;
435 
436     UINT WordCount;
437     UINT Context;
438 
439     struct list subscriptions;
440 
441     struct list sourcelist_info;
442     struct list sourcelist_media;
443 
444     unsigned char scheduled_action_running : 1;
445     unsigned char commit_action_running : 1;
446     unsigned char rollback_action_running : 1;
447     unsigned char need_reboot_at_end : 1;
448     unsigned char need_reboot_now : 1;
449     unsigned char need_rollback : 1;
450 } MSIPACKAGE;
451 
452 typedef struct tagMSIPREVIEW
453 {
454     MSIOBJECTHDR hdr;
455     MSIPACKAGE *package;
456     msi_dialog *dialog;
457 } MSIPREVIEW;
458 
459 #define MSI_MAX_PROPS 20
460 
461 typedef struct tagMSISUMMARYINFO
462 {
463     MSIOBJECTHDR hdr;
464     IStorage *storage;
465     DWORD update_count;
466     PROPVARIANT property[MSI_MAX_PROPS];
467 } MSISUMMARYINFO;
468 
469 typedef struct tagMSIFEATURE
470 {
471     struct list entry;
472     LPWSTR Feature;
473     LPWSTR Feature_Parent;
474     LPWSTR Title;
475     LPWSTR Description;
476     INT Display;
477     INT Level;
478     LPWSTR Directory;
479     INT Attributes;
480     INSTALLSTATE Installed;
481     INSTALLSTATE ActionRequest;
482     INSTALLSTATE Action;
483     struct list Children;
484     struct list Components;
485 } MSIFEATURE;
486 
487 typedef struct tagMSIASSEMBLY
488 {
489     LPWSTR feature;
490     LPWSTR manifest;
491     LPWSTR application;
492     DWORD attributes;
493     LPWSTR display_name;
494     LPWSTR tempdir;
495     BOOL installed;
496     BOOL clr_version[CLR_VERSION_MAX];
497 } MSIASSEMBLY;
498 
499 typedef struct tagMSICOMPONENT
500 {
501     struct list entry;
502     LPWSTR Component;
503     LPWSTR ComponentId;
504     LPWSTR Directory;
505     INT Attributes;
506     LPWSTR Condition;
507     LPWSTR KeyPath;
508     INSTALLSTATE Installed;
509     INSTALLSTATE ActionRequest;
510     INSTALLSTATE Action;
511     BOOL ForceLocalState;
512     BOOL Enabled;
513     INT  Cost;
514     INT  RefCount;
515     LPWSTR FullKeypath;
516     LPWSTR AdvertiseString;
517     MSIASSEMBLY *assembly;
518     int num_clients;
519 
520     unsigned int anyAbsent:1;
521     unsigned int hasAdvertisedFeature:1;
522     unsigned int hasLocalFeature:1;
523     unsigned int hasSourceFeature:1;
524 } MSICOMPONENT;
525 
526 typedef struct tagComponentList
527 {
528     struct list entry;
529     MSICOMPONENT *component;
530 } ComponentList;
531 
532 typedef struct tagFeatureList
533 {
534     struct list entry;
535     MSIFEATURE *feature;
536 } FeatureList;
537 
538 enum folder_state
539 {
540     FOLDER_STATE_UNINITIALIZED,
541     FOLDER_STATE_EXISTS,
542     FOLDER_STATE_CREATED,
543     FOLDER_STATE_CREATED_PERSISTENT,
544     FOLDER_STATE_REMOVED
545 };
546 
547 typedef struct tagMSIFOLDER
548 {
549     struct list entry;
550     struct list children;
551     LPWSTR Directory;
552     LPWSTR Parent;
553     LPWSTR TargetDefault;
554     LPWSTR SourceLongPath;
555     LPWSTR SourceShortPath;
556     LPWSTR ResolvedTarget;
557     LPWSTR ResolvedSource;
558     enum folder_state State;
559     BOOL persistent;
560     INT Cost;
561     INT Space;
562 } MSIFOLDER;
563 
564 typedef struct tagFolderList
565 {
566     struct list entry;
567     MSIFOLDER *folder;
568 } FolderList;
569 
570 typedef enum _msi_file_state {
571     msifs_invalid,
572     msifs_missing,
573     msifs_overwrite,
574     msifs_present,
575     msifs_installed,
576     msifs_skipped,
577     msifs_hashmatch
578 } msi_file_state;
579 
580 typedef struct tagMSIFILE
581 {
582     struct list entry;
583     LPWSTR File;
584     MSICOMPONENT *Component;
585     LPWSTR FileName;
586     LPWSTR ShortName;
587     LPWSTR LongName;
588     INT FileSize;
589     LPWSTR Version;
590     LPWSTR Language;
591     INT Attributes;
592     INT Sequence;
593     msi_file_state state;
594     LPWSTR  TargetPath;
595     BOOL IsCompressed;
596     MSIFILEHASHINFO hash;
597     UINT disk_id;
598 } MSIFILE;
599 
600 typedef struct tagMSIFILEPATCH
601 {
602     struct list entry;
603     MSIFILE *File;
604     INT Sequence;
605     INT PatchSize;
606     INT Attributes;
607     BOOL extracted;
608     UINT disk_id;
609     WCHAR *path;
610 } MSIFILEPATCH;
611 
612 typedef struct tagMSIAPPID
613 {
614     struct list entry;
615     LPWSTR AppID; /* Primary key */
616     LPWSTR RemoteServerName;
617     LPWSTR LocalServer;
618     LPWSTR ServiceParameters;
619     LPWSTR DllSurrogate;
620     BOOL ActivateAtStorage;
621     BOOL RunAsInteractiveUser;
622 } MSIAPPID;
623 
624 typedef struct tagMSIPROGID MSIPROGID;
625 
626 typedef struct tagMSICLASS
627 {
628     struct list entry;
629     LPWSTR clsid;     /* Primary Key */
630     LPWSTR Context;   /* Primary Key */
631     MSICOMPONENT *Component;
632     MSIPROGID *ProgID;
633     LPWSTR ProgIDText;
634     LPWSTR Description;
635     MSIAPPID *AppID;
636     LPWSTR FileTypeMask;
637     LPWSTR IconPath;
638     LPWSTR DefInprocHandler;
639     LPWSTR DefInprocHandler32;
640     LPWSTR Argument;
641     MSIFEATURE *Feature;
642     INT Attributes;
643     /* not in the table, set during installation */
644     INSTALLSTATE action;
645 } MSICLASS;
646 
647 typedef struct tagMSIMIME MSIMIME;
648 
649 typedef struct tagMSIEXTENSION
650 {
651     struct list entry;
652     LPWSTR Extension;  /* Primary Key */
653     MSICOMPONENT *Component;
654     MSIPROGID *ProgID;
655     LPWSTR ProgIDText;
656     MSIMIME *Mime;
657     MSIFEATURE *Feature;
658     /* not in the table, set during installation */
659     INSTALLSTATE action;
660     struct list verbs;
661 } MSIEXTENSION;
662 
663 struct tagMSIPROGID
664 {
665     struct list entry;
666     LPWSTR ProgID;  /* Primary Key */
667     MSIPROGID *Parent;
668     MSICLASS *Class;
669     LPWSTR Description;
670     LPWSTR IconPath;
671     /* not in the table, set during installation */
672     MSIPROGID *CurVer;
673     MSIPROGID *VersionInd;
674 };
675 
676 typedef struct tagMSIVERB
677 {
678     struct list entry;
679     LPWSTR Verb;
680     INT Sequence;
681     LPWSTR Command;
682     LPWSTR Argument;
683 } MSIVERB;
684 
685 struct tagMSIMIME
686 {
687     struct list entry;
688     LPWSTR ContentType;  /* Primary Key */
689     MSIEXTENSION *Extension;
690     LPWSTR suffix;
691     LPWSTR clsid;
692     MSICLASS *Class;
693 };
694 
695 #define SEQUENCE_UI       0x1
696 #define SEQUENCE_EXEC     0x2
697 
698 #define MSIHANDLETYPE_ANY 0
699 #define MSIHANDLETYPE_DATABASE 1
700 #define MSIHANDLETYPE_SUMMARYINFO 2
701 #define MSIHANDLETYPE_VIEW 3
702 #define MSIHANDLETYPE_RECORD 4
703 #define MSIHANDLETYPE_PACKAGE 5
704 #define MSIHANDLETYPE_PREVIEW 6
705 
706 #define MSI_MAJORVERSION 4
707 #define MSI_MINORVERSION 5
708 #define MSI_BUILDNUMBER 6001
709 
710 #define GUID_SIZE 39
711 #define SQUASHED_GUID_SIZE 33
712 
713 #define MSIHANDLE_MAGIC 0x4d434923
714 
715 /* handle unicode/ascii output in the Msi* API functions */
716 typedef struct {
717     BOOL unicode;
718     union {
719        LPSTR a;
720        LPWSTR w;
721     } str;
722 } awstring;
723 
724 typedef struct {
725     BOOL unicode;
726     union {
727        LPCSTR a;
728        LPCWSTR w;
729     } str;
730 } awcstring;
731 
732 UINT msi_strcpy_to_awstring(const WCHAR *, int, awstring *, DWORD *) DECLSPEC_HIDDEN;
733 
734 /* msi server interface */
735 extern HRESULT create_msi_custom_remote( IUnknown *pOuter, LPVOID *ppObj ) DECLSPEC_HIDDEN;
736 extern HRESULT create_msi_remote_package( IUnknown *pOuter, LPVOID *ppObj ) DECLSPEC_HIDDEN;
737 extern HRESULT create_msi_remote_database( IUnknown *pOuter, LPVOID *ppObj ) DECLSPEC_HIDDEN;
738 extern IUnknown *msi_get_remote(MSIHANDLE handle) DECLSPEC_HIDDEN;
739 
740 /* handle functions */
741 extern void *msihandle2msiinfo(MSIHANDLE handle, UINT type) DECLSPEC_HIDDEN;
742 extern MSIHANDLE alloc_msihandle( MSIOBJECTHDR * ) DECLSPEC_HIDDEN;
743 extern MSIHANDLE alloc_msi_remote_handle( IUnknown *unk ) DECLSPEC_HIDDEN;
744 extern void *alloc_msiobject(UINT type, UINT size, msihandledestructor destroy ) DECLSPEC_HIDDEN;
745 extern void msiobj_addref(MSIOBJECTHDR *) DECLSPEC_HIDDEN;
746 extern int msiobj_release(MSIOBJECTHDR *) DECLSPEC_HIDDEN;
747 extern void msiobj_lock(MSIOBJECTHDR *) DECLSPEC_HIDDEN;
748 extern void msiobj_unlock(MSIOBJECTHDR *) DECLSPEC_HIDDEN;
749 extern void msi_free_handle_table(void) DECLSPEC_HIDDEN;
750 
751 extern void free_cached_tables( MSIDATABASE *db ) DECLSPEC_HIDDEN;
752 extern UINT MSI_CommitTables( MSIDATABASE *db ) DECLSPEC_HIDDEN;
753 extern UINT msi_commit_streams( MSIDATABASE *db ) DECLSPEC_HIDDEN;
754 
755 
756 /* string table functions */
757 enum StringPersistence
758 {
759     StringPersistent = 0,
760     StringNonPersistent = 1
761 };
762 
763 extern BOOL msi_add_string( string_table *st, const WCHAR *data, int len, enum StringPersistence persistence ) DECLSPEC_HIDDEN;
764 extern UINT msi_string2id( const string_table *st, const WCHAR *data, int len, UINT *id ) DECLSPEC_HIDDEN;
765 extern VOID msi_destroy_stringtable( string_table *st ) DECLSPEC_HIDDEN;
766 extern const WCHAR *msi_string_lookup( const string_table *st, UINT id, int *len ) DECLSPEC_HIDDEN;
767 extern HRESULT msi_init_string_table( IStorage *stg ) DECLSPEC_HIDDEN;
768 extern string_table *msi_load_string_table( IStorage *stg, UINT *bytes_per_strref ) DECLSPEC_HIDDEN;
769 extern UINT msi_save_string_table( const string_table *st, IStorage *storage, UINT *bytes_per_strref ) DECLSPEC_HIDDEN;
770 extern UINT msi_get_string_table_codepage( const string_table *st ) DECLSPEC_HIDDEN;
771 extern UINT msi_set_string_table_codepage( string_table *st, UINT codepage ) DECLSPEC_HIDDEN;
772 extern WCHAR *msi_strdupW( const WCHAR *value, int len ) DECLSPEC_HIDDEN;
773 
774 extern BOOL TABLE_Exists( MSIDATABASE *db, LPCWSTR name ) DECLSPEC_HIDDEN;
775 extern MSICONDITION MSI_DatabaseIsTablePersistent( MSIDATABASE *db, LPCWSTR table ) DECLSPEC_HIDDEN;
776 
777 extern UINT read_stream_data( IStorage *stg, LPCWSTR stname, BOOL table,
778                               BYTE **pdata, UINT *psz ) DECLSPEC_HIDDEN;
779 extern UINT write_stream_data( IStorage *stg, LPCWSTR stname,
780                                LPCVOID data, UINT sz, BOOL bTable ) DECLSPEC_HIDDEN;
781 
782 /* transform functions */
783 extern UINT msi_table_apply_transform( MSIDATABASE *db, IStorage *stg ) DECLSPEC_HIDDEN;
784 extern UINT MSI_DatabaseApplyTransformW( MSIDATABASE *db,
785                  LPCWSTR szTransformFile, int iErrorCond ) DECLSPEC_HIDDEN;
786 extern void append_storage_to_db( MSIDATABASE *db, IStorage *stg ) DECLSPEC_HIDDEN;
787 extern UINT msi_apply_transforms( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
788 
789 /* patch functions */
790 extern UINT msi_check_patch_applicable( MSIPACKAGE *package, MSISUMMARYINFO *si ) DECLSPEC_HIDDEN;
791 extern UINT msi_apply_patches( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
792 extern UINT msi_apply_registered_patch( MSIPACKAGE *package, LPCWSTR patch_code ) DECLSPEC_HIDDEN;
793 extern void msi_free_patchinfo( MSIPATCHINFO *patch ) DECLSPEC_HIDDEN;
794 
795 /* action internals */
796 extern UINT MSI_InstallPackage( MSIPACKAGE *, LPCWSTR, LPCWSTR ) DECLSPEC_HIDDEN;
797 extern INT ACTION_ShowDialog( MSIPACKAGE*, LPCWSTR) DECLSPEC_HIDDEN;
798 extern INT ACTION_DialogBox( MSIPACKAGE*, LPCWSTR) DECLSPEC_HIDDEN;
799 extern UINT ACTION_ForceReboot(MSIPACKAGE *package) DECLSPEC_HIDDEN;
800 extern UINT MSI_Sequence( MSIPACKAGE *package, LPCWSTR szTable ) DECLSPEC_HIDDEN;
801 extern UINT MSI_SetFeatureStates( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
802 extern UINT msi_parse_command_line( MSIPACKAGE *package, LPCWSTR szCommandLine, BOOL preserve_case ) DECLSPEC_HIDDEN;
803 extern UINT msi_schedule_action( MSIPACKAGE *package, UINT script, const WCHAR *action ) DECLSPEC_HIDDEN;
804 extern INSTALLSTATE msi_get_component_action( MSIPACKAGE *package, MSICOMPONENT *comp ) DECLSPEC_HIDDEN;
805 extern INSTALLSTATE msi_get_feature_action( MSIPACKAGE *package, MSIFEATURE *feature ) DECLSPEC_HIDDEN;
806 extern UINT msi_load_all_components( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
807 extern UINT msi_load_all_features( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
808 extern UINT msi_validate_product_id( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
809 
810 /* record internals */
811 extern void MSI_CloseRecord( MSIOBJECTHDR * ) DECLSPEC_HIDDEN;
812 extern UINT MSI_RecordSetIStream( MSIRECORD *, UINT, IStream *) DECLSPEC_HIDDEN;
813 extern UINT MSI_RecordGetIStream( MSIRECORD *, UINT, IStream **) DECLSPEC_HIDDEN;
814 extern const WCHAR *MSI_RecordGetString( const MSIRECORD *, UINT ) DECLSPEC_HIDDEN;
815 extern MSIRECORD *MSI_CreateRecord( UINT ) DECLSPEC_HIDDEN;
816 extern UINT MSI_RecordSetInteger( MSIRECORD *, UINT, int ) DECLSPEC_HIDDEN;
817 extern UINT MSI_RecordSetIntPtr( MSIRECORD *, UINT, INT_PTR ) DECLSPEC_HIDDEN;
818 extern UINT MSI_RecordSetStringW( MSIRECORD *, UINT, LPCWSTR ) DECLSPEC_HIDDEN;
819 extern BOOL MSI_RecordIsNull( MSIRECORD *, UINT ) DECLSPEC_HIDDEN;
820 extern UINT MSI_RecordGetStringW( MSIRECORD * , UINT, LPWSTR, LPDWORD) DECLSPEC_HIDDEN;
821 extern UINT MSI_RecordGetStringA( MSIRECORD *, UINT, LPSTR, LPDWORD) DECLSPEC_HIDDEN;
822 extern int MSI_RecordGetInteger( MSIRECORD *, UINT ) DECLSPEC_HIDDEN;
823 extern INT_PTR MSI_RecordGetIntPtr( MSIRECORD *, UINT ) DECLSPEC_HIDDEN;
824 extern UINT MSI_RecordReadStream( MSIRECORD *, UINT, char *, LPDWORD) DECLSPEC_HIDDEN;
825 extern UINT MSI_RecordSetStream(MSIRECORD *, UINT, IStream *) DECLSPEC_HIDDEN;
826 extern UINT MSI_RecordGetFieldCount( const MSIRECORD *rec ) DECLSPEC_HIDDEN;
827 extern UINT MSI_RecordStreamToFile( MSIRECORD *, UINT, LPCWSTR ) DECLSPEC_HIDDEN;
828 extern UINT MSI_RecordSetStreamFromFileW( MSIRECORD *, UINT, LPCWSTR ) DECLSPEC_HIDDEN;
829 extern UINT MSI_RecordCopyField( MSIRECORD *, UINT, MSIRECORD *, UINT ) DECLSPEC_HIDDEN;
830 extern MSIRECORD *MSI_CloneRecord( MSIRECORD * ) DECLSPEC_HIDDEN;
831 extern BOOL MSI_RecordsAreEqual( MSIRECORD *, MSIRECORD * ) DECLSPEC_HIDDEN;
832 extern BOOL MSI_RecordsAreFieldsEqual(MSIRECORD *a, MSIRECORD *b, UINT field) DECLSPEC_HIDDEN;
833 extern UINT msi_record_set_string(MSIRECORD *, UINT, const WCHAR *, int) DECLSPEC_HIDDEN;
834 extern const WCHAR *msi_record_get_string(const MSIRECORD *, UINT, int *) DECLSPEC_HIDDEN;
835 extern void dump_record(MSIRECORD *) DECLSPEC_HIDDEN;
836 
837 /* stream internals */
838 extern void enum_stream_names( IStorage *stg ) DECLSPEC_HIDDEN;
839 extern LPWSTR encode_streamname(BOOL bTable, LPCWSTR in) DECLSPEC_HIDDEN;
840 extern BOOL decode_streamname(LPCWSTR in, LPWSTR out) DECLSPEC_HIDDEN;
841 
842 /* database internals */
843 extern UINT msi_get_stream( MSIDATABASE *, const WCHAR *, IStream ** ) DECLSPEC_HIDDEN;
844 extern UINT MSI_OpenDatabaseW( LPCWSTR, LPCWSTR, MSIDATABASE ** ) DECLSPEC_HIDDEN;
845 extern UINT MSI_DatabaseOpenViewW(MSIDATABASE *, LPCWSTR, MSIQUERY ** ) DECLSPEC_HIDDEN;
846 extern UINT MSI_OpenQuery( MSIDATABASE *, MSIQUERY **, LPCWSTR, ... ) DECLSPEC_HIDDEN;
847 typedef UINT (*record_func)( MSIRECORD *, LPVOID );
848 extern UINT MSI_IterateRecords( MSIQUERY *, LPDWORD, record_func, LPVOID ) DECLSPEC_HIDDEN;
849 extern MSIRECORD *MSI_QueryGetRecord( MSIDATABASE *db, LPCWSTR query, ... ) DECLSPEC_HIDDEN;
850 extern UINT MSI_DatabaseGetPrimaryKeys( MSIDATABASE *, LPCWSTR, MSIRECORD ** ) DECLSPEC_HIDDEN;
851 
852 /* view internals */
853 extern UINT MSI_ViewExecute( MSIQUERY*, MSIRECORD * ) DECLSPEC_HIDDEN;
854 extern UINT MSI_ViewFetch( MSIQUERY*, MSIRECORD ** ) DECLSPEC_HIDDEN;
855 extern UINT MSI_ViewClose( MSIQUERY* ) DECLSPEC_HIDDEN;
856 extern UINT MSI_ViewGetColumnInfo(MSIQUERY *, MSICOLINFO, MSIRECORD **) DECLSPEC_HIDDEN;
857 extern UINT MSI_ViewModify( MSIQUERY *, MSIMODIFY, MSIRECORD * ) DECLSPEC_HIDDEN;
858 extern UINT VIEW_find_column( MSIVIEW *, LPCWSTR, LPCWSTR, UINT * ) DECLSPEC_HIDDEN;
859 extern UINT msi_view_get_row(MSIDATABASE *, MSIVIEW *, UINT, MSIRECORD **) DECLSPEC_HIDDEN;
860 
861 /* install internals */
862 extern UINT MSI_SetInstallLevel( MSIPACKAGE *package, int iInstallLevel ) DECLSPEC_HIDDEN;
863 
864 /* package internals */
865 extern MSIPACKAGE *MSI_CreatePackage( MSIDATABASE *, LPCWSTR ) DECLSPEC_HIDDEN;
866 extern UINT MSI_OpenPackageW( LPCWSTR szPackage, MSIPACKAGE **pPackage ) DECLSPEC_HIDDEN;
867 extern UINT MSI_SetTargetPathW( MSIPACKAGE *, LPCWSTR, LPCWSTR ) DECLSPEC_HIDDEN;
868 extern INT MSI_ProcessMessageVerbatim( MSIPACKAGE *, INSTALLMESSAGE, MSIRECORD * ) DECLSPEC_HIDDEN;
869 extern INT MSI_ProcessMessage( MSIPACKAGE *, INSTALLMESSAGE, MSIRECORD * ) DECLSPEC_HIDDEN;
870 extern MSICONDITION MSI_EvaluateConditionW( MSIPACKAGE *, LPCWSTR ) DECLSPEC_HIDDEN;
871 extern UINT MSI_GetComponentStateW( MSIPACKAGE *, LPCWSTR, INSTALLSTATE *, INSTALLSTATE * ) DECLSPEC_HIDDEN;
872 extern UINT MSI_GetFeatureStateW( MSIPACKAGE *, LPCWSTR, INSTALLSTATE *, INSTALLSTATE * ) DECLSPEC_HIDDEN;
873 extern UINT MSI_SetFeatureStateW(MSIPACKAGE*, LPCWSTR, INSTALLSTATE ) DECLSPEC_HIDDEN;
874 extern UINT msi_download_file( LPCWSTR szUrl, LPWSTR filename ) DECLSPEC_HIDDEN;
875 extern UINT msi_package_add_info(MSIPACKAGE *, DWORD, DWORD, LPCWSTR, LPWSTR) DECLSPEC_HIDDEN;
876 extern UINT msi_package_add_media_disk(MSIPACKAGE *, DWORD, DWORD, DWORD, LPWSTR, LPWSTR) DECLSPEC_HIDDEN;
877 extern UINT msi_clone_properties(MSIDATABASE *) DECLSPEC_HIDDEN;
878 extern UINT msi_set_context(MSIPACKAGE *) DECLSPEC_HIDDEN;
879 extern void msi_adjust_privilege_properties(MSIPACKAGE *) DECLSPEC_HIDDEN;
880 extern UINT MSI_GetFeatureCost(MSIPACKAGE *, MSIFEATURE *, MSICOSTTREE, INSTALLSTATE, LPINT) DECLSPEC_HIDDEN;
881 
882 /* for deformating */
883 extern UINT MSI_FormatRecordW( MSIPACKAGE *, MSIRECORD *, LPWSTR, LPDWORD ) DECLSPEC_HIDDEN;
884 
885 /* registry data encoding/decoding functions */
886 extern BOOL unsquash_guid(LPCWSTR in, LPWSTR out) DECLSPEC_HIDDEN;
887 extern BOOL squash_guid(LPCWSTR in, LPWSTR out) DECLSPEC_HIDDEN;
888 extern BOOL encode_base85_guid(GUID *,LPWSTR) DECLSPEC_HIDDEN;
889 extern BOOL decode_base85_guid(LPCWSTR,GUID*) DECLSPEC_HIDDEN;
890 extern UINT MSIREG_OpenUninstallKey(const WCHAR *, enum platform, HKEY *, BOOL) DECLSPEC_HIDDEN;
891 extern UINT MSIREG_DeleteUninstallKey(const WCHAR *, enum platform) DECLSPEC_HIDDEN;
892 extern UINT MSIREG_OpenProductKey(LPCWSTR szProduct, LPCWSTR szUserSid,
893                                   MSIINSTALLCONTEXT context, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
894 extern UINT MSIREG_OpenFeaturesKey(LPCWSTR szProduct, LPCWSTR szUserSid, MSIINSTALLCONTEXT context,
895                                    HKEY *key, BOOL create) DECLSPEC_HIDDEN;
896 extern UINT MSIREG_OpenUserPatchesKey(LPCWSTR szPatch, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
897 UINT MSIREG_OpenUserDataFeaturesKey(LPCWSTR szProduct, LPCWSTR szUserSid, MSIINSTALLCONTEXT context,
898                                     HKEY *key, BOOL create) DECLSPEC_HIDDEN;
899 extern UINT MSIREG_OpenUserComponentsKey(LPCWSTR szComponent, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
900 extern UINT MSIREG_OpenUserDataComponentKey(LPCWSTR szComponent, LPCWSTR szUserSid,
901                                             HKEY *key, BOOL create) DECLSPEC_HIDDEN;
902 extern UINT MSIREG_OpenPatchesKey(LPCWSTR szPatch, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
903 extern UINT MSIREG_OpenUserDataProductKey(LPCWSTR szProduct, MSIINSTALLCONTEXT dwContext,
904                                           LPCWSTR szUserSid, HKEY *key, BOOL create) DECLSPEC_HIDDEN;
905 extern UINT MSIREG_OpenUserDataPatchKey(LPCWSTR szPatch, MSIINSTALLCONTEXT dwContext,
906                                         HKEY *key, BOOL create) DECLSPEC_HIDDEN;
907 extern UINT MSIREG_OpenUserDataProductPatchesKey(LPCWSTR product, MSIINSTALLCONTEXT context,
908                                                  HKEY *key, BOOL create) DECLSPEC_HIDDEN;
909 extern UINT MSIREG_OpenInstallProps(LPCWSTR szProduct, MSIINSTALLCONTEXT dwContext,
910                                     LPCWSTR szUserSid, HKEY *key, BOOL create) DECLSPEC_HIDDEN;
911 extern UINT MSIREG_OpenUpgradeCodesKey(LPCWSTR szProduct, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
912 extern UINT MSIREG_OpenUserUpgradeCodesKey(LPCWSTR szProduct, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
913 extern UINT MSIREG_DeleteProductKey(LPCWSTR szProduct) DECLSPEC_HIDDEN;
914 extern UINT MSIREG_DeleteUserProductKey(LPCWSTR szProduct) DECLSPEC_HIDDEN;
915 extern UINT MSIREG_DeleteUserDataPatchKey(LPCWSTR patch, MSIINSTALLCONTEXT context) DECLSPEC_HIDDEN;
916 extern UINT MSIREG_DeleteUserDataProductKey(LPCWSTR, MSIINSTALLCONTEXT) DECLSPEC_HIDDEN;
917 extern UINT MSIREG_DeleteUserFeaturesKey(LPCWSTR szProduct) DECLSPEC_HIDDEN;
918 extern UINT MSIREG_DeleteUserDataComponentKey(LPCWSTR szComponent, LPCWSTR szUserSid) DECLSPEC_HIDDEN;
919 extern UINT MSIREG_DeleteUserUpgradeCodesKey(LPCWSTR szUpgradeCode) DECLSPEC_HIDDEN;
920 extern UINT MSIREG_DeleteUpgradeCodesKey(const WCHAR *) DECLSPEC_HIDDEN;
921 extern UINT MSIREG_DeleteClassesUpgradeCodesKey(LPCWSTR szUpgradeCode) DECLSPEC_HIDDEN;
922 extern UINT MSIREG_OpenClassesUpgradeCodesKey(LPCWSTR szUpgradeCode, HKEY* key, BOOL create) DECLSPEC_HIDDEN;
923 extern UINT MSIREG_DeleteLocalClassesProductKey(LPCWSTR szProductCode) DECLSPEC_HIDDEN;
924 extern UINT MSIREG_DeleteLocalClassesFeaturesKey(LPCWSTR szProductCode) DECLSPEC_HIDDEN;
925 extern UINT msi_locate_product(LPCWSTR szProduct, MSIINSTALLCONTEXT *context) DECLSPEC_HIDDEN;
926 extern LPWSTR msi_reg_get_val_str( HKEY hkey, LPCWSTR name ) DECLSPEC_HIDDEN;
927 extern BOOL msi_reg_get_val_dword( HKEY hkey, LPCWSTR name, DWORD *val) DECLSPEC_HIDDEN;
928 
929 extern DWORD msi_version_str_to_dword(LPCWSTR p) DECLSPEC_HIDDEN;
930 extern void msi_parse_version_string(LPCWSTR, PDWORD, PDWORD) DECLSPEC_HIDDEN;
931 extern VS_FIXEDFILEINFO *msi_get_disk_file_version(LPCWSTR) DECLSPEC_HIDDEN;
932 extern int msi_compare_file_versions(VS_FIXEDFILEINFO *, const WCHAR *) DECLSPEC_HIDDEN;
933 extern int msi_compare_font_versions(const WCHAR *, const WCHAR *) DECLSPEC_HIDDEN;
934 extern DWORD msi_get_disk_file_size(LPCWSTR) DECLSPEC_HIDDEN;
935 extern BOOL msi_file_hash_matches(MSIFILE *) DECLSPEC_HIDDEN;
936 extern UINT msi_get_filehash(const WCHAR *, MSIFILEHASHINFO *) DECLSPEC_HIDDEN;
937 
938 extern LONG msi_reg_set_val_str( HKEY hkey, LPCWSTR name, LPCWSTR value ) DECLSPEC_HIDDEN;
939 extern LONG msi_reg_set_val_multi_str( HKEY hkey, LPCWSTR name, LPCWSTR value ) DECLSPEC_HIDDEN;
940 extern LONG msi_reg_set_val_dword( HKEY hkey, LPCWSTR name, DWORD val ) DECLSPEC_HIDDEN;
941 extern LONG msi_reg_set_subkey_val( HKEY hkey, LPCWSTR path, LPCWSTR name, LPCWSTR val ) DECLSPEC_HIDDEN;
942 
943 /* msi dialog interface */
944 extern void msi_dialog_check_messages( HANDLE ) DECLSPEC_HIDDEN;
945 extern void msi_dialog_destroy( msi_dialog* ) DECLSPEC_HIDDEN;
946 extern void msi_dialog_unregister_class( void ) DECLSPEC_HIDDEN;
947 
948 /* summary information */
949 extern UINT msi_get_suminfo( IStorage *stg, UINT uiUpdateCount, MSISUMMARYINFO **si ) DECLSPEC_HIDDEN;
950 extern UINT msi_get_db_suminfo( MSIDATABASE *db, UINT uiUpdateCount, MSISUMMARYINFO **si ) DECLSPEC_HIDDEN;
951 extern LPWSTR msi_suminfo_dup_string( MSISUMMARYINFO *si, UINT uiProperty ) DECLSPEC_HIDDEN;
952 extern INT msi_suminfo_get_int32( MSISUMMARYINFO *si, UINT uiProperty ) DECLSPEC_HIDDEN;
953 extern LPWSTR msi_get_suminfo_product( IStorage *stg ) DECLSPEC_HIDDEN;
954 extern UINT msi_add_suminfo( MSIDATABASE *db, LPWSTR **records, int num_records, int num_columns ) DECLSPEC_HIDDEN;
955 extern UINT msi_export_suminfo( MSIDATABASE *db, HANDLE handle ) DECLSPEC_HIDDEN;
956 extern UINT msi_load_suminfo_properties( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
957 
958 /* undocumented functions */
959 UINT WINAPI MsiCreateAndVerifyInstallerDirectory( DWORD );
960 UINT WINAPI MsiDecomposeDescriptorW( LPCWSTR, LPWSTR, LPWSTR, LPWSTR, LPDWORD );
961 UINT WINAPI MsiDecomposeDescriptorA( LPCSTR, LPSTR, LPSTR, LPSTR, LPDWORD );
962 LANGID WINAPI MsiLoadStringW( MSIHANDLE, UINT, LPWSTR, int, LANGID );
963 LANGID WINAPI MsiLoadStringA( MSIHANDLE, UINT, LPSTR, int, LANGID );
964 
965 /* UI globals */
966 extern INSTALLUILEVEL gUILevel DECLSPEC_HIDDEN;
967 extern HWND gUIhwnd DECLSPEC_HIDDEN;
968 extern INSTALLUI_HANDLERA gUIHandlerA DECLSPEC_HIDDEN;
969 extern INSTALLUI_HANDLERW gUIHandlerW DECLSPEC_HIDDEN;
970 extern INSTALLUI_HANDLER_RECORD gUIHandlerRecord DECLSPEC_HIDDEN;
971 extern DWORD gUIFilter DECLSPEC_HIDDEN;
972 extern DWORD gUIFilterRecord DECLSPEC_HIDDEN;
973 extern LPVOID gUIContext DECLSPEC_HIDDEN;
974 extern LPVOID gUIContextRecord DECLSPEC_HIDDEN;
975 extern WCHAR *gszLogFile DECLSPEC_HIDDEN;
976 extern HINSTANCE msi_hInstance DECLSPEC_HIDDEN;
977 
978 /* action related functions */
979 extern UINT ACTION_PerformAction(MSIPACKAGE *package, const WCHAR *action, UINT script) DECLSPEC_HIDDEN;
980 extern void ACTION_FinishCustomActions( const MSIPACKAGE* package) DECLSPEC_HIDDEN;
981 extern UINT ACTION_CustomAction(MSIPACKAGE *, const WCHAR *, UINT) DECLSPEC_HIDDEN;
982 
983 /* actions in other modules */
984 extern UINT ACTION_AppSearch(MSIPACKAGE *package) DECLSPEC_HIDDEN;
985 extern UINT ACTION_CCPSearch(MSIPACKAGE *package) DECLSPEC_HIDDEN;
986 extern UINT ACTION_FindRelatedProducts(MSIPACKAGE *package) DECLSPEC_HIDDEN;
987 extern UINT ACTION_InstallFiles(MSIPACKAGE *package) DECLSPEC_HIDDEN;
988 extern UINT ACTION_PatchFiles( MSIPACKAGE *package ) DECLSPEC_HIDDEN;
989 extern UINT ACTION_RemoveFiles(MSIPACKAGE *package) DECLSPEC_HIDDEN;
990 extern UINT ACTION_MoveFiles(MSIPACKAGE *package) DECLSPEC_HIDDEN;
991 extern UINT ACTION_DuplicateFiles(MSIPACKAGE *package) DECLSPEC_HIDDEN;
992 extern UINT ACTION_RemoveDuplicateFiles(MSIPACKAGE *package) DECLSPEC_HIDDEN;
993 extern UINT ACTION_RegisterClassInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
994 extern UINT ACTION_RegisterProgIdInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
995 extern UINT ACTION_RegisterExtensionInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
996 extern UINT ACTION_RegisterMIMEInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
997 extern UINT ACTION_RegisterFonts(MSIPACKAGE *package) DECLSPEC_HIDDEN;
998 extern UINT ACTION_UnregisterClassInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
999 extern UINT ACTION_UnregisterExtensionInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1000 extern UINT ACTION_UnregisterFonts(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1001 extern UINT ACTION_UnregisterMIMEInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1002 extern UINT ACTION_UnregisterProgIdInfo(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1003 extern UINT ACTION_MsiPublishAssemblies(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1004 extern UINT ACTION_MsiUnpublishAssemblies(MSIPACKAGE *package) DECLSPEC_HIDDEN;
1005 
1006 /* Helpers */
1007 extern DWORD deformat_string(MSIPACKAGE *package, LPCWSTR ptr, WCHAR** data ) DECLSPEC_HIDDEN;
1008 extern WCHAR *msi_dup_record_field(MSIRECORD *row, INT index) DECLSPEC_HIDDEN;
1009 extern LPWSTR msi_dup_property( MSIDATABASE *db, LPCWSTR prop ) DECLSPEC_HIDDEN;
1010 extern UINT msi_set_property( MSIDATABASE *, const WCHAR *, const WCHAR *, int ) DECLSPEC_HIDDEN;
1011 extern UINT msi_get_property( MSIDATABASE *, LPCWSTR, LPWSTR, LPDWORD ) DECLSPEC_HIDDEN;
1012 extern int msi_get_property_int( MSIDATABASE *package, LPCWSTR prop, int def ) DECLSPEC_HIDDEN;
1013 extern WCHAR *msi_resolve_source_folder(MSIPACKAGE *package, const WCHAR *name, MSIFOLDER **folder) DECLSPEC_HIDDEN;
1014 extern void msi_resolve_target_folder(MSIPACKAGE *package, const WCHAR *name, BOOL load_prop) DECLSPEC_HIDDEN;
1015 extern WCHAR *msi_normalize_path(const WCHAR *) DECLSPEC_HIDDEN;
1016 extern WCHAR *msi_resolve_file_source(MSIPACKAGE *package, MSIFILE *file) DECLSPEC_HIDDEN;
1017 extern const WCHAR *msi_get_target_folder(MSIPACKAGE *package, const WCHAR *name) DECLSPEC_HIDDEN;
1018 extern void msi_reset_folders( MSIPACKAGE *package, BOOL source ) DECLSPEC_HIDDEN;
1019 extern MSICOMPONENT *msi_get_loaded_component(MSIPACKAGE *package, const WCHAR *Component) DECLSPEC_HIDDEN;
1020 extern MSIFEATURE *msi_get_loaded_feature(MSIPACKAGE *package, const WCHAR *Feature) DECLSPEC_HIDDEN;
1021 extern MSIFILE *msi_get_loaded_file(MSIPACKAGE *package, const WCHAR *file) DECLSPEC_HIDDEN;
1022 extern MSIFOLDER *msi_get_loaded_folder(MSIPACKAGE *package, const WCHAR *dir) DECLSPEC_HIDDEN;
1023 extern WCHAR *msi_create_temp_file(MSIDATABASE *db) DECLSPEC_HIDDEN;
1024 extern void msi_free_action_script(MSIPACKAGE *package, UINT script) DECLSPEC_HIDDEN;
1025 extern WCHAR *msi_build_icon_path(MSIPACKAGE *, const WCHAR *) DECLSPEC_HIDDEN;
1026 extern WCHAR *msi_build_directory_name(DWORD , ...) DECLSPEC_HIDDEN;
1027 extern BOOL msi_create_full_path(const WCHAR *path) DECLSPEC_HIDDEN;
1028 extern void msi_reduce_to_long_filename(WCHAR *) DECLSPEC_HIDDEN;
1029 extern WCHAR *msi_create_component_advertise_string(MSIPACKAGE *, MSICOMPONENT *, const WCHAR *) DECLSPEC_HIDDEN;
1030 extern void ACTION_UpdateComponentStates(MSIPACKAGE *package, MSIFEATURE *feature) DECLSPEC_HIDDEN;
1031 extern UINT msi_register_unique_action(MSIPACKAGE *, const WCHAR *) DECLSPEC_HIDDEN;
1032 extern BOOL msi_action_is_unique(const MSIPACKAGE *, const WCHAR *) DECLSPEC_HIDDEN;
1033 extern UINT msi_set_last_used_source(LPCWSTR product, LPCWSTR usersid,
1034                         MSIINSTALLCONTEXT context, DWORD options, LPCWSTR value) DECLSPEC_HIDDEN;
1035 extern UINT msi_create_empty_local_file(LPWSTR path, LPCWSTR suffix) DECLSPEC_HIDDEN;
1036 extern UINT msi_set_sourcedir_props(MSIPACKAGE *package, BOOL replace) DECLSPEC_HIDDEN;
1037 extern MSIASSEMBLY *msi_load_assembly(MSIPACKAGE *, MSICOMPONENT *) DECLSPEC_HIDDEN;
1038 extern UINT msi_install_assembly(MSIPACKAGE *, MSICOMPONENT *) DECLSPEC_HIDDEN;
1039 extern UINT msi_uninstall_assembly(MSIPACKAGE *, MSICOMPONENT *) DECLSPEC_HIDDEN;
1040 extern BOOL msi_init_assembly_caches(MSIPACKAGE *) DECLSPEC_HIDDEN;
1041 extern void msi_destroy_assembly_caches(MSIPACKAGE *) DECLSPEC_HIDDEN;
1042 extern BOOL msi_is_global_assembly(MSICOMPONENT *) DECLSPEC_HIDDEN;
1043 extern IAssemblyEnum *msi_create_assembly_enum(MSIPACKAGE *, const WCHAR *) DECLSPEC_HIDDEN;
1044 extern WCHAR *msi_get_assembly_path(MSIPACKAGE *, const WCHAR *) DECLSPEC_HIDDEN;
1045 extern WCHAR *msi_font_version_from_file(const WCHAR *) DECLSPEC_HIDDEN;
1046 extern WCHAR **msi_split_string(const WCHAR *, WCHAR) DECLSPEC_HIDDEN;
1047 extern UINT msi_set_original_database_property(MSIDATABASE *, const WCHAR *) DECLSPEC_HIDDEN;
1048 extern WCHAR *msi_get_error_message(MSIDATABASE *, int) DECLSPEC_HIDDEN;
1049 
1050 /* media */
1051 
1052 typedef BOOL (*PMSICABEXTRACTCB)(MSIPACKAGE *, LPCWSTR, DWORD, LPWSTR *, DWORD *, PVOID);
1053 
1054 #define MSICABEXTRACT_BEGINEXTRACT  0x01
1055 #define MSICABEXTRACT_FILEEXTRACTED 0x02
1056 
1057 typedef struct
1058 {
1059     MSIPACKAGE* package;
1060     MSIMEDIAINFO *mi;
1061     PMSICABEXTRACTCB cb;
1062     LPWSTR curfile;
1063     PVOID user;
1064 } MSICABDATA;
1065 
1066 extern UINT ready_media(MSIPACKAGE *package, BOOL compressed, MSIMEDIAINFO *mi) DECLSPEC_HIDDEN;
1067 extern UINT msi_load_media_info(MSIPACKAGE *package, UINT Sequence, MSIMEDIAINFO *mi) DECLSPEC_HIDDEN;
1068 extern void msi_free_media_info(MSIMEDIAINFO *mi) DECLSPEC_HIDDEN;
1069 extern BOOL msi_cabextract(MSIPACKAGE* package, MSIMEDIAINFO *mi, LPVOID data) DECLSPEC_HIDDEN;
1070 extern UINT msi_add_cabinet_stream(MSIPACKAGE *, UINT, IStorage *, const WCHAR *) DECLSPEC_HIDDEN;
1071 
1072 /* control event stuff */
1073 extern void msi_event_fire(MSIPACKAGE *, const WCHAR *, MSIRECORD *) DECLSPEC_HIDDEN;
1074 extern void msi_event_cleanup_all_subscriptions( MSIPACKAGE * ) DECLSPEC_HIDDEN;
1075 
1076 /* OLE automation */
1077 typedef enum tid_t {
1078     Database_tid,
1079     Installer_tid,
1080     Record_tid,
1081     Session_tid,
1082     StringList_tid,
1083     SummaryInfo_tid,
1084     View_tid,
1085     LAST_tid
1086 } tid_t;
1087 
1088 extern HRESULT create_msiserver(IUnknown *pOuter, LPVOID *ppObj) DECLSPEC_HIDDEN;
1089 extern HRESULT create_session(MSIHANDLE msiHandle, IDispatch *pInstaller, IDispatch **pDispatch) DECLSPEC_HIDDEN;
1090 extern HRESULT get_typeinfo(tid_t tid, ITypeInfo **ti) DECLSPEC_HIDDEN;
1091 extern void release_typelib(void) DECLSPEC_HIDDEN;
1092 
1093 /* Scripting */
1094 extern DWORD call_script(MSIHANDLE hPackage, INT type, LPCWSTR script, LPCWSTR function, LPCWSTR action) DECLSPEC_HIDDEN;
1095 
1096 /* User interface messages from the actions */
1097 extern void msi_ui_progress(MSIPACKAGE *, int, int, int, int) DECLSPEC_HIDDEN;
1098 
1099 /* common strings */
1100 static const WCHAR szSourceDir[] = {'S','o','u','r','c','e','D','i','r',0};
1101 static const WCHAR szSOURCEDIR[] = {'S','O','U','R','C','E','D','I','R',0};
1102 static const WCHAR szRootDrive[] = {'R','O','O','T','D','R','I','V','E',0};
1103 static const WCHAR szTargetDir[] = {'T','A','R','G','E','T','D','I','R',0};
1104 static const WCHAR szLocalSid[] = {'S','-','1','-','5','-','1','8',0};
1105 static const WCHAR szAllSid[] = {'S','-','1','-','1','-','0',0};
1106 static const WCHAR szEmpty[] = {0};
1107 static const WCHAR szAll[] = {'A','L','L',0};
1108 static const WCHAR szOne[] = {'1',0};
1109 static const WCHAR szZero[] = {'0',0};
1110 static const WCHAR szSpace[] = {' ',0};
1111 static const WCHAR szBackSlash[] = {'\\',0};
1112 static const WCHAR szForwardSlash[] = {'/',0};
1113 static const WCHAR szDot[] = {'.',0};
1114 static const WCHAR szDotDot[] = {'.','.',0};
1115 static const WCHAR szSemiColon[] = {';',0};
1116 static const WCHAR szPreselected[] = {'P','r','e','s','e','l','e','c','t','e','d',0};
1117 static const WCHAR szPatches[] = {'P','a','t','c','h','e','s',0};
1118 static const WCHAR szState[] = {'S','t','a','t','e',0};
1119 static const WCHAR szMsi[] = {'m','s','i',0};
1120 static const WCHAR szPatch[] = {'P','A','T','C','H',0};
1121 static const WCHAR szSourceList[] = {'S','o','u','r','c','e','L','i','s','t',0};
1122 static const WCHAR szInstalled[] = {'I','n','s','t','a','l','l','e','d',0};
1123 static const WCHAR szReinstall[] = {'R','E','I','N','S','T','A','L','L',0};
1124 static const WCHAR szReinstallMode[] = {'R','E','I','N','S','T','A','L','L','M','O','D','E',0};
1125 static const WCHAR szRemove[] = {'R','E','M','O','V','E',0};
1126 static const WCHAR szUserSID[] = {'U','s','e','r','S','I','D',0};
1127 static const WCHAR szProductCode[] = {'P','r','o','d','u','c','t','C','o','d','e',0};
1128 static const WCHAR szRegisterClassInfo[] = {'R','e','g','i','s','t','e','r','C','l','a','s','s','I','n','f','o',0};
1129 static const WCHAR szRegisterProgIdInfo[] = {'R','e','g','i','s','t','e','r','P','r','o','g','I','d','I','n','f','o',0};
1130 static const WCHAR szRegisterExtensionInfo[] = {'R','e','g','i','s','t','e','r','E','x','t','e','n','s','i','o','n','I','n','f','o',0};
1131 static const WCHAR szRegisterMIMEInfo[] = {'R','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
1132 static const WCHAR szDuplicateFiles[] = {'D','u','p','l','i','c','a','t','e','F','i','l','e','s',0};
1133 static const WCHAR szRemoveDuplicateFiles[] = {'R','e','m','o','v','e','D','u','p','l','i','c','a','t','e','F','i','l','e','s',0};
1134 static const WCHAR szInstallFiles[] = {'I','n','s','t','a','l','l','F','i','l','e','s',0};
1135 static const WCHAR szPatchFiles[] = {'P','a','t','c','h','F','i','l','e','s',0};
1136 static const WCHAR szRemoveFiles[] = {'R','e','m','o','v','e','F','i','l','e','s',0};
1137 static const WCHAR szFindRelatedProducts[] = {'F','i','n','d','R','e','l','a','t','e','d','P','r','o','d','u','c','t','s',0};
1138 static const WCHAR szAllUsers[] = {'A','L','L','U','S','E','R','S',0};
1139 static const WCHAR szCustomActionData[] = {'C','u','s','t','o','m','A','c','t','i','o','n','D','a','t','a',0};
1140 static const WCHAR szUILevel[] = {'U','I','L','e','v','e','l',0};
1141 static const WCHAR szProductID[] = {'P','r','o','d','u','c','t','I','D',0};
1142 static const WCHAR szPIDTemplate[] = {'P','I','D','T','e','m','p','l','a','t','e',0};
1143 static const WCHAR szPIDKEY[] = {'P','I','D','K','E','Y',0};
1144 static const WCHAR szTYPELIB[] = {'T','Y','P','E','L','I','B',0};
1145 static const WCHAR szSumInfo[] = {5 ,'S','u','m','m','a','r','y','I','n','f','o','r','m','a','t','i','o','n',0};
1146 static const WCHAR szHCR[] = {'H','K','E','Y','_','C','L','A','S','S','E','S','_','R','O','O','T','\\',0};
1147 static const WCHAR szHCU[] = {'H','K','E','Y','_','C','U','R','R','E','N','T','_','U','S','E','R','\\',0};
1148 static const WCHAR szHLM[] = {'H','K','E','Y','_','L','O','C','A','L','_','M','A','C','H','I','N','E','\\',0};
1149 static const WCHAR szHU[] = {'H','K','E','Y','_','U','S','E','R','S','\\',0};
1150 static const WCHAR szWindowsFolder[] = {'W','i','n','d','o','w','s','F','o','l','d','e','r',0};
1151 static const WCHAR szAppSearch[] = {'A','p','p','S','e','a','r','c','h',0};
1152 static const WCHAR szMoveFiles[] = {'M','o','v','e','F','i','l','e','s',0};
1153 static const WCHAR szCCPSearch[] = {'C','C','P','S','e','a','r','c','h',0};
1154 static const WCHAR szUnregisterClassInfo[] = {'U','n','r','e','g','i','s','t','e','r','C','l','a','s','s','I','n','f','o',0};
1155 static const WCHAR szUnregisterExtensionInfo[] = {'U','n','r','e','g','i','s','t','e','r','E','x','t','e','n','s','i','o','n','I','n','f','o',0};
1156 static const WCHAR szUnregisterMIMEInfo[] = {'U','n','r','e','g','i','s','t','e','r','M','I','M','E','I','n','f','o',0};
1157 static const WCHAR szUnregisterProgIdInfo[] = {'U','n','r','e','g','i','s','t','e','r','P','r','o','g','I','d','I','n','f','o',0};
1158 static const WCHAR szRegisterFonts[] = {'R','e','g','i','s','t','e','r','F','o','n','t','s',0};
1159 static const WCHAR szUnregisterFonts[] = {'U','n','r','e','g','i','s','t','e','r','F','o','n','t','s',0};
1160 static const WCHAR szCLSID[] = {'C','L','S','I','D',0};
1161 static const WCHAR szProgID[] = {'P','r','o','g','I','D',0};
1162 static const WCHAR szVIProgID[] = {'V','e','r','s','i','o','n','I','n','d','e','p','e','n','d','e','n','t','P','r','o','g','I','D',0};
1163 static const WCHAR szAppID[] = {'A','p','p','I','D',0};
1164 static const WCHAR szDefaultIcon[] = {'D','e','f','a','u','l','t','I','c','o','n',0};
1165 static const WCHAR szInprocHandler[] = {'I','n','p','r','o','c','H','a','n','d','l','e','r',0};
1166 static const WCHAR szInprocHandler32[] = {'I','n','p','r','o','c','H','a','n','d','l','e','r','3','2',0};
1167 static const WCHAR szMIMEDatabase[] = {'M','I','M','E','\\','D','a','t','a','b','a','s','e','\\','C','o','n','t','e','n','t',' ','T','y','p','e','\\',0};
1168 static const WCHAR szLocalPackage[] = {'L','o','c','a','l','P','a','c','k','a','g','e',0};
1169 static const WCHAR szOriginalDatabase[] = {'O','r','i','g','i','n','a','l','D','a','t','a','b','a','s','e',0};
1170 static const WCHAR szUpgradeCode[] = {'U','p','g','r','a','d','e','C','o','d','e',0};
1171 static const WCHAR szAdminUser[] = {'A','d','m','i','n','U','s','e','r',0};
1172 static const WCHAR szIntel[] = {'I','n','t','e','l',0};
1173 static const WCHAR szIntel64[] = {'I','n','t','e','l','6','4',0};
1174 static const WCHAR szX64[] = {'x','6','4',0};
1175 static const WCHAR szAMD64[] = {'A','M','D','6','4',0};
1176 static const WCHAR szARM[] = {'A','r','m',0};
1177 static const WCHAR szWow6432NodeCLSID[] = {'W','o','w','6','4','3','2','N','o','d','e','\\','C','L','S','I','D',0};
1178 static const WCHAR szStreams[] = {'_','S','t','r','e','a','m','s',0};
1179 static const WCHAR szStorages[] = {'_','S','t','o','r','a','g','e','s',0};
1180 static const WCHAR szMsiPublishAssemblies[] = {'M','s','i','P','u','b','l','i','s','h','A','s','s','e','m','b','l','i','e','s',0};
1181 static const WCHAR szCostingComplete[] = {'C','o','s','t','i','n','g','C','o','m','p','l','e','t','e',0};
1182 static const WCHAR szTempFolder[] = {'T','e','m','p','F','o','l','d','e','r',0};
1183 static const WCHAR szDatabase[] = {'D','A','T','A','B','A','S','E',0};
1184 static const WCHAR szCRoot[] = {'C',':','\\',0};
1185 static const WCHAR szProductLanguage[] = {'P','r','o','d','u','c','t','L','a','n','g','u','a','g','e',0};
1186 static const WCHAR szProductVersion[] = {'P','r','o','d','u','c','t','V','e','r','s','i','o','n',0};
1187 static const WCHAR szWindowsInstaller[] = {'W','i','n','d','o','w','s','I','n','s','t','a','l','l','e','r',0};
1188 static const WCHAR szStringData[] = {'_','S','t','r','i','n','g','D','a','t','a',0};
1189 static const WCHAR szStringPool[] = {'_','S','t','r','i','n','g','P','o','o','l',0};
1190 static const WCHAR szInstallLevel[] = {'I','N','S','T','A','L','L','L','E','V','E','L',0};
1191 static const WCHAR szCostInitialize[] = {'C','o','s','t','I','n','i','t','i','a','l','i','z','e',0};
1192 static const WCHAR szAppDataFolder[] = {'A','p','p','D','a','t','a','F','o','l','d','e','r',0};
1193 static const WCHAR szRollbackDisabled[] = {'R','o','l','l','b','a','c','k','D','i','s','a','b','l','e','d',0};
1194 static const WCHAR szName[] = {'N','a','m','e',0};
1195 static const WCHAR szData[] = {'D','a','t','a',0};
1196 static const WCHAR szLangResource[] = {'\\','V','a','r','F','i','l','e','I','n','f','o','\\','T','r','a','n','s','l','a','t','i','o','n',0};
1197 static const WCHAR szInstallLocation[] = {'I','n','s','t','a','l','l','L','o','c','a','t','i','o','n',0};
1198 static const WCHAR szProperty[] = {'P','r','o','p','e','r','t','y',0};
1199 static const WCHAR szUninstallable[] = {'U','n','i','n','s','t','a','l','l','a','b','l','e',0};
1200 static const WCHAR szEXECUTEACTION[] = {'E','X','E','C','U','T','E','A','C','T','I','O','N',0};
1201 
1202 /* memory allocation macro functions */
1203 static void *msi_alloc( size_t len ) __WINE_ALLOC_SIZE(1);
1204 static inline void *msi_alloc( size_t len )
1205 {
1206     return HeapAlloc( GetProcessHeap(), 0, len );
1207 }
1208 
1209 static void *msi_alloc_zero( size_t len ) __WINE_ALLOC_SIZE(1);
1210 static inline void *msi_alloc_zero( size_t len )
1211 {
1212     return HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, len );
1213 }
1214 
1215 static void *msi_realloc( void *mem, size_t len ) __WINE_ALLOC_SIZE(2);
1216 static inline void *msi_realloc( void *mem, size_t len )
1217 {
1218     return HeapReAlloc( GetProcessHeap(), 0, mem, len );
1219 }
1220 
1221 static void *msi_realloc_zero( void *mem, size_t len ) __WINE_ALLOC_SIZE(2);
1222 static inline void *msi_realloc_zero( void *mem, size_t len )
1223 {
1224     return HeapReAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, mem, len );
1225 }
1226 
1227 static inline BOOL msi_free( void *mem )
1228 {
1229     return HeapFree( GetProcessHeap(), 0, mem );
1230 }
1231 
1232 static inline char *strdupWtoA( LPCWSTR str )
1233 {
1234     LPSTR ret = NULL;
1235     DWORD len;
1236 
1237     if (!str) return ret;
1238     len = WideCharToMultiByte( CP_ACP, 0, str, -1, NULL, 0, NULL, NULL);
1239     ret = msi_alloc( len );
1240     if (ret)
1241         WideCharToMultiByte( CP_ACP, 0, str, -1, ret, len, NULL, NULL );
1242     return ret;
1243 }
1244 
1245 static inline LPWSTR strdupAtoW( LPCSTR str )
1246 {
1247     LPWSTR ret = NULL;
1248     DWORD len;
1249 
1250     if (!str) return ret;
1251     len = MultiByteToWideChar( CP_ACP, 0, str, -1, NULL, 0 );
1252     ret = msi_alloc( len * sizeof(WCHAR) );
1253     if (ret)
1254         MultiByteToWideChar( CP_ACP, 0, str, -1, ret, len );
1255     return ret;
1256 }
1257 
1258 static inline LPWSTR strdupW( LPCWSTR src )
1259 {
1260     LPWSTR dest;
1261     if (!src) return NULL;
1262     dest = msi_alloc( (lstrlenW(src)+1)*sizeof(WCHAR) );
1263     if (dest)
1264         lstrcpyW(dest, src);
1265     return dest;
1266 }
1267 
1268 #endif /* __WINE_MSI_PRIVATE__ */
1269