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