Changeset 21866 in vbox for trunk/src/VBox/VMM
- Timestamp:
- Jul 29, 2009 4:15:42 PM (16 years ago)
- svn:sync-xref-src-repo-rev:
- 50509
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/VMM/SSM-new.cpp
r21862 r21866 152 152 #include <iprt/param.h> 153 153 #include <iprt/thread.h> 154 #include <iprt/semaphore.h> 154 155 #include <iprt/string.h> 155 156 #include <iprt/uuid.h> … … 281 282 } SSMSTATE; 282 283 284 /** Pointer to a SSM stream buffer. */ 285 typedef struct SSMSTRMBUF *PSSMSTRMBUF; 286 /** 287 * A SSM stream buffer. 288 */ 289 typedef struct SSMSTRMBUF 290 { 291 /** The buffer data. */ 292 uint8_t abData[_64K]; 293 294 /** The stream position of this buffer. */ 295 uint64_t offStream; 296 /** The amount of buffered data. */ 297 size_t cb; 298 /** End of stream indicator (for read streams only). */ 299 bool fEndOfStream; 300 /** Pointer to the next buffer in the chain. */ 301 PSSMSTRMBUF volatile pNext; 302 } SSMSTRMBUF; 303 304 /** 305 * SSM stream. 306 * 307 * This is a typical producer / consumer setup with a dedicated I/O thread and 308 * fixed number of buffers for read ahead and write back. 309 */ 310 typedef struct SSMSTRM 311 { 312 /** The file handle. */ 313 RTFILE hFile; 314 315 /** Write (set) or read (clear) stream. */ 316 bool fWrite; 317 /** Termination indicator. */ 318 bool volatile fTerminating; 319 /** Stream error status. */ 320 int32_t volatile rc; 321 322 /** The head of the consumer queue. 323 * For save the consumer is the I/O thread. For load the I/O thread is the 324 * producer. */ 325 PSSMSTRMBUF volatile pHead; 326 /** Chain of free buffers. 327 * The consumer/producer roles are the inverse of pHead. */ 328 PSSMSTRMBUF volatile pFree; 329 /** Event that's signalled when pHead is updated. */ 330 RTSEMEVENT hEvtHead; 331 /** Event that's signalled when pFree is updated. */ 332 RTSEMEVENT hEvtFree; 333 334 /** List of pending buffers that has been dequeued from pHead and reversed. */ 335 PSSMSTRMBUF pPending; 336 /** Pointer to the current buffer. */ 337 PSSMSTRMBUF pCur; 338 /** The stream offset of the current buffer. */ 339 uint64_t offCurStream; 340 /** The current buffer offset. */ 341 uint32_t off; 342 /** Whether we're checksumming reads/writes. */ 343 bool fChecksummed; 344 /** The stream CRC if fChecksummed is set. */ 345 uint32_t u32StreamCRC; 346 /** How far into the buffer u32StreamCRC is up-to-date. 347 * This may lag behind off as it's desirable to checksum as large blocks as 348 * possible. */ 349 uint32_t offStreamCRC; 350 } SSMSTRM; 351 /** Pointer to a SSM stream. */ 352 typedef SSMSTRM *PSSMSTRM; 353 283 354 284 355 /** … … 287 358 typedef struct SSMHANDLE 288 359 { 289 /** The file handle. */ 290 RTFILE hFile; 360 /** Stream/buffer manager. */ 361 SSMSTRM Strm; 362 291 363 /** The VM handle. */ 292 364 PVM pVM; … … 321 393 /** the amount of % we reserve for the 'done' stage */ 322 394 unsigned uPercentDone; 323 324 /** Whether we're checksumming reads/writes. */325 bool fChecksummed;326 /** The stream CRC if fChecksummed is set. */327 uint32_t u32StreamCRC;328 395 329 396 union … … 1281 1348 1282 1349 /** 1350 * Initializes the stream after/before opening the file/whatever. 1351 * 1352 * @returns VINF_SUCCESS or VERR_NO_MEMORY. 1353 * @param pStrm The stream handle. 1354 * @param fChecksummed Whether the stream is to be checksummed while 1355 * written/read. 1356 */ 1357 static int ssmR3StrmInit(PSSMSTRM pStrm, bool fChecksummed) 1358 { 1359 /* 1360 * Init the common data members. 1361 */ 1362 pStrm->fTerminating = false; 1363 pStrm->rc = VINF_SUCCESS; 1364 1365 pStrm->pHead = NULL; 1366 pStrm->pFree = NULL; 1367 pStrm->hEvtHead = NIL_RTSEMEVENT; 1368 pStrm->hEvtFree = NIL_RTSEMEVENT; 1369 1370 pStrm->pPending = NULL; 1371 pStrm->pCur = NULL; 1372 pStrm->offCurStream = 0; 1373 pStrm->off = 0; 1374 pStrm->fChecksummed = fChecksummed; 1375 pStrm->u32StreamCRC = fChecksummed ? RTCrc32Start() : 0; 1376 pStrm->offStreamCRC = 0; 1377 1378 /* 1379 * Allocate the buffers. Page align them in case that makes the kernel 1380 * and/or cpu happier in some way. 1381 */ 1382 int rc = VINF_SUCCESS; 1383 for (unsigned i = 0; i < 4; i++) 1384 { 1385 PSSMSTRMBUF pBuf = (PSSMSTRMBUF)RTMemPageAllocZ(sizeof(*pBuf)); 1386 if (!pBuf) 1387 { 1388 if (i > 2) 1389 { 1390 LogRel(("ssmR3StrmAllocBuffer: WARNING: Could only get %d stream buffers.\n", i)); 1391 break; 1392 } 1393 LogRel(("ssmR3StrmAllocBuffer: Failed to allocate stream buffers. (i=%d)\n", i)); 1394 return VERR_NO_MEMORY; 1395 } 1396 1397 /* link it */ 1398 pBuf->pNext = pStrm->pFree; 1399 pStrm->pFree = pBuf; 1400 } 1401 1402 /* 1403 * Create the event semaphores. 1404 */ 1405 rc = RTSemEventCreate(&pStrm->hEvtHead); 1406 if (RT_FAILURE(rc)) 1407 return rc; 1408 rc = RTSemEventCreate(&pStrm->hEvtFree); 1409 if (RT_FAILURE(rc)) 1410 return rc; 1411 1412 return VINF_SUCCESS; 1413 } 1414 1415 1416 /** 1417 * Destroys a list of buffers. 1418 * 1419 * @param pHead Pointer to the head. 1420 */ 1421 static void ssmR3StrmDestroyBufList(PSSMSTRMBUF pHead) 1422 { 1423 while (pHead) 1424 { 1425 PSSMSTRMBUF pCur = pHead; 1426 pHead = pCur->pNext; 1427 pCur->pNext = NULL; 1428 RTMemPageFree(pCur); 1429 } 1430 } 1431 1432 1433 /** 1434 * Cleans up a stream after ssmR3StrmInit has been called (regardless of it 1435 * succeeded or not). 1436 * 1437 * @param pStrm The stream handle. 1438 */ 1439 static void ssmR3StrmDelete(PSSMSTRM pStrm) 1440 { 1441 RTMemPageFree(pStrm->pCur); 1442 pStrm->pCur = NULL; 1443 ssmR3StrmDestroyBufList(pStrm->pHead); 1444 pStrm->pHead = NULL; 1445 ssmR3StrmDestroyBufList(pStrm->pPending); 1446 pStrm->pPending = NULL; 1447 ssmR3StrmDestroyBufList(pStrm->pFree); 1448 pStrm->pFree = NULL; 1449 1450 RTSemEventDestroy(pStrm->hEvtHead); 1451 pStrm->hEvtHead = NIL_RTSEMEVENT; 1452 1453 RTSemEventDestroy(pStrm->hEvtFree); 1454 pStrm->hEvtFree = NIL_RTSEMEVENT; 1455 } 1456 1457 1458 /** 1459 * Opens a file stream. 1460 * 1461 * @returns VBox status code. 1462 * @param pStrm The stream manager structure. 1463 * @param pszFilename The file to open or create. 1464 * @param fWrite Whether to open for writing or reading. 1465 * @param fChecksummed Whether the stream is to be checksummed while 1466 * written/read. 1467 */ 1468 static int ssmR3StrmOpenFile(PSSMSTRM pStrm, const char *pszFilename, bool fWrite, bool fChecksummed) 1469 { 1470 int rc = ssmR3StrmInit(pStrm, fChecksummed); 1471 if (RT_SUCCESS(rc)) 1472 { 1473 uint32_t fFlags = fWrite 1474 ? RTFILE_O_READWRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_DENY_WRITE 1475 : RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE; 1476 rc = RTFileOpen(&pStrm->hFile, pszFilename, fFlags); 1477 if (RT_SUCCESS(rc)) 1478 { 1479 pStrm->fWrite = fWrite; 1480 return VINF_SUCCESS; 1481 } 1482 } 1483 1484 ssmR3StrmDelete(pStrm); 1485 pStrm->rc = rc; 1486 return rc; 1487 } 1488 1489 1490 /** 1491 * Raise an error condition on the stream. 1492 * 1493 * @returns true if we raised the error condition, false if the stream already 1494 * had an error condition set. 1495 * 1496 * @param pStrm The stream handle. 1497 * @param rc The VBox error status code. 1498 * 1499 * @thread Any. 1500 */ 1501 DECLINLINE(bool) ssmR3StrmSetError(PSSMSTRM pStrm, int rc) 1502 { 1503 Assert(RT_FAILURE_NP(rc)); 1504 return ASMAtomicCmpXchgS32(&pStrm->rc, rc, VINF_SUCCESS); 1505 } 1506 1507 1508 /** 1509 * Puts a buffer into the free list. 1510 * 1511 * @param pStrm The stream handle. 1512 * @param pBuf The buffer. 1513 * 1514 * @thread The consumer. 1515 */ 1516 static void ssmR3StrmPutFreeBuf(PSSMSTRM pStrm, PSSMSTRMBUF pBuf) 1517 { 1518 for (;;) 1519 { 1520 PSSMSTRMBUF pCurFreeHead = (PSSMSTRMBUF)ASMAtomicUoReadPtr((void * volatile *)&pStrm->pFree); 1521 ASMAtomicUoWritePtr((void * volatile *)&pBuf->pNext, pCurFreeHead); 1522 if (ASMAtomicCmpXchgPtr((void * volatile *)&pStrm->pFree, pBuf, pCurFreeHead)) 1523 { 1524 int rc = RTSemEventSignal(pStrm->hEvtFree); 1525 AssertRC(rc); 1526 return; 1527 } 1528 } 1529 } 1530 1531 1532 /** 1533 * Gets a free buffer, waits for one if necessary. 1534 * 1535 * @returns Pointer to the buffer on success. NULL if we're terminating. 1536 * @param pStrm The stream handle. 1537 * 1538 * @thread The producer. 1539 */ 1540 static PSSMSTRMBUF ssmR3StrmGetFreeBuf(PSSMSTRM pStrm) 1541 { 1542 for (;;) 1543 { 1544 PSSMSTRMBUF pMine = (PSSMSTRMBUF)ASMAtomicUoReadPtr((void * volatile *)pStrm->pFree); 1545 if (!pMine) 1546 { 1547 if (pStrm->fTerminating) 1548 return NULL; 1549 if (RT_FAILURE(pStrm->rc)) 1550 return NULL; 1551 int rc = RTSemEventWaitNoResume(pStrm->hEvtFree, 30000); 1552 if ( rc == VERR_SEM_DESTROYED 1553 || pStrm->fTerminating) 1554 return NULL; 1555 continue; 1556 } 1557 1558 if (ASMAtomicCmpXchgPtr((void * volatile *)&pStrm->pFree, pMine->pNext, pMine)) 1559 { 1560 pMine->offStream = UINT64_MAX; 1561 pMine->cb = 0; 1562 pMine->pNext = NULL; 1563 pMine->fEndOfStream = false; 1564 return pMine; 1565 } 1566 } 1567 } 1568 1569 1570 /** 1571 * Puts a buffer onto the queue. 1572 * 1573 * @param pBuf The buffer. 1574 * 1575 * @thread The producer. 1576 */ 1577 static void ssmR3StrmPutBuf(PSSMSTRM pStrm, PSSMSTRMBUF pBuf) 1578 { 1579 for (;;) 1580 { 1581 PSSMSTRMBUF pCurHead = (PSSMSTRMBUF)ASMAtomicUoReadPtr((void * volatile *)&pStrm->pHead); 1582 ASMAtomicUoWritePtr((void * volatile *)&pBuf->pNext, pCurHead); 1583 if (ASMAtomicCmpXchgPtr((void * volatile *)&pStrm->pHead, pBuf, pCurHead)) 1584 { 1585 int rc = RTSemEventSignal(pStrm->hEvtHead); 1586 AssertRC(rc); 1587 return; 1588 } 1589 } 1590 } 1591 1592 1593 /** 1594 * Reverses the list. 1595 * 1596 * @returns The head of the reversed list. 1597 * @param pHead The head of the list to reverse. 1598 */ 1599 static PSSMSTRMBUF ssmR3StrmReverseList(PSSMSTRMBUF pHead) 1600 { 1601 PSSMSTRMBUF pRevHead = NULL; 1602 while (pHead) 1603 { 1604 PSSMSTRMBUF pCur = pHead; 1605 pHead = pCur->pNext; 1606 pCur->pNext = pRevHead; 1607 pRevHead = pCur; 1608 } 1609 return pRevHead; 1610 } 1611 1612 1613 /** 1614 * Gets one buffer from the queue, will wait for one to become ready if 1615 * necessary. 1616 * 1617 * @returns Pointer to the buffer on success. NULL if we're terminating. 1618 * @param pBuf The buffer. 1619 * 1620 * @thread The consumer. 1621 */ 1622 static PSSMSTRMBUF ssmR3StrmGetBuf(PSSMSTRM pStrm) 1623 { 1624 for (;;) 1625 { 1626 PSSMSTRMBUF pMine = pStrm->pPending; 1627 if (pMine) 1628 { 1629 pStrm->pPending = pMine->pNext; 1630 pMine->pNext = NULL; 1631 return pMine; 1632 } 1633 1634 pMine = (PSSMSTRMBUF)ASMAtomicXchgPtr((void * volatile *)pStrm->pHead, NULL); 1635 if (pMine) 1636 pStrm->pPending = ssmR3StrmReverseList(pMine); 1637 else 1638 { 1639 if (pStrm->fTerminating) 1640 return NULL; 1641 if (RT_FAILURE(pStrm->rc)) 1642 return NULL; 1643 int rc = RTSemEventWaitNoResume(pStrm->hEvtHead, 30000); 1644 if ( rc == VERR_SEM_DESTROYED 1645 || pStrm->fTerminating) 1646 return NULL; 1647 } 1648 } 1649 } 1650 1651 1652 /** 1653 * Flushes the current buffer (both write and read streams). 1654 * 1655 * @param pStrm The stream handle. 1656 */ 1657 static void ssmR3StrmFlushCurBuf(PSSMSTRM pStrm) 1658 { 1659 if (pStrm->pCur) 1660 { 1661 PSSMSTRMBUF pCur = pStrm->pCur; 1662 pStrm->pCur = NULL; 1663 1664 if (pStrm->fWrite) 1665 { 1666 uint32_t cb = pStrm->off; 1667 pCur->cb = cb; 1668 pCur->offStream = pStrm->offCurStream; 1669 if ( pStrm->fChecksummed 1670 && pStrm->offStreamCRC < cb) 1671 pStrm->u32StreamCRC = RTCrc32Process(pStrm->u32StreamCRC, 1672 &pCur->abData[pStrm->offStreamCRC], 1673 cb - pStrm->offStreamCRC); 1674 pStrm->offCurStream += cb; 1675 pStrm->off = 0; 1676 pStrm->offStreamCRC = 0; 1677 1678 ssmR3StrmPutBuf(pStrm, pCur); 1679 } 1680 else 1681 { 1682 uint32_t cb = pCur->cb; 1683 if ( pStrm->fChecksummed 1684 && pStrm->offStreamCRC < cb) 1685 pStrm->u32StreamCRC = RTCrc32Process(pStrm->u32StreamCRC, 1686 &pCur->abData[pStrm->offStreamCRC], 1687 cb - pStrm->offStreamCRC); 1688 pStrm->offCurStream += cb; 1689 pStrm->off = 0; 1690 pStrm->offStreamCRC = 0; 1691 1692 ssmR3StrmPutFreeBuf(pStrm, pCur); 1693 } 1694 } 1695 } 1696 1697 1698 /** 1699 * Flush any buffered data. 1700 * 1701 * @returns VBox status code. 1702 * @param pStrm The stream handle. 1703 * 1704 * @thread The producer thread. 1705 */ 1706 static int ssmR3StrmFlush(PSSMSTRM pStrm) 1707 { 1708 /* 1709 * Read streams have nothing that needs to be flushed. 1710 */ 1711 if (!pStrm->fWrite) 1712 return VINF_SUCCESS; 1713 int rc = pStrm->rc; 1714 if (RT_FAILURE(rc)) 1715 return rc; 1716 1717 /* 1718 * Complete the current buffer and queue it. 1719 */ 1720 ssmR3StrmFlushCurBuf(pStrm); 1721 1722 /** @todo This is what the I/O thread will be doing when we finally get one. */ 1723 1724 /* 1725 * Grab the pending list and write it out. 1726 */ 1727 PSSMSTRMBUF pHead = (PSSMSTRMBUF)ASMAtomicXchgPtr((void * volatile *)&pStrm->pHead, NULL); 1728 if (!pHead) 1729 return VINF_SUCCESS; 1730 pHead = ssmR3StrmReverseList(pHead); 1731 1732 while (pHead) 1733 { 1734 /* pop */ 1735 PSSMSTRMBUF pCur = pHead; 1736 pHead = pCur->pNext; 1737 1738 /* flush */ 1739 int rc = RTFileWriteAt(pStrm->hFile, pCur->offStream, &pCur->abData[0], pCur->cb, NULL); 1740 if ( RT_FAILURE(rc) 1741 && ssmR3StrmSetError(pStrm, rc)) 1742 LogRel(("ssmR3StrmFlush: RTFileWriteAt failed with rc=%Rrc at offStream=%#llx\n", rc, pCur->offStream)); 1743 1744 /* free it */ 1745 ssmR3StrmPutFreeBuf(pStrm, pCur); 1746 } 1747 1748 return pStrm->rc; 1749 } 1750 1751 1752 /** 1753 * Closes the stream after first flushing any pending write. 1754 * 1755 * @returns VBox status code. 1756 * @param pStrm The stream handle. 1757 */ 1758 static int ssmR3StrmClose(PSSMSTRM pStrm) 1759 { 1760 /* 1761 * Flush and close the stream. 1762 */ 1763 ssmR3StrmFlush(pStrm); 1764 int rc = RTFileClose(pStrm->hFile); 1765 if (RT_FAILURE(rc)) 1766 ssmR3StrmSetError(pStrm, rc); 1767 pStrm->hFile = NIL_RTFILE; 1768 1769 rc = pStrm->rc; 1770 ssmR3StrmDelete(pStrm); 1771 1772 return rc; 1773 } 1774 1775 1776 /** 1283 1777 * Stream output routine. 1284 1778 * 1285 1779 * @returns VBox status code. 1286 * @param pS SM The SSMhandle.1780 * @param pStrm The stream handle. 1287 1781 * @param pvBuf What to write. 1288 1782 * @param cbToWrite How much to write. 1289 */ 1290 static int ssmR3StrmWrite(PSSMHANDLE pSSM, const void *pvBuf, size_t cbToWrite) 1291 { 1292 Assert(pSSM->enmOp <= SSMSTATE_SAVE_DONE); 1293 int rc = RTFileWrite(pSSM->hFile, pvBuf, cbToWrite, NULL); 1783 * 1784 * @thread The producer in a write stream (never the I/O thread). 1785 */ 1786 static int ssmR3StrmWrite(PSSMSTRM pStrm, const void *pvBuf, size_t cbToWrite) 1787 { 1788 AssertReturn(cbToWrite > 0, VINF_SUCCESS); 1789 Assert(pStrm->fWrite); 1790 1791 /* 1792 * Squeeze as much as possible into the current buffer. 1793 */ 1794 PSSMSTRMBUF pBuf = pStrm->pCur; 1795 if (RT_LIKELY(pBuf)) 1796 { 1797 uint32_t cbLeft = pStrm->off - RT_SIZEOFMEMB(SSMSTRMBUF, abData); 1798 if (RT_LIKELY(cbLeft >= cbToWrite)) 1799 { 1800 memcpy(&pBuf->abData[pStrm->off], pvBuf, cbToWrite); 1801 pStrm->off += cbToWrite; 1802 return VINF_SUCCESS; 1803 } 1804 1805 if (cbLeft > 0) 1806 { 1807 memcpy(&pBuf->abData[pStrm->off], pvBuf, cbLeft); 1808 pStrm->off += cbLeft; 1809 cbToWrite -= cbLeft; 1810 pvBuf = (uint8_t const *)pvBuf + cbLeft; 1811 } 1812 } 1813 1814 /* 1815 * Need one or more new buffers. 1816 */ 1817 do 1818 { 1819 /* 1820 * Flush the current buffer and replace it with a new one. 1821 */ 1822 int rc = ssmR3StrmFlush(pStrm); 1823 if (RT_FAILURE(rc)) 1824 return rc; 1825 pBuf = ssmR3StrmGetFreeBuf(pStrm); 1826 if (!pBuf) 1827 break; 1828 pStrm->pCur = pBuf; 1829 Assert(pStrm->off == 0); 1830 1831 /* 1832 * Copy data to the buffer. 1833 */ 1834 uint32_t cbCopy = pStrm->off - RT_SIZEOFMEMB(SSMSTRMBUF, abData); 1835 if (cbCopy > cbToWrite) 1836 cbCopy = cbToWrite; 1837 memcpy(&pBuf->abData[pStrm->off], pvBuf, cbCopy); 1838 pStrm->off += cbCopy; 1839 cbToWrite -= cbCopy; 1840 pvBuf = (uint8_t const *)pvBuf + cbCopy; 1841 } while (cbToWrite > 0); 1842 1843 return pStrm->rc; 1844 } 1845 1846 1847 1848 /** 1849 * Read more from the stream. 1850 * 1851 * @returns VBox status code. VERR_EOF is passed up. 1852 * @param pStrm The stream handle. 1853 * 1854 * @thread The I/O thread when we get one. 1855 */ 1856 static int ssmR3StrmReadMore(PSSMSTRM pStrm) 1857 { 1858 PSSMSTRMBUF pBuf = ssmR3StrmGetFreeBuf(pStrm); 1859 if (!pBuf) 1860 return pStrm->rc; 1861 1862 pBuf->offStream = RTFileTell(pStrm->hFile); 1863 size_t cbRead = sizeof(pBuf->abData); 1864 int rc = RTFileRead(pStrm->hFile, &pBuf->abData[0], cbRead, &cbRead); 1294 1865 if (RT_SUCCESS(rc)) 1295 1866 { 1296 if (pSSM->fChecksummed) 1297 pSSM->u32StreamCRC = RTCrc32Process(pSSM->u32StreamCRC, pvBuf, cbToWrite); 1298 return VINF_SUCCESS; 1299 } 1300 1301 LogRel(("ssmR3StrmWrite: RTFileWrite(,,%zu) -> %Rrc\n", cbToWrite, rc)); 1302 if (RT_SUCCESS(pSSM->rc)) 1303 pSSM->rc = rc; 1867 pBuf->cb = cbRead; 1868 pBuf->fEndOfStream = cbRead == 0; 1869 ssmR3StrmPutBuf(pStrm, pBuf); 1870 } 1871 else if (rc == VERR_EOF) 1872 { 1873 pBuf->cb = 0; 1874 pBuf->fEndOfStream = true; 1875 ssmR3StrmPutBuf(pStrm, pBuf); 1876 } 1877 else 1878 { 1879 if (ssmR3StrmSetError(pStrm, rc)) 1880 LogRel(("ssmR3StrmReadMore: RTFileRead(,,%#x,) -> %Rrc at offset %#llx\n", 1881 sizeof(pBuf->abData), rc, pBuf->offStream)); 1882 ssmR3StrmPutFreeBuf(pStrm, pBuf); 1883 } 1304 1884 return rc; 1305 1885 } … … 1310 1890 * 1311 1891 * @returns VBox status code. 1312 * @param pS SM The SSMhandle.1892 * @param pStrm The stream handle. 1313 1893 * @param pvBuf Where to put what we read. 1314 1894 * @param cbToRead How much to read. 1315 1895 */ 1316 static int ssmR3StrmRead(PSSMHANDLE pSSM, void *pvBuf, size_t cbToRead) 1317 { 1318 Assert(pSSM->enmOp >= SSMSTATE_LOAD_PREP); 1319 int rc = RTFileRead(pSSM->hFile, pvBuf, cbToRead, NULL); 1320 if (RT_SUCCESS(rc)) 1321 { 1322 if (pSSM->fChecksummed) 1323 pSSM->u32StreamCRC = RTCrc32Process(pSSM->u32StreamCRC, pvBuf, cbToRead); 1324 return VINF_SUCCESS; 1325 } 1326 1327 LogRel(("ssmR3StrmRead: RTFileRead(,,%zu) -> %Rrc\n", cbToRead, rc)); 1328 if (RT_SUCCESS(pSSM->rc)) 1329 pSSM->rc = rc; 1896 static int ssmR3StrmRead(PSSMSTRM pStrm, void *pvBuf, size_t cbToRead) 1897 { 1898 AssertReturn(cbToRead > 0, VINF_SUCCESS); 1899 Assert(!pStrm->fWrite); 1900 1901 /* 1902 * Read from the current buffer if we got one. 1903 */ 1904 PSSMSTRMBUF pBuf = pStrm->pCur; 1905 if (RT_LIKELY(pBuf)) 1906 { 1907 Assert(pStrm->off <= pBuf->cb); 1908 uint32_t cbLeft = pStrm->off - pBuf->cb; 1909 if (cbLeft >= cbToRead) 1910 { 1911 memcpy(pvBuf, &pBuf->abData[pStrm->off], cbToRead); 1912 pStrm->off += cbToRead; 1913 return VINF_SUCCESS; 1914 } 1915 if (cbLeft) 1916 { 1917 memcpy(pvBuf, &pBuf->abData[pStrm->off], cbLeft); 1918 pStrm->off += cbLeft; 1919 cbToRead -= cbLeft; 1920 pvBuf = (uint8_t *)pvBuf + cbLeft; 1921 } 1922 else if (pBuf->fEndOfStream) 1923 return VERR_EOF; 1924 } 1925 1926 /* 1927 * Get more buffers from the stream. 1928 */ 1929 int rc; 1930 do 1931 { 1932 /* 1933 * Check for EOF first - never flush the EOF buffer. 1934 */ 1935 if ( pBuf 1936 && pBuf->fEndOfStream) 1937 return VERR_EOF; 1938 1939 /* 1940 * Flush the current buffer and get the next one. 1941 */ 1942 ssmR3StrmFlushCurBuf(pStrm); 1943 rc = ssmR3StrmReadMore(pStrm); /** @todo Remove this when adding the I/O thread! */ 1944 if (RT_FAILURE(rc)) 1945 break; 1946 PSSMSTRMBUF pBuf = ssmR3StrmGetBuf(pStrm); 1947 if (!pBuf) 1948 { 1949 rc = pStrm->rc; 1950 break; 1951 } 1952 pStrm->pCur = pBuf; 1953 Assert(pStrm->off == 0); 1954 Assert(pStrm->offCurStream == pBuf->offStream); 1955 if (!pBuf->cb) 1956 { 1957 Assert(pBuf->fEndOfStream); 1958 return VERR_EOF; 1959 } 1960 1961 /* 1962 * Read data from the buffer. 1963 */ 1964 uint32_t cbCopy = pStrm->off - pBuf->cb; 1965 if (cbCopy > cbToRead) 1966 cbCopy = cbToRead; 1967 memcpy(pvBuf, &pBuf->abData[pStrm->off], cbCopy); 1968 pStrm->off += cbCopy; 1969 cbToRead -= cbCopy; 1970 pvBuf = (uint8_t *)pvBuf + cbCopy; 1971 } while (cbToRead > 0); 1972 1330 1973 return rc; 1331 1974 } … … 1333 1976 1334 1977 /** 1335 * Flush any buffered data. 1978 * Tell current stream position. 1979 * 1980 * @returns stream position. 1981 * @param pStrm The stream handle. 1982 */ 1983 static uint64_t ssmR3StrmTell(PSSMSTRM pStrm) 1984 { 1985 return pStrm->offCurStream + pStrm->off; 1986 } 1987 1988 1989 /** 1990 * Gets the intermediate stream CRC up to the current position. 1991 * 1992 * @returns CRC. 1993 * @param pStrm The stream handle. 1994 */ 1995 static uint32_t ssmR3StrmCurCRC(PSSMSTRM pStrm) 1996 { 1997 if (!pStrm->fChecksummed) 1998 return 0; 1999 if (pStrm->offStreamCRC < pStrm->off) 2000 { 2001 PSSMSTRMBUF pBuf = pStrm->pCur; Assert(pBuf); 2002 pStrm->u32StreamCRC = RTCrc32Process(pStrm->u32StreamCRC, &pBuf->abData[pStrm->off], pStrm->off - pStrm->offStreamCRC); 2003 } 2004 else 2005 Assert(pStrm->offStreamCRC == pStrm->off); 2006 return pStrm->u32StreamCRC; 2007 } 2008 2009 2010 /** 2011 * Gets the final stream CRC up to the current position. 2012 * 2013 * @returns CRC. 2014 * @param pStrm The stream handle. 2015 */ 2016 static uint32_t ssmR3StrmFinalCRC(PSSMSTRM pStrm) 2017 { 2018 if (!pStrm->fChecksummed) 2019 return 0; 2020 return RTCrc32Finish(ssmR3StrmCurCRC(pStrm)); 2021 } 2022 2023 2024 /** 2025 * Disables checksumming of the stream. 2026 * 2027 * @param pStrm The stream handle. 2028 */ 2029 static void ssmR3StrmDisableChecksumming(PSSMSTRM pStrm) 2030 { 2031 pStrm->fChecksummed = false; 2032 } 2033 2034 2035 /** 2036 * Skip some bytes in the stream. 2037 * 2038 * This is only used if someone didn't read all of their data in the V1 format, 2039 * so don't bother making this very efficient yet. 1336 2040 * 1337 2041 * @returns VBox status code. 1338 * @param pSSM The SSM handle. 1339 */ 1340 static int ssmR3StrmFlush(PSSMHANDLE pSSM) 1341 { 1342 /* no buffering yet. */ 1343 return VINF_SUCCESS; 1344 } 1345 1346 1347 /** 1348 * Tell current stream position. 1349 * 1350 * @returns stream position. 1351 * @param pSSM The SSM handle. 1352 */ 1353 static uint64_t ssmR3StrmTell(PSSMHANDLE pSSM) 1354 { 1355 return RTFileTell(pSSM->hFile); 2042 * @param pStrm The stream handle. 2043 * @param offDst The destination offset. 2044 */ 2045 static int ssmR3StrmSkipTo(PSSMSTRM pStrm, uint64_t offDst) 2046 { 2047 /* dead simple - lazy bird! */ 2048 for (;;) 2049 { 2050 uint64_t offCur = ssmR3StrmTell(pStrm); 2051 AssertReturn(offCur <= offDst, VERR_INTERNAL_ERROR_4); 2052 if (offCur == offDst) 2053 return VINF_SUCCESS; 2054 2055 uint8_t abBuf[4096]; 2056 size_t cbToRead = RT_MIN(sizeof(abBuf), offDst - offCur); 2057 int rc = ssmR3StrmRead(pStrm, abBuf, cbToRead); 2058 if (RT_FAILURE(rc)) 2059 return rc; 2060 } 2061 } 2062 2063 2064 /** 2065 * Get the size of the file. 2066 * 2067 * This does not work for non-file streams! 2068 * 2069 * @returns The file size, or UINT64_MAX if not a file stream. 2070 * @param pStrm The stream handle. 2071 */ 2072 static uint64_t ssmR3StrmGetSize(PSSMSTRM pStrm) 2073 { 2074 uint64_t cbFile; 2075 int rc = RTFileGetSize(pStrm->hFile, &cbFile); 2076 AssertLogRelRCReturn(rc, UINT64_MAX); 2077 return cbFile; 1356 2078 } 1357 2079 … … 1484 2206 cbDir = RT_OFFSETOF(SSMFILEDIR, aEntries[pDir->cEntries]); 1485 2207 pDir->u32CRC = RTCrc32(pDir, cbDir); 1486 int rc = ssmR3StrmWrite( pSSM, pDir, cbDir);2208 int rc = ssmR3StrmWrite(&pSSM->Strm, pDir, cbDir); 1487 2209 RTMemTmpFree(pDir); 1488 2210 return rc; … … 1525 2247 * the saved state file might take a long time. 1526 2248 */ 1527 SSMHANDLE Handle = {0};1528 Handle.hFile = NIL_RTFILE;2249 SSMHANDLE Handle; 2250 RT_ZERO(Handle); 1529 2251 Handle.pVM = pVM; 1530 2252 Handle.enmOp = SSMSTATE_INVALID; … … 1542 2264 Handle.uPercentPrepare = 20; 1543 2265 Handle.uPercentDone = 2; 1544 Handle.fChecksummed = true;1545 Handle.u32StreamCRC = RTCrc32Start();1546 2266 Handle.u.Write.offDataBuffer = 0; 1547 2267 1548 int rc = RTFileOpen(&Handle.hFile, pszFilename, RTFILE_O_READWRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_DENY_WRITE);2268 int rc = ssmR3StrmOpenFile(&Handle.Strm, pszFilename, true /*fWrite*/, true /*fChecksummed*/); 1549 2269 if (RT_FAILURE(rc)) 1550 2270 { … … 1579 2299 u.FileHdr.u32CRC = 0; 1580 2300 u.FileHdr.u32CRC = RTCrc32(&u.FileHdr, sizeof(u.FileHdr)); 1581 rc = ssmR3StrmWrite(&Handle , &u.FileHdr, sizeof(u.FileHdr));2301 rc = ssmR3StrmWrite(&Handle.Strm, &u.FileHdr, sizeof(u.FileHdr)); 1582 2302 if (RT_SUCCESS(rc)) 1583 2303 { … … 1654 2374 continue; 1655 2375 } 1656 pUnit->offStream = ssmR3StrmTell(&Handle );2376 pUnit->offStream = ssmR3StrmTell(&Handle.Strm); 1657 2377 1658 2378 /* … … 1661 2381 memcpy(&u.UnitHdr.szMagic[0], SSMFILEUNITHDR_MAGIC, sizeof(u.UnitHdr.szMagic)); 1662 2382 u.UnitHdr.offStream = pUnit->offStream; 1663 if (Handle.fChecksummed) 1664 ssmR3StrmFlush(&Handle); 1665 u.UnitHdr.u32CurStreamCRC = Handle.u32StreamCRC; 2383 u.UnitHdr.u32CurStreamCRC = ssmR3StrmCurCRC(&Handle.Strm); 1666 2384 u.UnitHdr.u32CRC = 0; 1667 2385 u.UnitHdr.u32Version = pUnit->u32Version; … … 1674 2392 Log(("SSM: Unit at %#9llx: '%s', instance %u, phase %#x, version %u\n", 1675 2393 u.UnitHdr.offStream, u.UnitHdr.szName, u.UnitHdr.u32Instance, u.UnitHdr.u32Phase, u.UnitHdr.u32Version)); 1676 rc = ssmR3StrmWrite(&Handle , &u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[u.UnitHdr.cbName]));2394 rc = ssmR3StrmWrite(&Handle.Strm, &u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[u.UnitHdr.cbName])); 1677 2395 if (RT_SUCCESS(rc)) 1678 2396 { … … 1708 2426 TermRec.u8TypeAndFlags = SSM_REC_FLAGS_FIXED | SSM_REC_FLAGS_IMPORTANT | SSM_REC_TYPE_TERM; 1709 2427 TermRec.cbRec = sizeof(TermRec) - 2; 1710 if (Handle. fChecksummed)2428 if (Handle.Strm.fChecksummed) 1711 2429 { 1712 2430 TermRec.fFlags = SSMRECTERM_FLAGS_CRC32; 1713 TermRec.u32StreamCRC = RTCrc32Finish(Handle.u32StreamCRC);2431 TermRec.u32StreamCRC = ssmR3StrmFinalCRC(&Handle.Strm); 1714 2432 } 1715 2433 else … … 1793 2511 /* Write the end unit. */ 1794 2512 memcpy(&u.UnitHdr.szMagic[0], SSMFILEUNITHDR_END, sizeof(u.UnitHdr.szMagic)); 1795 u.UnitHdr.offStream = ssmR3StrmTell(&Handle );1796 u.UnitHdr.u32CurStreamCRC = Handle.u32StreamCRC;2513 u.UnitHdr.offStream = ssmR3StrmTell(&Handle.Strm); 2514 u.UnitHdr.u32CurStreamCRC = ssmR3StrmCurCRC(&Handle.Strm); 1797 2515 u.UnitHdr.u32CRC = 0; 1798 2516 u.UnitHdr.u32Version = 0; … … 1803 2521 u.UnitHdr.u32CRC = RTCrc32(&u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[0])); 1804 2522 Log(("SSM: Unit at %#9llx: END UNIT\n", u.UnitHdr.offStream)); 1805 rc = ssmR3StrmWrite(&Handle , &u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[0]));2523 rc = ssmR3StrmWrite(&Handle.Strm, &u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[0])); 1806 2524 if (RT_SUCCESS(rc)) 1807 2525 { 1808 2526 /* Write the directory for the final units and then the footer. */ 1809 2527 rc = ssmR3WriteDirectory(pVM, &Handle, &u.Footer.cDirEntries); 1810 /* Flush the stream buffers so that the CRC is up to date. */1811 if (RT_SUCCESS(rc))1812 rc = ssmR3StrmFlush(&Handle);1813 2528 if (RT_SUCCESS(rc)) 1814 2529 { 1815 2530 memcpy(u.Footer.szMagic, SSMFILEFTR_MAGIC, sizeof(u.Footer.szMagic)); 1816 u.Footer.offStream = ssmR3StrmTell(&Handle );1817 u.Footer.u32StreamCRC = Handle.fChecksummed ? RTCrc32Finish(Handle.u32StreamCRC) : 0;2531 u.Footer.offStream = ssmR3StrmTell(&Handle.Strm); 2532 u.Footer.u32StreamCRC = ssmR3StrmFinalCRC(&Handle.Strm); 1818 2533 u.Footer.u32Reserved = 0; 1819 2534 u.Footer.u32CRC = 0; 1820 2535 u.Footer.u32CRC = RTCrc32(&u.Footer, sizeof(u.Footer)); 1821 2536 Log(("SSM: Footer at %#9llx: \n", u.Footer.offStream)); 1822 rc = ssmR3StrmWrite(&Handle , &u.Footer, sizeof(u.Footer));2537 rc = ssmR3StrmWrite(&Handle.Strm, &u.Footer, sizeof(u.Footer)); 1823 2538 if (RT_SUCCESS(rc)) 1824 2539 { 1825 rc = RTFileClose(Handle.hFile); 1826 AssertRC(rc); 1827 if (pfnProgress) 1828 pfnProgress(pVM, 100, pvUser); 1829 Log(("SSM: Successfully saved the vm state to '%s'.\n", pszFilename)); 1830 return VINF_SUCCESS; 2540 rc = ssmR3StrmClose(&Handle.Strm); 2541 if (RT_SUCCESS(rc)) 2542 { 2543 if (pfnProgress) 2544 pfnProgress(pVM, 100, pvUser); 2545 Log(("SSM: Successfully saved the vm state to '%s'.\n", pszFilename)); 2546 return VINF_SUCCESS; 2547 } 2548 return rc; 1831 2549 } 1832 2550 } … … 1837 2555 1838 2556 /* 1839 * Delete the file on failure and destroy any compressors. 1840 */ 1841 int rc2 = RTFileClose(Handle.hFile); 1842 AssertRC(rc2); 1843 rc2 = RTFileDelete(pszFilename); 2557 * Delete the file on failure. 2558 */ 2559 ssmR3StrmClose(&Handle.Strm); 2560 int rc2 = RTFileDelete(pszFilename); 1844 2561 AssertRC(rc2); 1845 2562 … … 1929 2646 { 1930 2647 /* 1931 * Read the header.2648 * Read and check the header magic. 1932 2649 */ 1933 2650 union … … 1937 2654 SSMFILEHDRV11 v1_1; 1938 2655 } uHdr; 1939 int rc = RTFileRead(pSSM->hFile, &uHdr, sizeof(uHdr), NULL);2656 int rc = ssmR3StrmRead(&pSSM->Strm, &uHdr, sizeof(uHdr.v2_0.szMagic)); 1940 2657 if (RT_FAILURE(rc)) 1941 2658 { 1942 LogRel(("SSM: Failed to read file header. rc=%Rrc\n", rc));2659 LogRel(("SSM: Failed to read file magic header. rc=%Rrc\n", rc)); 1943 2660 return rc; 1944 2661 } 1945 1946 /*1947 * Verify the magic and make adjustments for versions differences.1948 */1949 2662 if (memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_BASE, sizeof(SSMFILEHDR_MAGIC_BASE) - 1)) 1950 2663 { … … 1952 2665 return VERR_SSM_INTEGRITY_MAGIC; 1953 2666 } 1954 if (memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_V1_X, sizeof(SSMFILEHDR_MAGIC_V1_X) - 1)) 2667 2668 /* 2669 * Find the header size and read the rest. 2670 */ 2671 static const struct 2672 { 2673 char szMagic[sizeof(SSMFILEHDR_MAGIC_V2_0)]; 2674 size_t cbHdr; 2675 unsigned uFmtVerMajor; 2676 unsigned uFmtVerMinor; 2677 } s_aVers[] = 2678 { 2679 { SSMFILEHDR_MAGIC_V2_0, sizeof(SSMFILEHDR), 2, 0 }, 2680 { SSMFILEHDR_MAGIC_V1_2, sizeof(SSMFILEHDRV12), 1, 2 }, 2681 { SSMFILEHDR_MAGIC_V1_1, sizeof(SSMFILEHDRV11), 1, 1 }, 2682 }; 2683 int iVer = RT_ELEMENTS(s_aVers); 2684 while (iVer-- > 0) 2685 if (!memcmp(uHdr.v2_0.szMagic, s_aVers[iVer].szMagic, sizeof(uHdr.v2_0.szMagic))) 2686 break; 2687 if (iVer < 0) 2688 { 2689 Log(("SSM: Unknown file format version. magic=%.*s\n", sizeof(uHdr.v2_0.szMagic) - 1, uHdr.v2_0.szMagic)); 2690 return VERR_SSM_INTEGRITY_VERSION; 2691 } 2692 pSSM->u.Read.uFmtVerMajor = s_aVers[iVer].uFmtVerMajor; 2693 pSSM->u.Read.uFmtVerMinor = s_aVers[iVer].uFmtVerMinor; 2694 pSSM->u.Read.cbFileHdr = s_aVers[iVer].cbHdr; 2695 2696 rc = ssmR3StrmRead(&pSSM->Strm, (uint8_t *)&uHdr + sizeof(uHdr.v2_0.szMagic), pSSM->u.Read.cbFileHdr - sizeof(uHdr.v2_0.szMagic)); 2697 if (RT_FAILURE(rc)) 2698 { 2699 LogRel(("SSM: Failed to read the file header. rc=%Rrc\n", rc)); 2700 return rc; 2701 } 2702 2703 /* 2704 * Make version specific adjustments. 2705 */ 2706 if (pSSM->u.Read.uFmtVerMajor >= 2) 1955 2707 { 1956 2708 /* … … 1958 2710 */ 1959 2711 bool fChecksummed; 1960 if ( !memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_V2_0, sizeof(SSMFILEHDR_MAGIC_V2_0)))2712 if (pSSM->u.Read.uFmtVerMinor == 0) 1961 2713 { 1962 2714 /* validate the header. */ … … 1981 2733 1982 2734 /* set the header info. */ 1983 pSSM->u.Read.uFmtVerMajor = 2;1984 pSSM->u.Read.uFmtVerMinor = 2;1985 pSSM->u.Read.cbFileHdr = sizeof(uHdr.v2_0);1986 2735 pSSM->u.Read.cHostBits = uHdr.v2_0.cHostBits; 1987 2736 pSSM->u.Read.u16VerMajor = uHdr.v2_0.u16VerMajor; … … 1995 2744 } 1996 2745 else 1997 { 1998 Log(("SSM: Unknown file format version. magic=%.*s\n", sizeof(uHdr.v2_0.szMagic) - 1, uHdr.v2_0.szMagic)); 1999 return VERR_SSM_INTEGRITY_VERSION; 2000 } 2001 2002 if (fChecksummed && fChecksumOnRead) 2003 pSSM->u32StreamCRC = RTCrc32Process(RTCrc32Start(), &uHdr, pSSM->u.Read.cbFileHdr); 2746 AssertFailedReturn(VERR_INTERNAL_ERROR); 2004 2747 2005 2748 /* 2006 2749 * Read and validate the footer. 2007 2750 */ 2751 /** @todo this needs to be fixed when starting to buffer input using a separate thread. */ 2752 uint64_t offRestore = RTFileTell(pSSM->Strm.hFile); 2008 2753 SSMFILEFTR Footer; 2009 2754 uint64_t offFooter; 2010 rc = RTFileSeek(pSSM-> hFile, -(RTFOFF)sizeof(Footer), RTFILE_SEEK_END, &offFooter);2755 rc = RTFileSeek(pSSM->Strm.hFile, -(RTFOFF)sizeof(Footer), RTFILE_SEEK_END, &offFooter); 2011 2756 AssertLogRelRCReturn(rc, rc); 2012 rc = RTFileRead(pSSM-> hFile, &Footer, sizeof(Footer), NULL);2757 rc = RTFileRead(pSSM->Strm.hFile, &Footer, sizeof(Footer), NULL); 2013 2758 AssertLogRelRCReturn(rc, rc); 2014 2759 if (memcmp(Footer.szMagic, SSMFILEFTR_MAGIC, sizeof(Footer.szMagic))) … … 2052 2797 && !fChecksumOnRead) 2053 2798 { 2054 rc = RTFileSeek(pSSM-> hFile, 0, RTFILE_SEEK_BEGIN, NULL);2799 rc = RTFileSeek(pSSM->Strm.hFile, 0, RTFILE_SEEK_BEGIN, NULL); 2055 2800 AssertLogRelRCReturn(rc, rc); 2056 2801 uint32_t u32CRC; 2057 rc = ssmR3CalcChecksum(pSSM-> hFile, offFooter, &u32CRC);2802 rc = ssmR3CalcChecksum(pSSM->Strm.hFile, offFooter, &u32CRC); 2058 2803 if (RT_FAILURE(rc)) 2059 2804 return rc; … … 2064 2809 } 2065 2810 } 2811 2812 rc = RTFileSeek(pSSM->Strm.hFile, RTFILE_SEEK_BEGIN, offRestore, NULL); 2813 AssertRCReturn(rc, rc); 2066 2814 } 2067 2815 else … … 2074 2822 RTUUID MachineUuidFromHdr; 2075 2823 2076 pSSM->fChecksummed = false; 2077 pSSM->u.Read.uFmtVerMajor = 1; 2078 if (!memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_V1_1, sizeof(SSMFILEHDR_MAGIC_V1_1))) 2079 { 2080 pSSM->u.Read.uFmtVerMinor = 1; 2081 pSSM->u.Read.cbFileHdr = sizeof(uHdr.v1_1); 2824 ssmR3StrmDisableChecksumming(&pSSM->Strm); 2825 if (pSSM->u.Read.uFmtVerMinor == 1) 2826 { 2082 2827 pSSM->u.Read.cHostBits = 0; /* unknown */ 2083 2828 pSSM->u.Read.u16VerMajor = 0; … … 2094 2839 fHaveHostBits = false; 2095 2840 } 2096 else if (!memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_V1_2, sizeof(SSMFILEHDR_MAGIC_V1_2))) 2097 { 2098 pSSM->u.Read.uFmtVerMinor = 2; 2099 pSSM->u.Read.cbFileHdr = sizeof(uHdr.v1_2); 2841 else if (pSSM->u.Read.uFmtVerMinor == 2) 2842 { 2100 2843 pSSM->u.Read.cHostBits = uHdr.v1_2.cHostBits; 2101 2844 pSSM->u.Read.u16VerMajor = uHdr.v1_2.u16VerMajor; … … 2113 2856 } 2114 2857 else 2115 { 2116 LogRel(("SSM: Unknown file format version. magic=%.*s\n", sizeof(uHdr.v2_0.szMagic) - 1, uHdr.v2_0.szMagic)); 2117 return VERR_SSM_INTEGRITY_VERSION; 2118 } 2858 AssertFailedReturn(VERR_INTERNAL_ERROR); 2119 2859 2120 2860 /* … … 2130 2870 * Verify the file size. 2131 2871 */ 2132 uint64_t cbFile; 2133 rc = RTFileGetSize(pSSM->hFile, &cbFile); 2134 AssertLogRelRCReturn(rc, rc); 2872 uint64_t cbFile = ssmR3StrmGetSize(&pSSM->Strm); 2135 2873 if (cbFile != pSSM->u.Read.cbLoadFile) 2136 2874 { … … 2156 2894 || fChecksumOnRead) 2157 2895 { 2158 rc = RTFileSeek(pSSM->hFile, RT_OFFSETOF(SSMFILEHDRV11, u32CRC) + sizeof(uHdr.v1_1.u32CRC), RTFILE_SEEK_BEGIN, NULL); 2896 /** @todo this needs to be fixed when starting to buffer input using a separate thread. */ 2897 uint64_t offRestore = RTFileTell(pSSM->Strm.hFile); 2898 rc = RTFileSeek(pSSM->Strm.hFile, RT_OFFSETOF(SSMFILEHDRV11, u32CRC) + sizeof(uHdr.v1_1.u32CRC), RTFILE_SEEK_BEGIN, NULL); 2159 2899 AssertLogRelRCReturn(rc, rc); 2160 2900 uint32_t u32CRC; 2161 rc = ssmR3CalcChecksum(pSSM-> hFile, cbFile - pSSM->u.Read.cbFileHdr, &u32CRC);2901 rc = ssmR3CalcChecksum(pSSM->Strm.hFile, cbFile - pSSM->u.Read.cbFileHdr, &u32CRC); 2162 2902 if (RT_FAILURE(rc)) 2163 2903 return rc; … … 2167 2907 return VERR_SSM_INTEGRITY_CRC; 2168 2908 } 2909 rc = RTFileSeek(pSSM->Strm.hFile, RTFILE_SEEK_BEGIN, offRestore, NULL); 2910 AssertRCReturn(rc, rc); 2169 2911 } 2170 2912 } … … 2195 2937 * Initialize the handle. 2196 2938 */ 2197 pSSM->hFile = NIL_RTFILE;2198 2939 pSSM->pVM = pVM; 2199 2940 pSSM->enmOp = SSMSTATE_INVALID; … … 2211 2952 pSSM->uPercentPrepare = 5; 2212 2953 pSSM->uPercentDone = 2; 2213 pSSM->fChecksummed = fChecksumOnRead;2214 pSSM->u32StreamCRC = 0;2215 2954 2216 2955 pSSM->u.Read.pZipDecompV1 = NULL; … … 2237 2976 * Try open and validate the file. 2238 2977 */ 2239 int rc = RTFileOpen(&pSSM->hFile, pszFilename, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE);2978 int rc = ssmR3StrmOpenFile(&pSSM->Strm, pszFilename, false /*fWrite*/, fChecksumOnRead); 2240 2979 if (RT_SUCCESS(rc)) 2241 2980 { 2242 2981 rc = ssmR3ValidateFile(pSSM, fChecksumIt, fChecksumOnRead); 2243 2982 if (RT_SUCCESS(rc)) 2244 { 2245 rc = RTFileSeek(pSSM->hFile, pSSM->u.Read.cbFileHdr, RTFILE_SEEK_BEGIN, NULL); 2246 if (RT_SUCCESS(rc)) 2247 return rc; 2248 } 2249 2250 RTFileClose(pSSM->hFile); 2983 return rc; 2984 2985 /* failure path */ 2986 ssmR3StrmClose(&pSSM->Strm); 2251 2987 } 2252 2988 else 2253 2989 Log(("SSM: Failed to open save state file '%s', rc=%Rrc.\n", pszFilename, rc)); 2254 pSSM->hFile = NIL_RTFILE;2255 2990 return rc; 2256 2991 } … … 2298 3033 * Save the current file position and read the data unit header. 2299 3034 */ 2300 uint64_t offUnit = ssmR3StrmTell( pSSM);3035 uint64_t offUnit = ssmR3StrmTell(&pSSM->Strm); 2301 3036 SSMFILEUNITHDRV1 UnitHdr; 2302 rc = ssmR3StrmRead( pSSM, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));3037 rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName)); 2303 3038 if (RT_SUCCESS(rc)) 2304 3039 { … … 2334 3069 if (pszName) 2335 3070 { 2336 rc = ssmR3StrmRead( pSSM, pszName, UnitHdr.cchName);3071 rc = ssmR3StrmRead(&pSSM->Strm, pszName, UnitHdr.cchName); 2337 3072 if (RT_SUCCESS(rc)) 2338 3073 { … … 2395 3130 * really bad ones here. 2396 3131 */ 2397 uint64_t off = ssmR3StrmTell( pSSM);3132 uint64_t off = ssmR3StrmTell(&pSSM->Strm); 2398 3133 int64_t i64Diff = off - (offUnit + UnitHdr.cbUnit); 2399 3134 if (i64Diff < 0) 2400 3135 { 2401 3136 Log(("SSM: Unit '%s' left %lld bytes unread!\n", pszName, -i64Diff)); 2402 rc = RTFileSeek(pSSM->hFile, offUnit + UnitHdr.cbUnit, RTFILE_SEEK_BEGIN, NULL);3137 rc = ssmR3StrmSkipTo(&pSSM->Strm, offUnit + UnitHdr.cbUnit); 2403 3138 ssmR3Progress(pSSM, offUnit + UnitHdr.cbUnit - pSSM->offEst); 2404 3139 } … … 2429 3164 if (pSSM->enmAfter != SSMAFTER_DEBUG_IT) 2430 3165 break; 2431 rc = RTFileSeek(pSSM->hFile, offUnit + UnitHdr.cbUnit, RTFILE_SEEK_BEGIN, NULL);3166 rc = ssmR3StrmSkipTo(&pSSM->Strm, offUnit + UnitHdr.cbUnit); 2432 3167 } 2433 3168 } … … 2467 3202 * Read the unit header and check its integrity. 2468 3203 */ 2469 uint64_t offUnit = ssmR3StrmTell( pSSM);3204 uint64_t offUnit = ssmR3StrmTell(&pSSM->Strm); 2470 3205 uint32_t u32CurStreamCRC = pSSM->u32StreamCRC; 2471 3206 SSMFILEUNITHDR UnitHdr; 2472 int rc = ssmR3StrmRead( pSSM, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));3207 int rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName)); 2473 3208 if (RT_FAILURE(rc)) 2474 3209 return rc; … … 2488 3223 return VERR_SSM_INTEGRITY; 2489 3224 } 2490 rc = ssmR3StrmRead( pSSM, &UnitHdr.szName[0], UnitHdr.cbName);3225 rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr.szName[0], UnitHdr.cbName); 2491 3226 if (RT_FAILURE(rc)) 2492 3227 return rc; … … 2746 3481 * Done 2747 3482 */ 2748 int rc2 = RTFileClose(Handle.hFile); 2749 AssertRC(rc2); 3483 ssmR3StrmClose(&Handle.Strm); 2750 3484 if (RT_SUCCESS(rc)) 2751 3485 { … … 2783 3517 int rc = ssmR3OpenFile(NULL, pszFilename, fChecksumIt, false /*fChecksumOnRead*/, &Handle); 2784 3518 if (RT_SUCCESS(rc)) 2785 RTFileClose(Handle.hFile);3519 ssmR3StrmClose(&Handle.Strm); 2786 3520 else 2787 3521 Log(("SSM: Failed to open saved state file '%s', rc=%Rrc.\n", pszFilename, rc)); … … 2859 3593 2860 3594 /* 2861 * Close the file and free the handle. 2862 */ 2863 int rc = RTFileClose(pSSM->hFile); 2864 AssertRC(rc); 3595 * Close the stream and free the handle. 3596 */ 3597 ssmR3StrmClose(&Handle.Strm); 2865 3598 if (pSSM->u.Read.pZipDecompV1) 2866 3599 { … … 3161 3894 { 3162 3895 Log2(("ssmR3DataWriteRaw: %08llx|%08llx: pvBuf=%p cbBuf=%#x %.*Rhxs%s\n", 3163 ssmR3StrmTell( pSSM), pSSM->offUnit, pvBuf, cbBuf, RT_MIN(cbBuf, SSM_LOG_BYTES), pvBuf, cbBuf > SSM_LOG_BYTES ? "..." : ""));3896 ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pvBuf, cbBuf, RT_MIN(cbBuf, SSM_LOG_BYTES), pvBuf, cbBuf > SSM_LOG_BYTES ? "..." : "")); 3164 3897 3165 3898 /* … … 3175 3908 { 3176 3909 size_t cbChunk = RT_MIN(cbBuf, _1M); 3177 int rc = ssmR3StrmWrite( pSSM, pvBuf, cbChunk);3910 int rc = ssmR3StrmWrite(&pSSM->Strm, pvBuf, cbChunk); 3178 3911 if (RT_FAILURE(rc)) 3179 3912 return rc; … … 3250 3983 3251 3984 Log3(("ssmR3DataWriteRecHdr: %08llx|%08llx/%08x: Type=%02x fImportant=%RTbool cbHdr=%u\n", 3252 ssmR3StrmTell( pSSM) + cbHdr, pSSM->offUnit + cbHdr, cb, u8TypeAndFlags & SSM_REC_TYPE_MASK, !!(u8TypeAndFlags & SSM_REC_FLAGS_IMPORTANT), cbHdr));3985 ssmR3StrmTell(&pSSM->Strm) + cbHdr, pSSM->offUnit + cbHdr, cb, u8TypeAndFlags & SSM_REC_TYPE_MASK, !!(u8TypeAndFlags & SSM_REC_FLAGS_IMPORTANT), cbHdr)); 3253 3986 3254 3987 return ssmR3DataWriteRaw(pSSM, &abHdr[0], cbHdr); … … 3329 4062 3330 4063 Log3(("ssmR3DataFlushBuffer: %08llx|%08llx/%08x: Type=RAW fImportant=true cbHdr=%u\n", 3331 ssmR3StrmTell( pSSM) + (cbTotal - cb), pSSM->offUnit + (cbTotal - cb), cb, (cbTotal - cb) ));4064 ssmR3StrmTell(&pSSM->Strm) + (cbTotal - cb), pSSM->offUnit + (cbTotal - cb), cb, (cbTotal - cb) )); 3332 4065 3333 4066 /* … … 3949 4682 { 3950 4683 //Log2(("ssmR3ReadInV1: %#010llx cbBug=%#x cbRead=%#x\n", RTFileTell(pSSM->File), cbBuf, cbRead)); 3951 int rc = ssmR3StrmRead( pSSM, pvBuf, cbRead);4684 int rc = ssmR3StrmRead(&pSSM->Strm, pvBuf, cbRead); 3952 4685 if (RT_SUCCESS(rc)) 3953 4686 { … … 4060 4793 { 4061 4794 /** @todo buffered reads! (at some level or another) */ 4062 int rc = ssmR3StrmRead( pSSM, pvBuf, cbToRead);4795 int rc = ssmR3StrmRead(&pSSM->Strm, pvBuf, cbToRead); 4063 4796 if (RT_SUCCESS(rc)) 4064 4797 { … … 4159 4892 * Read the two mandatory bytes. 4160 4893 */ 4161 uint32_t u32StreamCRC = pSSM->u32StreamCRC;4894 uint32_t u32StreamCRC = ssmR3StrmCurCRC(pSSM); /** @todo NOT good. */ 4162 4895 uint8_t abHdr[8]; 4163 4896 int rc = ssmR3DataReadV2Raw(pSSM, abHdr, 2); … … 4190 4923 if (!(TermRec.fFlags & SSMRECTERM_FLAGS_CRC32)) 4191 4924 AssertLogRelMsgReturn(TermRec.u32StreamCRC == 0, ("%#x\n", TermRec.u32StreamCRC), VERR_SSM_INTEGRITY); 4192 else if (pSSM-> fChecksummed)4925 else if (pSSM->Strm.fChecksummed) 4193 4926 { 4194 4927 u32StreamCRC = RTCrc32Finish(u32StreamCRC); … … 4196 4929 } 4197 4930 4198 Log3(("ssmR3DataReadRecHdrV2: %08llx|%08llx: TERM\n", ssmR3StrmTell( pSSM) - sizeof(SSMRECTERM), pSSM->offUnit));4931 Log3(("ssmR3DataReadRecHdrV2: %08llx|%08llx: TERM\n", ssmR3StrmTell(&pSSM->Strm) - sizeof(SSMRECTERM), pSSM->offUnit)); 4199 4932 return VINF_SUCCESS; 4200 4933 } … … 4303 5036 4304 5037 Log3(("ssmR3DataReadRecHdrV2: %08llx|%08llx/%08x: Type=%02x fImportant=%RTbool cbHdr=%u\n", 4305 ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft,5038 ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, 4306 5039 pSSM->u.Read.u8TypeAndFlags & SSM_REC_TYPE_MASK, 4307 5040 !!(pSSM->u.Read.u8TypeAndFlags & SSM_REC_FLAGS_IMPORTANT), … … 4329 5062 uint32_t off = pSSM->u.Read.offDataBuffer; 4330 5063 int32_t cbInBuffer = pSSM->u.Read.cbDataBuffer - off; 4331 Log4(("ssmR3DataReadUnbufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x\n", ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft, cbInBuffer, cbBufOrg));5064 Log4(("ssmR3DataReadUnbufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x\n", ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, cbInBuffer, cbBufOrg)); 4332 5065 if (cbInBuffer > 0) 4333 5066 { … … 4409 5142 4410 5143 Log4(("ssmR3DataReadUnBufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x %.*Rhxs%s\n", 4411 ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft, 0, cbBufOrg, RT_MIN(SSM_LOG_BYTES, cbBufOrg), pvBufOrg, cbBufOrg > SSM_LOG_BYTES ? "..." : ""));5144 ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, 0, cbBufOrg, RT_MIN(SSM_LOG_BYTES, cbBufOrg), pvBufOrg, cbBufOrg > SSM_LOG_BYTES ? "..." : "")); 4412 5145 return VINF_SUCCESS; 4413 5146 } … … 4431 5164 uint32_t off = pSSM->u.Read.offDataBuffer; 4432 5165 int32_t cbInBuffer = pSSM->u.Read.cbDataBuffer - off; 4433 Log4(("ssmR3DataReadBufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x\n", ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft, cbInBuffer, cbBufOrg));5166 Log4(("ssmR3DataReadBufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x\n", ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, cbInBuffer, cbBufOrg)); 4434 5167 if (cbInBuffer > 0) 4435 5168 { … … 4507 5240 4508 5241 Log4(("ssmR3DataReadBufferedV2: %08llx|%08llx/%08x/%08x: cbBuf=%#x %.*Rhxs%s\n", 4509 ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft, pSSM->u.Read.cbDataBuffer - pSSM->u.Read.offDataBuffer,5242 ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, pSSM->u.Read.cbDataBuffer - pSSM->u.Read.offDataBuffer, 4510 5243 cbBufOrg, RT_MIN(SSM_LOG_BYTES, cbBufOrg), pvBufOrg, cbBufOrg > SSM_LOG_BYTES ? "..." : "")); 4511 5244 return VINF_SUCCESS; … … 4547 5280 ? "ssmR3DataRead: %08llx|%08llx/%08x/%08x: cbBuf=%#x %.*Rhxs%s\n" 4548 5281 : "ssmR3DataRead: %08llx|%08llx/%08x/%08x: cbBuf=%#x\n", 4549 ssmR3StrmTell( pSSM), pSSM->offUnit, pSSM->u.Read.cbRecLeft, pSSM->u.Read.cbDataBuffer - pSSM->u.Read.offDataBuffer,5282 ssmR3StrmTell(&pSSM->Strm), pSSM->offUnit, pSSM->u.Read.cbRecLeft, pSSM->u.Read.cbDataBuffer - pSSM->u.Read.offDataBuffer, 4550 5283 cbBuf, RT_MIN(SSM_LOG_BYTES, cbBuf), pvBuf, cbBuf > SSM_LOG_BYTES ? "..." : "")); 4551 5284
Note:
See TracChangeset
for help on using the changeset viewer.