/* * Implementation of the Microsoft Installer (msi.dll) * * Copyright 2002, 2005 Mike McCormack for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #define COBJMACROS #define NONAMELESSUNION #include "windef.h" #include "winbase.h" #include "winnls.h" #include "shlwapi.h" #include "debug.h" #include "unicode.h" #include "msiquery.h" #include "msipriv.h" #include "objidl.h" #include "propvarutil.h" #include "msiserver.h" const CLSID FMTID_SummaryInformation = { 0xf29f85e0, 0x4ff9, 0x1068, {0xab, 0x91, 0x08, 0x00, 0x2b, 0x27, 0xb3, 0xd9}}; #include "pshpack1.h" typedef struct { WORD wByteOrder; WORD wFormat; DWORD dwOSVer; CLSID clsID; DWORD reserved; } PROPERTYSETHEADER; typedef struct { FMTID fmtid; DWORD dwOffset; } FORMATIDOFFSET; typedef struct { DWORD cbSection; DWORD cProperties; } PROPERTYSECTIONHEADER; typedef struct { DWORD propid; DWORD dwOffset; } PROPERTYIDOFFSET; typedef struct { DWORD type; union { INT i4; SHORT i2; FILETIME ft; struct { DWORD len; BYTE str[1]; } str; } u; } PROPERTY_DATA; #include "poppack.h" static HRESULT (WINAPI *pPropVariantChangeType) (PROPVARIANT *ppropvarDest, REFPROPVARIANT propvarSrc, PROPVAR_CHANGE_FLAGS flags, VARTYPE vt); #define SECT_HDR_SIZE (sizeof(PROPERTYSECTIONHEADER)) static void free_prop( PROPVARIANT *prop ) { if (prop->vt == VT_LPSTR ) msi_free( prop->pszVal ); prop->vt = VT_EMPTY; } static void MSI_CloseSummaryInfo( MSIOBJECTHDR *arg ) { MSISUMMARYINFO *si = (MSISUMMARYINFO *) arg; DWORD i; for( i = 0; i < MSI_MAX_PROPS; i++ ) free_prop( &si->property[i] ); IStorage_Release( si->storage ); } static UINT get_type( UINT uiProperty ) { switch( uiProperty ) { case PID_CODEPAGE: return VT_I2; case PID_SUBJECT: case PID_AUTHOR: case PID_KEYWORDS: case PID_COMMENTS: case PID_TEMPLATE: case PID_LASTAUTHOR: case PID_REVNUMBER: case PID_APPNAME: case PID_TITLE: return VT_LPSTR; case PID_LASTPRINTED: case PID_CREATE_DTM: case PID_LASTSAVE_DTM: return VT_FILETIME; case PID_WORDCOUNT: case PID_CHARCOUNT: case PID_SECURITY: case PID_PAGECOUNT: return VT_I4; } return VT_EMPTY; } static UINT get_property_count( const PROPVARIANT *property ) { UINT i, n = 0; if( !property ) return n; for( i = 0; i < MSI_MAX_PROPS; i++ ) if( property[i].vt != VT_EMPTY ) n++; return n; } static UINT propvar_changetype(PROPVARIANT *changed, PROPVARIANT *property, VARTYPE vt) { HRESULT hr; HMODULE propsys = LoadLibraryA("propsys.dll"); pPropVariantChangeType = (void *)GetProcAddress(propsys, "PropVariantChangeType"); if (!pPropVariantChangeType) { ERR("PropVariantChangeType function missing!\n"); return ERROR_FUNCTION_FAILED; } hr = pPropVariantChangeType(changed, property, 0, vt); return (hr == S_OK) ? ERROR_SUCCESS : ERROR_FUNCTION_FAILED; } /* FIXME: doesn't deal with endian conversion */ static void read_properties_from_data( PROPVARIANT *prop, LPBYTE data, DWORD sz ) { UINT type; DWORD i, size; PROPERTY_DATA *propdata; PROPVARIANT property, *ptr; PROPVARIANT changed; PROPERTYIDOFFSET *idofs; PROPERTYSECTIONHEADER *section_hdr; section_hdr = (PROPERTYSECTIONHEADER*) &data[0]; idofs = (PROPERTYIDOFFSET*) &data[SECT_HDR_SIZE]; /* now set all the properties */ for( i = 0; i < section_hdr->cProperties; i++ ) { if( idofs[i].propid >= MSI_MAX_PROPS ) { ERR("Unknown property ID %d\n", idofs[i].propid ); break; } type = get_type( idofs[i].propid ); if( type == VT_EMPTY ) { ERR("propid %d has unknown type\n", idofs[i].propid); break; } propdata = (PROPERTY_DATA*) &data[ idofs[i].dwOffset ]; /* check we don't run off the end of the data */ size = sz - idofs[i].dwOffset - sizeof(DWORD); if( sizeof(DWORD) > size || ( propdata->type == VT_FILETIME && sizeof(FILETIME) > size ) || ( propdata->type == VT_LPSTR && (propdata->u.str.len + sizeof(DWORD)) > size ) ) { ERR("not enough data\n"); break; } property.vt = propdata->type; if( propdata->type == VT_LPSTR ) { LPSTR str = msi_alloc( propdata->u.str.len ); memcpy( str, propdata->u.str.str, propdata->u.str.len ); str[ propdata->u.str.len - 1 ] = 0; property.pszVal = str; } else if( propdata->type == VT_FILETIME ) property.filetime = propdata->u.ft; else if( propdata->type == VT_I2 ) property.iVal = propdata->u.i2; else if( propdata->type == VT_I4 ) property.lVal = propdata->u.i4; /* check the type is the same as we expect */ if( type != propdata->type ) { propvar_changetype(&changed, &property, type); ptr = &changed; } else ptr = &property; prop[ idofs[i].propid ] = *ptr; } } static UINT load_summary_info( MSISUMMARYINFO *si, IStream *stm ) { UINT ret = ERROR_FUNCTION_FAILED; PROPERTYSETHEADER set_hdr; FORMATIDOFFSET format_hdr; PROPERTYSECTIONHEADER section_hdr; LPBYTE data = NULL; LARGE_INTEGER ofs; ULONG count, sz; HRESULT r; TRACE("%p %p\n", si, stm); /* read the header */ sz = sizeof set_hdr; r = IStream_Read( stm, &set_hdr, sz, &count ); if( FAILED(r) || count != sz ) return ret; if( set_hdr.wByteOrder != 0xfffe ) { ERR("property set not big-endian %04X\n", set_hdr.wByteOrder); return ret; } sz = sizeof format_hdr; r = IStream_Read( stm, &format_hdr, sz, &count ); if( FAILED(r) || count != sz ) return ret; /* check the format id is correct */ if( !IsEqualGUID( &FMTID_SummaryInformation, &format_hdr.fmtid ) ) return ret; /* seek to the location of the section */ ofs.QuadPart = format_hdr.dwOffset; r = IStream_Seek( stm, ofs, STREAM_SEEK_SET, NULL ); if( FAILED(r) ) return ret; /* read the section itself */ sz = SECT_HDR_SIZE; r = IStream_Read( stm, §ion_hdr, sz, &count ); if( FAILED(r) || count != sz ) return ret; if( section_hdr.cProperties > MSI_MAX_PROPS ) { ERR("too many properties %d\n", section_hdr.cProperties); return ret; } data = msi_alloc( section_hdr.cbSection); if( !data ) return ret; memcpy( data, §ion_hdr, SECT_HDR_SIZE ); /* read all the data in one go */ sz = section_hdr.cbSection - SECT_HDR_SIZE; r = IStream_Read( stm, &data[ SECT_HDR_SIZE ], sz, &count ); if( SUCCEEDED(r) && count == sz ) read_properties_from_data( si->property, data, sz + SECT_HDR_SIZE ); else ERR("failed to read properties %d %d\n", count, sz); msi_free( data ); return ret; } static DWORD write_dword( LPBYTE data, DWORD ofs, DWORD val ) { if( data ) { data[ofs++] = val&0xff; data[ofs++] = (val>>8)&0xff; data[ofs++] = (val>>16)&0xff; data[ofs++] = (val>>24)&0xff; } return 4; } static DWORD write_filetime( LPBYTE data, DWORD ofs, const FILETIME *ft ) { write_dword( data, ofs, ft->dwLowDateTime ); write_dword( data, ofs + 4, ft->dwHighDateTime ); return 8; } static DWORD write_string( LPBYTE data, DWORD ofs, LPCSTR str ) { DWORD len = lstrlenA( str ) + 1; write_dword( data, ofs, len ); if( data ) memcpy( &data[ofs + 4], str, len ); return (7 + len) & ~3; } static UINT write_property_to_data( const PROPVARIANT *prop, LPBYTE data ) { DWORD sz = 0; if( prop->vt == VT_EMPTY ) return sz; /* add the type */ sz += write_dword( data, sz, prop->vt ); switch( prop->vt ) { case VT_I2: sz += write_dword( data, sz, prop->iVal ); break; case VT_I4: sz += write_dword( data, sz, prop->lVal ); break; case VT_FILETIME: sz += write_filetime( data, sz, &prop->filetime ); break; case VT_LPSTR: sz += write_string( data, sz, prop->pszVal ); break; } return sz; } static UINT save_summary_info( const MSISUMMARYINFO * si, IStream *stm ) { UINT ret = ERROR_FUNCTION_FAILED; PROPERTYSETHEADER set_hdr; FORMATIDOFFSET format_hdr; PROPERTYSECTIONHEADER section_hdr; PROPERTYIDOFFSET idofs[MSI_MAX_PROPS]; LPBYTE data = NULL; ULONG count, sz; HRESULT r; int i; /* write the header */ sz = sizeof set_hdr; memset( &set_hdr, 0, sz ); set_hdr.wByteOrder = 0xfffe; set_hdr.wFormat = 0; set_hdr.dwOSVer = 0x00020005; /* build 5, platform id 2 */ /* set_hdr.clsID is {00000000-0000-0000-0000-000000000000} */ set_hdr.reserved = 1; r = IStream_Write( stm, &set_hdr, sz, &count ); if( FAILED(r) || count != sz ) return ret; /* write the format header */ sz = sizeof format_hdr; format_hdr.fmtid = FMTID_SummaryInformation; format_hdr.dwOffset = sizeof format_hdr + sizeof set_hdr; r = IStream_Write( stm, &format_hdr, sz, &count ); if( FAILED(r) || count != sz ) return ret; /* add up how much space the data will take and calculate the offsets */ section_hdr.cbSection = sizeof section_hdr; section_hdr.cbSection += (get_property_count( si->property ) * sizeof idofs[0]); section_hdr.cProperties = 0; for( i = 0; i < MSI_MAX_PROPS; i++ ) { sz = write_property_to_data( &si->property[i], NULL ); if( !sz ) continue; idofs[ section_hdr.cProperties ].propid = i; idofs[ section_hdr.cProperties ].dwOffset = section_hdr.cbSection; section_hdr.cProperties++; section_hdr.cbSection += sz; } data = msi_alloc_zero( section_hdr.cbSection ); sz = 0; memcpy( &data[sz], §ion_hdr, sizeof section_hdr ); sz += sizeof section_hdr; memcpy( &data[sz], idofs, section_hdr.cProperties * sizeof idofs[0] ); sz += section_hdr.cProperties * sizeof idofs[0]; /* write out the data */ for( i = 0; i < MSI_MAX_PROPS; i++ ) sz += write_property_to_data( &si->property[i], &data[sz] ); r = IStream_Write( stm, data, sz, &count ); msi_free( data ); if( FAILED(r) || count != sz ) return ret; return ERROR_SUCCESS; } MSISUMMARYINFO *MSI_GetSummaryInformationW( IStorage *stg, UINT uiUpdateCount ) { IStream *stm = NULL; MSISUMMARYINFO *si; DWORD grfMode; HRESULT r; TRACE("%p %d\n", stg, uiUpdateCount ); si = alloc_msiobject( MSIHANDLETYPE_SUMMARYINFO, sizeof (MSISUMMARYINFO), MSI_CloseSummaryInfo ); if( !si ) return si; si->update_count = uiUpdateCount; IStorage_AddRef( stg ); si->storage = stg; /* read the stream... if we fail, we'll start with an empty property set */ grfMode = STGM_READ | STGM_SHARE_EXCLUSIVE; r = IStorage_OpenStream( si->storage, szSumInfo, 0, grfMode, 0, &stm ); if( SUCCEEDED(r) ) { load_summary_info( si, stm ); IStream_Release( stm ); } return si; } UINT WINAPI MsiGetSummaryInformationW( MSIHANDLE hDatabase, LPCWSTR szDatabase, UINT uiUpdateCount, MSIHANDLE *pHandle ) { MSISUMMARYINFO *si; MSIDATABASE *db; UINT ret = ERROR_FUNCTION_FAILED; TRACE("%d %s %d %p\n", hDatabase, debugstr_w(szDatabase), uiUpdateCount, pHandle); if( !pHandle ) return ERROR_INVALID_PARAMETER; if( szDatabase && szDatabase[0] ) { LPCWSTR persist = uiUpdateCount ? MSIDBOPEN_TRANSACT : MSIDBOPEN_READONLY; ret = MSI_OpenDatabaseW( szDatabase, persist, &db ); if( ret != ERROR_SUCCESS ) return ret; } else { db = msihandle2msiinfo( hDatabase, MSIHANDLETYPE_DATABASE ); if( !db ) return ERROR_INVALID_HANDLE; } si = MSI_GetSummaryInformationW( db->storage, uiUpdateCount ); if (si) { *pHandle = alloc_msihandle( &si->hdr ); if( *pHandle ) ret = ERROR_SUCCESS; else ret = ERROR_NOT_ENOUGH_MEMORY; msiobj_release( &si->hdr ); } msiobj_release( &db->hdr ); return ret; } UINT WINAPI MsiGetSummaryInformationA(MSIHANDLE hDatabase, LPCSTR szDatabase, UINT uiUpdateCount, MSIHANDLE *pHandle) { LPWSTR szwDatabase = NULL; UINT ret; TRACE("%d %s %d %p\n", hDatabase, debugstr_a(szDatabase), uiUpdateCount, pHandle); if( szDatabase ) { szwDatabase = strdupAtoW( szDatabase ); if( !szwDatabase ) return ERROR_FUNCTION_FAILED; } ret = MsiGetSummaryInformationW(hDatabase, szwDatabase, uiUpdateCount, pHandle); msi_free( szwDatabase ); return ret; } UINT WINAPI MsiSummaryInfoGetPropertyCount(MSIHANDLE hSummaryInfo, PUINT pCount) { MSISUMMARYINFO *si; TRACE("%d %p\n", hSummaryInfo, pCount); si = msihandle2msiinfo( hSummaryInfo, MSIHANDLETYPE_SUMMARYINFO ); if( !si ) return ERROR_INVALID_HANDLE; if( pCount ) *pCount = get_property_count( si->property ); msiobj_release( &si->hdr ); return ERROR_SUCCESS; } static UINT get_prop( MSIHANDLE handle, UINT uiProperty, UINT *puiDataType, INT *piValue, FILETIME *pftValue, awstring *str, DWORD *pcchValueBuf) { MSISUMMARYINFO *si; PROPVARIANT *prop; UINT ret = ERROR_SUCCESS; TRACE("%d %d %p %p %p %p %p\n", handle, uiProperty, puiDataType, piValue, pftValue, str, pcchValueBuf); if ( uiProperty >= MSI_MAX_PROPS ) { if (puiDataType) *puiDataType = VT_EMPTY; return ERROR_UNKNOWN_PROPERTY; } si = msihandle2msiinfo( handle, MSIHANDLETYPE_SUMMARYINFO ); if( !si ) return ERROR_INVALID_HANDLE; prop = &si->property[uiProperty]; if( puiDataType ) *puiDataType = prop->vt; switch( prop->vt ) { case VT_I2: if( piValue ) *piValue = prop->iVal; break; case VT_I4: if( piValue ) *piValue = prop->lVal; break; case VT_LPSTR: if( pcchValueBuf ) { DWORD len = 0; if( str->unicode ) { len = MultiByteToWideChar( CP_ACP, 0, prop->pszVal, -1, NULL, 0 ) - 1; MultiByteToWideChar( CP_ACP, 0, prop->pszVal, -1, str->str.w, *pcchValueBuf ); } else { len = lstrlenA( prop->pszVal ); if( str->str.a ) lstrcpynA(str->str.a, prop->pszVal, *pcchValueBuf ); } if (len >= *pcchValueBuf) ret = ERROR_MORE_DATA; *pcchValueBuf = len; } break; case VT_FILETIME: if( pftValue ) *pftValue = prop->filetime; break; case VT_EMPTY: break; default: FIXME("Unknown property variant type\n"); break; } msiobj_release( &si->hdr ); return ret; } LPWSTR msi_suminfo_dup_string( MSISUMMARYINFO *si, UINT uiProperty ) { PROPVARIANT *prop; if ( uiProperty >= MSI_MAX_PROPS ) return NULL; prop = &si->property[uiProperty]; if( prop->vt != VT_LPSTR ) return NULL; return strdupAtoW( prop->pszVal ); } INT msi_suminfo_get_int32( MSISUMMARYINFO *si, UINT uiProperty ) { PROPVARIANT *prop; if ( uiProperty >= MSI_MAX_PROPS ) return -1; prop = &si->property[uiProperty]; if( prop->vt != VT_I4 ) return -1; return prop->lVal; } LPWSTR msi_get_suminfo_product( IStorage *stg ) { MSISUMMARYINFO *si; LPWSTR prod; si = MSI_GetSummaryInformationW( stg, 0 ); if (!si) { ERR("no summary information!\n"); return NULL; } prod = msi_suminfo_dup_string( si, PID_REVNUMBER ); msiobj_release( &si->hdr ); return prod; } UINT WINAPI MsiSummaryInfoGetPropertyA( MSIHANDLE handle, UINT uiProperty, PUINT puiDataType, LPINT piValue, FILETIME *pftValue, LPSTR szValueBuf, LPDWORD pcchValueBuf) { awstring str; TRACE("%d %d %p %p %p %p %p\n", handle, uiProperty, puiDataType, piValue, pftValue, szValueBuf, pcchValueBuf ); str.unicode = FALSE; str.str.a = szValueBuf; return get_prop( handle, uiProperty, puiDataType, piValue, pftValue, &str, pcchValueBuf ); } UINT WINAPI MsiSummaryInfoGetPropertyW( MSIHANDLE handle, UINT uiProperty, PUINT puiDataType, LPINT piValue, FILETIME *pftValue, LPWSTR szValueBuf, LPDWORD pcchValueBuf) { awstring str; TRACE("%d %d %p %p %p %p %p\n", handle, uiProperty, puiDataType, piValue, pftValue, szValueBuf, pcchValueBuf ); str.unicode = TRUE; str.str.w = szValueBuf; return get_prop( handle, uiProperty, puiDataType, piValue, pftValue, &str, pcchValueBuf ); } static UINT set_prop( MSISUMMARYINFO *si, UINT uiProperty, UINT type, INT iValue, FILETIME* pftValue, awcstring *str ) { PROPVARIANT *prop; UINT len; TRACE("%p %u %u %i %p %p\n", si, uiProperty, type, iValue, pftValue, str ); prop = &si->property[uiProperty]; if( prop->vt == VT_EMPTY ) { if( !si->update_count ) return ERROR_FUNCTION_FAILED; si->update_count--; } else if( prop->vt != type ) return ERROR_SUCCESS; free_prop( prop ); prop->vt = type; switch( type ) { case VT_I4: prop->lVal = iValue; break; case VT_I2: prop->iVal = iValue; break; case VT_FILETIME: prop->filetime = *pftValue; break; case VT_LPSTR: if( str->unicode ) { len = WideCharToMultiByte( CP_ACP, 0, str->str.w, -1, NULL, 0, NULL, NULL ); prop->pszVal = msi_alloc( len ); WideCharToMultiByte( CP_ACP, 0, str->str.w, -1, prop->pszVal, len, NULL, NULL ); } else { len = lstrlenA( str->str.a ) + 1; prop->pszVal = msi_alloc( len ); lstrcpyA( prop->pszVal, str->str.a ); } break; } return ERROR_SUCCESS; } UINT WINAPI MsiSummaryInfoSetPropertyW( MSIHANDLE handle, UINT uiProperty, UINT uiDataType, INT iValue, FILETIME* pftValue, LPCWSTR szValue ) { awcstring str; MSISUMMARYINFO *si; UINT type, ret; TRACE("%d %u %u %i %p %s\n", handle, uiProperty, uiDataType, iValue, pftValue, debugstr_w(szValue) ); type = get_type( uiProperty ); if( type == VT_EMPTY || type != uiDataType ) return ERROR_DATATYPE_MISMATCH; if( uiDataType == VT_LPSTR && !szValue ) return ERROR_INVALID_PARAMETER; if( uiDataType == VT_FILETIME && !pftValue ) return ERROR_INVALID_PARAMETER; si = msihandle2msiinfo( handle, MSIHANDLETYPE_SUMMARYINFO ); if( !si ) return ERROR_INVALID_HANDLE; str.unicode = TRUE; str.str.w = szValue; ret = set_prop( si, uiProperty, type, iValue, pftValue, &str ); msiobj_release( &si->hdr ); return ret; } UINT WINAPI MsiSummaryInfoSetPropertyA( MSIHANDLE handle, UINT uiProperty, UINT uiDataType, INT iValue, FILETIME* pftValue, LPCSTR szValue ) { awcstring str; MSISUMMARYINFO *si; UINT type, ret; TRACE("%d %u %u %i %p %s\n", handle, uiProperty, uiDataType, iValue, pftValue, debugstr_a(szValue) ); type = get_type( uiProperty ); if( type == VT_EMPTY || type != uiDataType ) return ERROR_DATATYPE_MISMATCH; if( uiDataType == VT_LPSTR && !szValue ) return ERROR_INVALID_PARAMETER; if( uiDataType == VT_FILETIME && !pftValue ) return ERROR_INVALID_PARAMETER; si = msihandle2msiinfo( handle, MSIHANDLETYPE_SUMMARYINFO ); if( !si ) return ERROR_INVALID_HANDLE; str.unicode = FALSE; str.str.a = szValue; ret = set_prop( si, uiProperty, uiDataType, iValue, pftValue, &str ); msiobj_release( &si->hdr ); return ret; } static UINT suminfo_persist( MSISUMMARYINFO *si ) { UINT ret = ERROR_FUNCTION_FAILED; IStream *stm = NULL; DWORD grfMode; HRESULT r; grfMode = STGM_CREATE | STGM_READWRITE | STGM_SHARE_EXCLUSIVE; r = IStorage_CreateStream( si->storage, szSumInfo, grfMode, 0, 0, &stm ); if( SUCCEEDED(r) ) { ret = save_summary_info( si, stm ); IStream_Release( stm ); } return ret; } static void parse_filetime( LPCWSTR str, FILETIME *ft ) { SYSTEMTIME lt, utc; const WCHAR *p = str; WCHAR *end; memset( <, 0, sizeof(lt) ); /* YYYY/MM/DD hh:mm:ss */ while ( *p == ' ' || *p == '\t' ) p++; lt.wYear = strtolW( p, &end, 10 ); if (*end != '/') return; p = end + 1; lt.wMonth = strtolW( p, &end, 10 ); if (*end != '/') return; p = end + 1; lt.wDay = strtolW( p, &end, 10 ); if (*end != ' ') return; p = end + 1; while ( *p == ' ' || *p == '\t' ) p++; lt.wHour = strtolW( p, &end, 10 ); if (*end != ':') return; p = end + 1; lt.wMinute = strtolW( p, &end, 10 ); if (*end != ':') return; p = end + 1; lt.wSecond = strtolW( p, &end, 10 ); TzSpecificLocalTimeToSystemTime( NULL, <, &utc ); SystemTimeToFileTime( &utc, ft ); } static UINT parse_prop( LPCWSTR prop, LPCWSTR value, UINT *pid, INT *int_value, FILETIME *ft_value, awcstring *str_value ) { *pid = atoiW( prop ); switch (*pid) { case PID_CODEPAGE: case PID_WORDCOUNT: case PID_CHARCOUNT: case PID_SECURITY: case PID_PAGECOUNT: *int_value = atoiW( value ); break; case PID_LASTPRINTED: case PID_CREATE_DTM: case PID_LASTSAVE_DTM: parse_filetime( value, ft_value ); break; case PID_SUBJECT: case PID_AUTHOR: case PID_KEYWORDS: case PID_COMMENTS: case PID_TEMPLATE: case PID_LASTAUTHOR: case PID_REVNUMBER: case PID_APPNAME: case PID_TITLE: str_value->str.w = value; str_value->unicode = TRUE; break; default: WARN("unhandled prop id %u\n", *pid); return ERROR_FUNCTION_FAILED; } return ERROR_SUCCESS; } UINT msi_add_suminfo( MSIDATABASE *db, LPWSTR **records, int num_records, int num_columns ) { UINT r = ERROR_FUNCTION_FAILED; DWORD i, j; MSISUMMARYINFO *si; si = MSI_GetSummaryInformationW( db->storage, num_records * (num_columns / 2) ); if (!si) { ERR("no summary information!\n"); return ERROR_FUNCTION_FAILED; } for (i = 0; i < num_records; i++) { for (j = 0; j < num_columns; j += 2) { UINT pid; INT int_value = 0; FILETIME ft_value; awcstring str_value; r = parse_prop( records[i][j], records[i][j + 1], &pid, &int_value, &ft_value, &str_value ); if (r != ERROR_SUCCESS) goto end; r = set_prop( si, pid, get_type(pid), int_value, &ft_value, &str_value ); if (r != ERROR_SUCCESS) goto end; } } end: if (r == ERROR_SUCCESS) r = suminfo_persist( si ); msiobj_release( &si->hdr ); return r; } UINT WINAPI MsiSummaryInfoPersist( MSIHANDLE handle ) { MSISUMMARYINFO *si; UINT ret; TRACE("%d\n", handle ); si = msihandle2msiinfo( handle, MSIHANDLETYPE_SUMMARYINFO ); if( !si ) return ERROR_INVALID_HANDLE; ret = suminfo_persist( si ); msiobj_release( &si->hdr ); return ret; } UINT WINAPI MsiCreateTransformSummaryInfoA( MSIHANDLE db, MSIHANDLE db_ref, LPCSTR transform, int error, int validation ) { UINT r; WCHAR *transformW = NULL; TRACE("%u, %u, %s, %d, %d\n", db, db_ref, debugstr_a(transform), error, validation); if (transform && !(transformW = strdupAtoW( transform ))) return ERROR_OUTOFMEMORY; r = MsiCreateTransformSummaryInfoW( db, db_ref, transformW, error, validation ); msi_free( transformW ); return r; } UINT WINAPI MsiCreateTransformSummaryInfoW( MSIHANDLE db, MSIHANDLE db_ref, LPCWSTR transform, int error, int validation ) { FIXME("%u, %u, %s, %d, %d\n", db, db_ref, debugstr_w(transform), error, validation); return ERROR_FUNCTION_FAILED; }