1 /* 2 * Implementation of the Microsoft Installer (msi.dll) 3 * 4 * Copyright 2007 James Hawkins 5 * Copyright 2015 Hans Leidekker 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 #include <stdarg.h> 23 24 #define COBJMACROS 25 26 #include "windef.h" 27 #include "winbase.h" 28 #include "winerror.h" 29 #include "msi.h" 30 #include "msiquery.h" 31 #include "objbase.h" 32 #include "msipriv.h" 33 #include "query.h" 34 35 #include "wine/debug.h" 36 37 WINE_DEFAULT_DEBUG_CHANNEL(msidb); 38 39 #define NUM_STREAMS_COLS 2 40 41 typedef struct tagMSISTREAMSVIEW 42 { 43 MSIVIEW view; 44 MSIDATABASE *db; 45 UINT num_cols; 46 } MSISTREAMSVIEW; 47 48 static BOOL streams_resize_table( MSIDATABASE *db, UINT size ) 49 { 50 if (!db->num_streams_allocated) 51 { 52 if (!(db->streams = msi_alloc_zero( size * sizeof(MSISTREAM) ))) return FALSE; 53 db->num_streams_allocated = size; 54 return TRUE; 55 } 56 while (size >= db->num_streams_allocated) 57 { 58 MSISTREAM *tmp; 59 UINT new_size = db->num_streams_allocated * 2; 60 if (!(tmp = msi_realloc( db->streams, new_size * sizeof(*tmp) ))) return FALSE; 61 memset( tmp + db->num_streams_allocated, 0, (new_size - db->num_streams_allocated) * sizeof(*tmp) ); 62 db->streams = tmp; 63 db->num_streams_allocated = new_size; 64 } 65 return TRUE; 66 } 67 68 static UINT STREAMS_fetch_int(struct tagMSIVIEW *view, UINT row, UINT col, UINT *val) 69 { 70 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 71 72 TRACE("(%p, %d, %d, %p)\n", view, row, col, val); 73 74 if (col != 1) 75 return ERROR_INVALID_PARAMETER; 76 77 if (row >= sv->db->num_streams) 78 return ERROR_NO_MORE_ITEMS; 79 80 *val = sv->db->streams[row].str_index; 81 82 return ERROR_SUCCESS; 83 } 84 85 static UINT STREAMS_fetch_stream(struct tagMSIVIEW *view, UINT row, UINT col, IStream **stm) 86 { 87 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 88 LARGE_INTEGER pos; 89 HRESULT hr; 90 91 TRACE("(%p, %d, %d, %p)\n", view, row, col, stm); 92 93 if (row >= sv->db->num_streams) 94 return ERROR_FUNCTION_FAILED; 95 96 pos.QuadPart = 0; 97 hr = IStream_Seek( sv->db->streams[row].stream, pos, STREAM_SEEK_SET, NULL ); 98 if (FAILED( hr )) 99 return ERROR_FUNCTION_FAILED; 100 101 *stm = sv->db->streams[row].stream; 102 IStream_AddRef( *stm ); 103 104 return ERROR_SUCCESS; 105 } 106 107 static UINT STREAMS_set_string( struct tagMSIVIEW *view, UINT row, UINT col, const WCHAR *val, int len ) 108 { 109 ERR("Cannot modify primary key.\n"); 110 return ERROR_FUNCTION_FAILED; 111 } 112 113 static UINT STREAMS_set_stream( MSIVIEW *view, UINT row, UINT col, IStream *stream ) 114 { 115 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 116 IStream *prev; 117 118 TRACE("view %p, row %u, col %u, stream %p.\n", view, row, col, stream); 119 120 prev = sv->db->streams[row].stream; 121 IStream_AddRef(sv->db->streams[row].stream = stream); 122 if (prev) IStream_Release(prev); 123 return ERROR_SUCCESS; 124 } 125 126 static UINT STREAMS_set_row(struct tagMSIVIEW *view, UINT row, MSIRECORD *rec, UINT mask) 127 { 128 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 129 130 TRACE("(%p, %d, %p, %08x)\n", view, row, rec, mask); 131 132 if (row > sv->db->num_streams || mask >= (1 << sv->num_cols)) 133 return ERROR_INVALID_PARAMETER; 134 135 if (mask & 1) 136 { 137 const WCHAR *name = MSI_RecordGetString( rec, 1 ); 138 139 if (!name) return ERROR_INVALID_PARAMETER; 140 sv->db->streams[row].str_index = msi_add_string( sv->db->strings, name, -1, FALSE ); 141 } 142 if (mask & 2) 143 { 144 IStream *old, *new; 145 HRESULT hr; 146 UINT r; 147 148 r = MSI_RecordGetIStream( rec, 2, &new ); 149 if (r != ERROR_SUCCESS) 150 return r; 151 152 old = sv->db->streams[row].stream; 153 hr = IStream_QueryInterface( new, &IID_IStream, (void **)&sv->db->streams[row].stream ); 154 IStream_Release( new ); 155 if (FAILED( hr )) 156 { 157 return ERROR_FUNCTION_FAILED; 158 } 159 if (old) IStream_Release( old ); 160 } 161 162 return ERROR_SUCCESS; 163 } 164 165 static UINT streams_find_row( MSISTREAMSVIEW *sv, MSIRECORD *rec, UINT *row ) 166 { 167 const WCHAR *str; 168 UINT r, i, id, val; 169 170 str = MSI_RecordGetString( rec, 1 ); 171 r = msi_string2id( sv->db->strings, str, -1, &id ); 172 if (r != ERROR_SUCCESS) 173 return r; 174 175 for (i = 0; i < sv->db->num_streams; i++) 176 { 177 STREAMS_fetch_int( &sv->view, i, 1, &val ); 178 179 if (val == id) 180 { 181 if (row) *row = i; 182 return ERROR_SUCCESS; 183 } 184 } 185 186 return ERROR_FUNCTION_FAILED; 187 } 188 189 static UINT STREAMS_insert_row(struct tagMSIVIEW *view, MSIRECORD *rec, UINT row, BOOL temporary) 190 { 191 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 192 UINT i, r, num_rows = sv->db->num_streams + 1; 193 194 TRACE("(%p, %p, %d, %d)\n", view, rec, row, temporary); 195 196 r = streams_find_row( sv, rec, NULL ); 197 if (r == ERROR_SUCCESS) 198 return ERROR_FUNCTION_FAILED; 199 200 if (!streams_resize_table( sv->db, num_rows )) 201 return ERROR_FUNCTION_FAILED; 202 203 if (row == -1) 204 row = num_rows - 1; 205 206 /* shift the rows to make room for the new row */ 207 for (i = num_rows - 1; i > row; i--) 208 { 209 sv->db->streams[i] = sv->db->streams[i - 1]; 210 } 211 212 r = STREAMS_set_row( view, row, rec, (1 << sv->num_cols) - 1 ); 213 if (r == ERROR_SUCCESS) 214 sv->db->num_streams = num_rows; 215 216 return r; 217 } 218 219 static UINT STREAMS_delete_row(struct tagMSIVIEW *view, UINT row) 220 { 221 MSIDATABASE *db = ((MSISTREAMSVIEW *)view)->db; 222 UINT i, num_rows = db->num_streams - 1; 223 const WCHAR *name; 224 WCHAR *encname; 225 HRESULT hr; 226 227 TRACE("(%p %d)\n", view, row); 228 229 if (!db->num_streams || row > num_rows) 230 return ERROR_FUNCTION_FAILED; 231 232 name = msi_string_lookup( db->strings, db->streams[row].str_index, NULL ); 233 if (!(encname = encode_streamname( FALSE, name ))) return ERROR_OUTOFMEMORY; 234 IStream_Release( db->streams[row].stream ); 235 236 for (i = row; i < num_rows; i++) 237 db->streams[i] = db->streams[i + 1]; 238 db->num_streams = num_rows; 239 240 hr = IStorage_DestroyElement( db->storage, encname ); 241 msi_free( encname ); 242 return FAILED( hr ) ? ERROR_FUNCTION_FAILED : ERROR_SUCCESS; 243 } 244 245 static UINT STREAMS_execute(struct tagMSIVIEW *view, MSIRECORD *record) 246 { 247 TRACE("(%p, %p)\n", view, record); 248 return ERROR_SUCCESS; 249 } 250 251 static UINT STREAMS_close(struct tagMSIVIEW *view) 252 { 253 TRACE("(%p)\n", view); 254 return ERROR_SUCCESS; 255 } 256 257 static UINT STREAMS_get_dimensions(struct tagMSIVIEW *view, UINT *rows, UINT *cols) 258 { 259 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 260 261 TRACE("(%p, %p, %p)\n", view, rows, cols); 262 263 if (cols) *cols = sv->num_cols; 264 if (rows) *rows = sv->db->num_streams; 265 266 return ERROR_SUCCESS; 267 } 268 269 static UINT STREAMS_get_column_info( struct tagMSIVIEW *view, UINT n, LPCWSTR *name, 270 UINT *type, BOOL *temporary, LPCWSTR *table_name ) 271 { 272 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 273 274 TRACE("(%p, %d, %p, %p, %p, %p)\n", view, n, name, type, temporary, table_name); 275 276 if (!n || n > sv->num_cols) 277 return ERROR_INVALID_PARAMETER; 278 279 switch (n) 280 { 281 case 1: 282 if (name) *name = L"Name"; 283 if (type) *type = MSITYPE_STRING | MSITYPE_VALID | MAX_STREAM_NAME_LEN; 284 break; 285 286 case 2: 287 if (name) *name = L"Data"; 288 if (type) *type = MSITYPE_STRING | MSITYPE_VALID | MSITYPE_NULLABLE; 289 break; 290 } 291 if (table_name) *table_name = L"_Streams"; 292 if (temporary) *temporary = FALSE; 293 return ERROR_SUCCESS; 294 } 295 296 static UINT streams_modify_update(struct tagMSIVIEW *view, MSIRECORD *rec) 297 { 298 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 299 UINT r, row; 300 301 r = streams_find_row(sv, rec, &row); 302 if (r != ERROR_SUCCESS) 303 return ERROR_FUNCTION_FAILED; 304 305 return STREAMS_set_row( view, row, rec, (1 << sv->num_cols) - 1 ); 306 } 307 308 static UINT streams_modify_assign(struct tagMSIVIEW *view, MSIRECORD *rec) 309 { 310 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 311 UINT r; 312 313 r = streams_find_row( sv, rec, NULL ); 314 if (r == ERROR_SUCCESS) 315 return streams_modify_update(view, rec); 316 317 return STREAMS_insert_row(view, rec, -1, FALSE); 318 } 319 320 static UINT STREAMS_modify(struct tagMSIVIEW *view, MSIMODIFY eModifyMode, MSIRECORD *rec, UINT row) 321 { 322 UINT r; 323 324 TRACE("%p %d %p\n", view, eModifyMode, rec); 325 326 switch (eModifyMode) 327 { 328 case MSIMODIFY_ASSIGN: 329 r = streams_modify_assign(view, rec); 330 break; 331 332 case MSIMODIFY_INSERT: 333 r = STREAMS_insert_row(view, rec, -1, FALSE); 334 break; 335 336 case MSIMODIFY_UPDATE: 337 r = streams_modify_update(view, rec); 338 break; 339 340 case MSIMODIFY_DELETE: 341 r = STREAMS_delete_row(view, row - 1); 342 break; 343 344 case MSIMODIFY_VALIDATE_NEW: 345 case MSIMODIFY_INSERT_TEMPORARY: 346 case MSIMODIFY_REFRESH: 347 case MSIMODIFY_REPLACE: 348 case MSIMODIFY_MERGE: 349 case MSIMODIFY_VALIDATE: 350 case MSIMODIFY_VALIDATE_FIELD: 351 case MSIMODIFY_VALIDATE_DELETE: 352 FIXME("%p %d %p - mode not implemented\n", view, eModifyMode, rec ); 353 r = ERROR_CALL_NOT_IMPLEMENTED; 354 break; 355 356 default: 357 r = ERROR_INVALID_DATA; 358 } 359 360 return r; 361 } 362 363 static UINT STREAMS_delete(struct tagMSIVIEW *view) 364 { 365 MSISTREAMSVIEW *sv = (MSISTREAMSVIEW *)view; 366 367 TRACE("(%p)\n", view); 368 369 msi_free(sv); 370 return ERROR_SUCCESS; 371 } 372 373 static const MSIVIEWOPS streams_ops = 374 { 375 STREAMS_fetch_int, 376 STREAMS_fetch_stream, 377 NULL, 378 STREAMS_set_string, 379 STREAMS_set_stream, 380 STREAMS_set_row, 381 STREAMS_insert_row, 382 STREAMS_delete_row, 383 STREAMS_execute, 384 STREAMS_close, 385 STREAMS_get_dimensions, 386 STREAMS_get_column_info, 387 STREAMS_modify, 388 STREAMS_delete, 389 NULL, 390 NULL, 391 NULL, 392 NULL, 393 NULL, 394 }; 395 396 static HRESULT open_stream( MSIDATABASE *db, const WCHAR *name, IStream **stream ) 397 { 398 HRESULT hr; 399 400 hr = IStorage_OpenStream( db->storage, name, NULL, STGM_READ|STGM_SHARE_EXCLUSIVE, 0, stream ); 401 if (FAILED( hr )) 402 { 403 MSITRANSFORM *transform; 404 405 LIST_FOR_EACH_ENTRY( transform, &db->transforms, MSITRANSFORM, entry ) 406 { 407 hr = IStorage_OpenStream( transform->stg, name, NULL, STGM_READ|STGM_SHARE_EXCLUSIVE, 0, stream ); 408 if (SUCCEEDED( hr )) 409 break; 410 } 411 } 412 return hr; 413 } 414 415 static MSISTREAM *find_stream( MSIDATABASE *db, const WCHAR *name ) 416 { 417 UINT r, id, i; 418 419 r = msi_string2id( db->strings, name, -1, &id ); 420 if (r != ERROR_SUCCESS) 421 return NULL; 422 423 for (i = 0; i < db->num_streams; i++) 424 { 425 if (db->streams[i].str_index == id) return &db->streams[i]; 426 } 427 return NULL; 428 } 429 430 static UINT append_stream( MSIDATABASE *db, const WCHAR *name, IStream *stream ) 431 { 432 UINT i = db->num_streams; 433 434 if (!streams_resize_table( db, db->num_streams + 1 )) 435 return ERROR_OUTOFMEMORY; 436 437 db->streams[i].str_index = msi_add_string( db->strings, name, -1, FALSE ); 438 db->streams[i].stream = stream; 439 db->num_streams++; 440 441 TRACE("added %s\n", debugstr_w( name )); 442 return ERROR_SUCCESS; 443 } 444 445 static UINT load_streams( MSIDATABASE *db ) 446 { 447 WCHAR decoded[MAX_STREAM_NAME_LEN + 1]; 448 IEnumSTATSTG *stgenum; 449 STATSTG stat; 450 HRESULT hr; 451 ULONG count; 452 UINT r = ERROR_SUCCESS; 453 IStream *stream; 454 455 hr = IStorage_EnumElements( db->storage, 0, NULL, 0, &stgenum ); 456 if (FAILED( hr )) 457 return ERROR_FUNCTION_FAILED; 458 459 for (;;) 460 { 461 count = 0; 462 hr = IEnumSTATSTG_Next( stgenum, 1, &stat, &count ); 463 if (FAILED( hr ) || !count) 464 break; 465 466 /* table streams are not in the _Streams table */ 467 if (stat.type != STGTY_STREAM || *stat.pwcsName == 0x4840) 468 { 469 CoTaskMemFree( stat.pwcsName ); 470 continue; 471 } 472 decode_streamname( stat.pwcsName, decoded ); 473 if (find_stream( db, decoded )) 474 { 475 CoTaskMemFree( stat.pwcsName ); 476 continue; 477 } 478 TRACE("found new stream %s\n", debugstr_w( decoded )); 479 480 hr = open_stream( db, stat.pwcsName, &stream ); 481 CoTaskMemFree( stat.pwcsName ); 482 if (FAILED( hr )) 483 { 484 ERR( "unable to open stream %#lx\n", hr ); 485 r = ERROR_FUNCTION_FAILED; 486 break; 487 } 488 489 r = append_stream( db, decoded, stream ); 490 if (r != ERROR_SUCCESS) 491 break; 492 } 493 494 TRACE("loaded %u streams\n", db->num_streams); 495 IEnumSTATSTG_Release( stgenum ); 496 return r; 497 } 498 499 UINT msi_get_stream( MSIDATABASE *db, const WCHAR *name, IStream **ret ) 500 { 501 MSISTREAM *stream; 502 WCHAR *encname; 503 HRESULT hr; 504 UINT r; 505 506 if ((stream = find_stream( db, name ))) 507 { 508 LARGE_INTEGER pos; 509 510 pos.QuadPart = 0; 511 hr = IStream_Seek( stream->stream, pos, STREAM_SEEK_SET, NULL ); 512 if (FAILED( hr )) 513 return ERROR_FUNCTION_FAILED; 514 515 *ret = stream->stream; 516 IStream_AddRef( *ret ); 517 return ERROR_SUCCESS; 518 } 519 520 if (!(encname = encode_streamname( FALSE, name ))) 521 return ERROR_OUTOFMEMORY; 522 523 hr = open_stream( db, encname, ret ); 524 msi_free( encname ); 525 if (FAILED( hr )) 526 return ERROR_FUNCTION_FAILED; 527 528 r = append_stream( db, name, *ret ); 529 if (r != ERROR_SUCCESS) 530 { 531 IStream_Release( *ret ); 532 return r; 533 } 534 535 IStream_AddRef( *ret ); 536 return ERROR_SUCCESS; 537 } 538 539 UINT STREAMS_CreateView(MSIDATABASE *db, MSIVIEW **view) 540 { 541 MSISTREAMSVIEW *sv; 542 UINT r; 543 544 TRACE("(%p, %p)\n", db, view); 545 546 r = load_streams( db ); 547 if (r != ERROR_SUCCESS) 548 return r; 549 550 if (!(sv = msi_alloc_zero( sizeof(MSISTREAMSVIEW) ))) 551 return ERROR_OUTOFMEMORY; 552 553 sv->view.ops = &streams_ops; 554 sv->num_cols = NUM_STREAMS_COLS; 555 sv->db = db; 556 557 *view = (MSIVIEW *)sv; 558 559 return ERROR_SUCCESS; 560 } 561 562 static HRESULT write_stream( IStream *dst, IStream *src ) 563 { 564 HRESULT hr; 565 char buf[4096]; 566 STATSTG stat; 567 LARGE_INTEGER pos; 568 ULONG count; 569 UINT size; 570 571 hr = IStream_Stat( src, &stat, STATFLAG_NONAME ); 572 if (FAILED( hr )) return hr; 573 574 hr = IStream_SetSize( dst, stat.cbSize ); 575 if (FAILED( hr )) return hr; 576 577 pos.QuadPart = 0; 578 hr = IStream_Seek( dst, pos, STREAM_SEEK_SET, NULL ); 579 if (FAILED( hr )) return hr; 580 581 for (;;) 582 { 583 size = min( sizeof(buf), stat.cbSize.QuadPart ); 584 hr = IStream_Read( src, buf, size, &count ); 585 if (FAILED( hr ) || count != size) 586 { 587 WARN( "failed to read stream: %#lx\n", hr ); 588 return E_INVALIDARG; 589 } 590 stat.cbSize.QuadPart -= count; 591 if (count) 592 { 593 size = count; 594 hr = IStream_Write( dst, buf, size, &count ); 595 if (FAILED( hr ) || count != size) 596 { 597 WARN( "failed to write stream: %#lx\n", hr ); 598 return E_INVALIDARG; 599 } 600 } 601 if (!stat.cbSize.QuadPart) break; 602 } 603 604 return S_OK; 605 } 606 607 UINT msi_commit_streams( MSIDATABASE *db ) 608 { 609 UINT i; 610 const WCHAR *name; 611 WCHAR *encname; 612 IStream *stream; 613 HRESULT hr; 614 615 TRACE("got %u streams\n", db->num_streams); 616 617 for (i = 0; i < db->num_streams; i++) 618 { 619 name = msi_string_lookup( db->strings, db->streams[i].str_index, NULL ); 620 if (!wcscmp( name, L"\5SummaryInformation" )) continue; 621 622 if (!(encname = encode_streamname( FALSE, name ))) return ERROR_OUTOFMEMORY; 623 TRACE("saving stream %s as %s\n", debugstr_w(name), debugstr_w(encname)); 624 625 hr = IStorage_CreateStream( db->storage, encname, STGM_WRITE|STGM_SHARE_EXCLUSIVE, 0, 0, &stream ); 626 if (SUCCEEDED( hr )) 627 { 628 hr = write_stream( stream, db->streams[i].stream ); 629 if (FAILED( hr )) 630 { 631 ERR( "failed to write stream %s (hr = %#lx)\n", debugstr_w(encname), hr ); 632 msi_free( encname ); 633 IStream_Release( stream ); 634 return ERROR_FUNCTION_FAILED; 635 } 636 hr = IStream_Commit( stream, 0 ); 637 IStream_Release( stream ); 638 if (FAILED( hr )) 639 { 640 ERR( "failed to commit stream %s (hr = %#lx)\n", debugstr_w(encname), hr ); 641 msi_free( encname ); 642 return ERROR_FUNCTION_FAILED; 643 } 644 } 645 else if (hr != STG_E_FILEALREADYEXISTS) 646 { 647 ERR( "failed to create stream %s (hr = %#lx)\n", debugstr_w(encname), hr ); 648 msi_free( encname ); 649 return ERROR_FUNCTION_FAILED; 650 } 651 msi_free( encname ); 652 } 653 654 return ERROR_SUCCESS; 655 } 656