VirtualBox

Changeset 21866 in vbox for trunk/src/VBox/VMM


Ignore:
Timestamp:
Jul 29, 2009 4:15:42 PM (16 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
50509
Message:

SSM: Incomplete buffering code (have to go).

File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/VMM/SSM-new.cpp

    r21862 r21866  
    152152#include <iprt/param.h>
    153153#include <iprt/thread.h>
     154#include <iprt/semaphore.h>
    154155#include <iprt/string.h>
    155156#include <iprt/uuid.h>
     
    281282} SSMSTATE;
    282283
     284/** Pointer to a SSM stream buffer. */
     285typedef struct SSMSTRMBUF *PSSMSTRMBUF;
     286/**
     287 * A SSM stream buffer.
     288 */
     289typedef 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 */
     310typedef 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. */
     352typedef SSMSTRM *PSSMSTRM;
     353
    283354
    284355/**
     
    287358typedef struct SSMHANDLE
    288359{
    289     /** The file handle. */
    290     RTFILE          hFile;
     360    /** Stream/buffer manager. */
     361    SSMSTRM         Strm;
     362
    291363    /** The VM handle. */
    292364    PVM             pVM;
     
    321393    /** the amount of % we reserve for the 'done' stage */
    322394    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;
    328395
    329396    union
     
    12811348
    12821349/**
     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 */
     1357static 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 */
     1421static 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 */
     1439static 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 */
     1468static 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 */
     1501DECLINLINE(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 */
     1516static 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 */
     1540static 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 */
     1577static 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 */
     1599static 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 */
     1622static 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 */
     1657static 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 */
     1706static 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 */
     1758static 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/**
    12831777 * Stream output routine.
    12841778 *
    12851779 * @returns VBox status code.
    1286  * @param   pSSM        The SSM handle.
     1780 * @param   pStrm       The stream handle.
    12871781 * @param   pvBuf       What to write.
    12881782 * @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 */
     1786static 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 */
     1856static 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);
    12941865    if (RT_SUCCESS(rc))
    12951866    {
    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    }
    13041884    return rc;
    13051885}
     
    13101890 *
    13111891 * @returns VBox status code.
    1312  * @param   pSSM        The SSM handle.
     1892 * @param   pStrm       The stream handle.
    13131893 * @param   pvBuf       Where to put what we read.
    13141894 * @param   cbToRead    How much to read.
    13151895 */
    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;
     1896static 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
    13301973    return rc;
    13311974}
     
    13331976
    13341977/**
    1335  * Flush any buffered data.
     1978 * Tell current stream position.
     1979 *
     1980 * @returns stream position.
     1981 * @param   pStrm       The stream handle.
     1982 */
     1983static 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 */
     1995static 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 */
     2016static 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 */
     2029static 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.
    13362040 *
    13372041 * @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 */
     2045static 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 */
     2072static 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;
    13562078}
    13572079
     
    14842206    cbDir = RT_OFFSETOF(SSMFILEDIR, aEntries[pDir->cEntries]);
    14852207    pDir->u32CRC = RTCrc32(pDir, cbDir);
    1486     int rc = ssmR3StrmWrite(pSSM, pDir, cbDir);
     2208    int rc = ssmR3StrmWrite(&pSSM->Strm, pDir, cbDir);
    14872209    RTMemTmpFree(pDir);
    14882210    return rc;
     
    15252247     * the saved state file might take a long time.
    15262248     */
    1527     SSMHANDLE Handle        = {0};
    1528     Handle.hFile            = NIL_RTFILE;
     2249    SSMHANDLE Handle;
     2250    RT_ZERO(Handle);
    15292251    Handle.pVM              = pVM;
    15302252    Handle.enmOp            = SSMSTATE_INVALID;
     
    15422264    Handle.uPercentPrepare  = 20;
    15432265    Handle.uPercentDone     = 2;
    1544     Handle.fChecksummed     = true;
    1545     Handle.u32StreamCRC     = RTCrc32Start();
    15462266    Handle.u.Write.offDataBuffer    = 0;
    15472267
    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*/);
    15492269    if (RT_FAILURE(rc))
    15502270    {
     
    15792299    u.FileHdr.u32CRC       = 0;
    15802300    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));
    15822302    if (RT_SUCCESS(rc))
    15832303    {
     
    16542374                    continue;
    16552375                }
    1656                 pUnit->offStream = ssmR3StrmTell(&Handle);
     2376                pUnit->offStream = ssmR3StrmTell(&Handle.Strm);
    16572377
    16582378                /*
     
    16612381                memcpy(&u.UnitHdr.szMagic[0], SSMFILEUNITHDR_MAGIC, sizeof(u.UnitHdr.szMagic));
    16622382                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);
    16662384                u.UnitHdr.u32CRC          = 0;
    16672385                u.UnitHdr.u32Version      = pUnit->u32Version;
     
    16742392                Log(("SSM: Unit at %#9llx: '%s', instance %u, phase %#x, version %u\n",
    16752393                     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]));
    16772395                if (RT_SUCCESS(rc))
    16782396                {
     
    17082426                        TermRec.u8TypeAndFlags = SSM_REC_FLAGS_FIXED | SSM_REC_FLAGS_IMPORTANT | SSM_REC_TYPE_TERM;
    17092427                        TermRec.cbRec       = sizeof(TermRec) - 2;
    1710                         if (Handle.fChecksummed)
     2428                        if (Handle.Strm.fChecksummed)
    17112429                        {
    17122430                            TermRec.fFlags       = SSMRECTERM_FLAGS_CRC32;
    1713                             TermRec.u32StreamCRC = RTCrc32Finish(Handle.u32StreamCRC);
     2431                            TermRec.u32StreamCRC = ssmR3StrmFinalCRC(&Handle.Strm);
    17142432                        }
    17152433                        else
     
    17932511            /* Write the end unit. */
    17942512            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);
    17972515            u.UnitHdr.u32CRC          = 0;
    17982516            u.UnitHdr.u32Version      = 0;
     
    18032521            u.UnitHdr.u32CRC          = RTCrc32(&u.UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName[0]));
    18042522            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]));
    18062524            if (RT_SUCCESS(rc))
    18072525            {
    18082526                /* Write the directory for the final units and then the footer. */
    18092527                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);
    18132528                if (RT_SUCCESS(rc))
    18142529                {
    18152530                    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);
    18182533                    u.Footer.u32Reserved  = 0;
    18192534                    u.Footer.u32CRC       = 0;
    18202535                    u.Footer.u32CRC       = RTCrc32(&u.Footer, sizeof(u.Footer));
    18212536                    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));
    18232538                    if (RT_SUCCESS(rc))
    18242539                    {
    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;
    18312549                    }
    18322550                }
     
    18372555
    18382556    /*
    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);
    18442561    AssertRC(rc2);
    18452562
     
    19292646{
    19302647    /*
    1931      * Read the header.
     2648     * Read and check the header magic.
    19322649     */
    19332650    union
     
    19372654        SSMFILEHDRV11       v1_1;
    19382655    } uHdr;
    1939     int rc = RTFileRead(pSSM->hFile, &uHdr, sizeof(uHdr), NULL);
     2656    int rc = ssmR3StrmRead(&pSSM->Strm, &uHdr, sizeof(uHdr.v2_0.szMagic));
    19402657    if (RT_FAILURE(rc))
    19412658    {
    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));
    19432660        return rc;
    19442661    }
    1945 
    1946     /*
    1947      * Verify the magic and make adjustments for versions differences.
    1948      */
    19492662    if (memcmp(uHdr.v2_0.szMagic, SSMFILEHDR_MAGIC_BASE, sizeof(SSMFILEHDR_MAGIC_BASE) - 1))
    19502663    {
     
    19522665        return VERR_SSM_INTEGRITY_MAGIC;
    19532666    }
    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)
    19552707    {
    19562708        /*
     
    19582710         */
    19592711        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)
    19612713        {
    19622714            /* validate the header. */
     
    19812733
    19822734            /* 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);
    19862735            pSSM->u.Read.cHostBits      = uHdr.v2_0.cHostBits;
    19872736            pSSM->u.Read.u16VerMajor    = uHdr.v2_0.u16VerMajor;
     
    19952744        }
    19962745        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);
    20042747
    20052748        /*
    20062749         * Read and validate the footer.
    20072750         */
     2751/** @todo this needs to be fixed when starting to buffer input using a separate thread. */
     2752        uint64_t    offRestore = RTFileTell(pSSM->Strm.hFile);
    20082753        SSMFILEFTR  Footer;
    20092754        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);
    20112756        AssertLogRelRCReturn(rc, rc);
    2012         rc = RTFileRead(pSSM->hFile, &Footer, sizeof(Footer), NULL);
     2757        rc = RTFileRead(pSSM->Strm.hFile, &Footer, sizeof(Footer), NULL);
    20132758        AssertLogRelRCReturn(rc, rc);
    20142759        if (memcmp(Footer.szMagic, SSMFILEFTR_MAGIC, sizeof(Footer.szMagic)))
     
    20522797            &&  !fChecksumOnRead)
    20532798        {
    2054             rc = RTFileSeek(pSSM->hFile, 0, RTFILE_SEEK_BEGIN, NULL);
     2799            rc = RTFileSeek(pSSM->Strm.hFile, 0, RTFILE_SEEK_BEGIN, NULL);
    20552800            AssertLogRelRCReturn(rc, rc);
    20562801            uint32_t u32CRC;
    2057             rc = ssmR3CalcChecksum(pSSM->hFile, offFooter, &u32CRC);
     2802            rc = ssmR3CalcChecksum(pSSM->Strm.hFile, offFooter, &u32CRC);
    20582803            if (RT_FAILURE(rc))
    20592804                return rc;
     
    20642809            }
    20652810        }
     2811
     2812        rc = RTFileSeek(pSSM->Strm.hFile, RTFILE_SEEK_BEGIN, offRestore, NULL);
     2813        AssertRCReturn(rc, rc);
    20662814    }
    20672815    else
     
    20742822        RTUUID      MachineUuidFromHdr;
    20752823
    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        {
    20822827            pSSM->u.Read.cHostBits      = 0; /* unknown */
    20832828            pSSM->u.Read.u16VerMajor    = 0;
     
    20942839            fHaveHostBits       = false;
    20952840        }
    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        {
    21002843            pSSM->u.Read.cHostBits      = uHdr.v1_2.cHostBits;
    21012844            pSSM->u.Read.u16VerMajor    = uHdr.v1_2.u16VerMajor;
     
    21132856        }
    21142857        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);
    21192859
    21202860        /*
     
    21302870         * Verify the file size.
    21312871         */
    2132         uint64_t cbFile;
    2133         rc = RTFileGetSize(pSSM->hFile, &cbFile);
    2134         AssertLogRelRCReturn(rc, rc);
     2872        uint64_t cbFile = ssmR3StrmGetSize(&pSSM->Strm);
    21352873        if (cbFile != pSSM->u.Read.cbLoadFile)
    21362874        {
     
    21562894            ||  fChecksumOnRead)
    21572895        {
    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);
    21592899            AssertLogRelRCReturn(rc, rc);
    21602900            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);
    21622902            if (RT_FAILURE(rc))
    21632903                return rc;
     
    21672907                return VERR_SSM_INTEGRITY_CRC;
    21682908            }
     2909            rc = RTFileSeek(pSSM->Strm.hFile, RTFILE_SEEK_BEGIN, offRestore, NULL);
     2910            AssertRCReturn(rc, rc);
    21692911        }
    21702912    }
     
    21952937     * Initialize the handle.
    21962938     */
    2197     pSSM->hFile            = NIL_RTFILE;
    21982939    pSSM->pVM              = pVM;
    21992940    pSSM->enmOp            = SSMSTATE_INVALID;
     
    22112952    pSSM->uPercentPrepare  = 5;
    22122953    pSSM->uPercentDone     = 2;
    2213     pSSM->fChecksummed     = fChecksumOnRead;
    2214     pSSM->u32StreamCRC     = 0;
    22152954
    22162955    pSSM->u.Read.pZipDecompV1   = NULL;
     
    22372976     * Try open and validate the file.
    22382977     */
    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);
    22402979    if (RT_SUCCESS(rc))
    22412980    {
    22422981        rc = ssmR3ValidateFile(pSSM, fChecksumIt, fChecksumOnRead);
    22432982        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);
    22512987    }
    22522988    else
    22532989        Log(("SSM: Failed to open save state file '%s', rc=%Rrc.\n",  pszFilename, rc));
    2254     pSSM->hFile = NIL_RTFILE;
    22552990    return rc;
    22562991}
     
    22983033         * Save the current file position and read the data unit header.
    22993034         */
    2300         uint64_t         offUnit = ssmR3StrmTell(pSSM);
     3035        uint64_t         offUnit = ssmR3StrmTell(&pSSM->Strm);
    23013036        SSMFILEUNITHDRV1 UnitHdr;
    2302         rc = ssmR3StrmRead(pSSM, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));
     3037        rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));
    23033038        if (RT_SUCCESS(rc))
    23043039        {
     
    23343069            if (pszName)
    23353070            {
    2336                 rc = ssmR3StrmRead(pSSM, pszName, UnitHdr.cchName);
     3071                rc = ssmR3StrmRead(&pSSM->Strm, pszName, UnitHdr.cchName);
    23373072                if (RT_SUCCESS(rc))
    23383073                {
     
    23953130                             * really bad ones here.
    23963131                             */
    2397                             uint64_t off = ssmR3StrmTell(pSSM);
     3132                            uint64_t off = ssmR3StrmTell(&pSSM->Strm);
    23983133                            int64_t i64Diff = off - (offUnit + UnitHdr.cbUnit);
    23993134                            if (i64Diff < 0)
    24003135                            {
    24013136                                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);
    24033138                                ssmR3Progress(pSSM, offUnit + UnitHdr.cbUnit - pSSM->offEst);
    24043139                            }
     
    24293164                        if (pSSM->enmAfter != SSMAFTER_DEBUG_IT)
    24303165                            break;
    2431                         rc = RTFileSeek(pSSM->hFile, offUnit + UnitHdr.cbUnit, RTFILE_SEEK_BEGIN, NULL);
     3166                        rc = ssmR3StrmSkipTo(&pSSM->Strm, offUnit + UnitHdr.cbUnit);
    24323167                    }
    24333168                }
     
    24673202         * Read the unit header and check its integrity.
    24683203         */
    2469         uint64_t        offUnit         = ssmR3StrmTell(pSSM);
     3204        uint64_t        offUnit         = ssmR3StrmTell(&pSSM->Strm);
    24703205        uint32_t        u32CurStreamCRC = pSSM->u32StreamCRC;
    24713206        SSMFILEUNITHDR  UnitHdr;
    2472         int rc = ssmR3StrmRead(pSSM, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));
     3207        int rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr, RT_OFFSETOF(SSMFILEUNITHDR, szName));
    24733208        if (RT_FAILURE(rc))
    24743209            return rc;
     
    24883223                return VERR_SSM_INTEGRITY;
    24893224            }
    2490             rc = ssmR3StrmRead(pSSM, &UnitHdr.szName[0], UnitHdr.cbName);
     3225            rc = ssmR3StrmRead(&pSSM->Strm, &UnitHdr.szName[0], UnitHdr.cbName);
    24913226            if (RT_FAILURE(rc))
    24923227                return rc;
     
    27463481     * Done
    27473482     */
    2748     int rc2 = RTFileClose(Handle.hFile);
    2749     AssertRC(rc2);
     3483    ssmR3StrmClose(&Handle.Strm);
    27503484    if (RT_SUCCESS(rc))
    27513485    {
     
    27833517    int rc = ssmR3OpenFile(NULL, pszFilename, fChecksumIt, false /*fChecksumOnRead*/, &Handle);
    27843518    if (RT_SUCCESS(rc))
    2785         RTFileClose(Handle.hFile);
     3519        ssmR3StrmClose(&Handle.Strm);
    27863520    else
    27873521        Log(("SSM: Failed to open saved state file '%s', rc=%Rrc.\n",  pszFilename, rc));
     
    28593593
    28603594    /*
    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);
    28653598    if (pSSM->u.Read.pZipDecompV1)
    28663599    {
     
    31613894{
    31623895    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 ? "..." : ""));
    31643897
    31653898    /*
     
    31753908    {
    31763909        size_t cbChunk = RT_MIN(cbBuf, _1M);
    3177         int rc = ssmR3StrmWrite(pSSM, pvBuf, cbChunk);
     3910        int rc = ssmR3StrmWrite(&pSSM->Strm, pvBuf, cbChunk);
    31783911        if (RT_FAILURE(rc))
    31793912            return rc;
     
    32503983
    32513984    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));
    32533986
    32543987    return ssmR3DataWriteRaw(pSSM, &abHdr[0], cbHdr);
     
    33294062
    33304063    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) ));
    33324065
    33334066    /*
     
    39494682    {
    39504683        //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);
    39524685        if (RT_SUCCESS(rc))
    39534686        {
     
    40604793{
    40614794    /** @todo buffered reads! (at some level or another) */
    4062     int rc = ssmR3StrmRead(pSSM, pvBuf, cbToRead);
     4795    int rc = ssmR3StrmRead(&pSSM->Strm, pvBuf, cbToRead);
    40634796    if (RT_SUCCESS(rc))
    40644797    {
     
    41594892     * Read the two mandatory bytes.
    41604893     */
    4161     uint32_t u32StreamCRC = pSSM->u32StreamCRC;
     4894    uint32_t u32StreamCRC = ssmR3StrmCurCRC(pSSM); /** @todo NOT good. */
    41624895    uint8_t  abHdr[8];
    41634896    int rc = ssmR3DataReadV2Raw(pSSM, abHdr, 2);
     
    41904923        if (!(TermRec.fFlags & SSMRECTERM_FLAGS_CRC32))
    41914924            AssertLogRelMsgReturn(TermRec.u32StreamCRC == 0, ("%#x\n", TermRec.u32StreamCRC), VERR_SSM_INTEGRITY);
    4192         else if (pSSM->fChecksummed)
     4925        else if (pSSM->Strm.fChecksummed)
    41934926        {
    41944927            u32StreamCRC = RTCrc32Finish(u32StreamCRC);
     
    41964929        }
    41974930
    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));
    41994932        return VINF_SUCCESS;
    42004933    }
     
    43035036
    43045037    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,
    43065039          pSSM->u.Read.u8TypeAndFlags & SSM_REC_TYPE_MASK,
    43075040          !!(pSSM->u.Read.u8TypeAndFlags & SSM_REC_FLAGS_IMPORTANT),
     
    43295062    uint32_t off        = pSSM->u.Read.offDataBuffer;
    43305063    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));
    43325065    if (cbInBuffer > 0)
    43335066    {
     
    44095142
    44105143    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 ? "..." : ""));
    44125145    return VINF_SUCCESS;
    44135146}
     
    44315164    uint32_t off        = pSSM->u.Read.offDataBuffer;
    44325165    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));
    44345167    if (cbInBuffer > 0)
    44355168    {
     
    45075240
    45085241    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,
    45105243          cbBufOrg, RT_MIN(SSM_LOG_BYTES, cbBufOrg), pvBufOrg, cbBufOrg > SSM_LOG_BYTES ? "..." : ""));
    45115244    return VINF_SUCCESS;
     
    45475280          ? "ssmR3DataRead: %08llx|%08llx/%08x/%08x: cbBuf=%#x %.*Rhxs%s\n"
    45485281          : "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,
    45505283          cbBuf, RT_MIN(SSM_LOG_BYTES, cbBuf), pvBuf, cbBuf > SSM_LOG_BYTES ? "..." : ""));
    45515284
Note: See TracChangeset for help on using the changeset viewer.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette