Changeset 24356 in vbox for trunk/src/VBox
- Timestamp:
- Nov 4, 2009 8:32:13 PM (15 years ago)
- svn:sync-xref-src-repo-rev:
- 54379
- Location:
- trunk/src/VBox/VMM
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/VMM/PDMAsyncCompletionFileCache.cpp
r24278 r24356 45 45 *******************************************************************************/ 46 46 #define LOG_GROUP LOG_GROUP_PDM_ASYNC_COMPLETION 47 #define RT_STRICT 47 48 #include <iprt/types.h> 48 49 #include <iprt/mem.h> … … 269 270 && (ASMAtomicReadU32(&pCurr->cRefs) == 0)) 270 271 { 272 AssertMsg(!(pCurr->fFlags & PDMACFILECACHE_ENTRY_IS_DEPRECATED), 273 ("This entry is deprecated so it should have the I/O in progress flag set\n")); 274 Assert(!pCurr->pbDataReplace); 275 271 276 LogFlow(("Evicting entry %#p (%u bytes)\n", pCurr, pCurr->cbData)); 277 272 278 if (pCurr->pbData) 273 279 { … … 502 508 if (pTask->enmTransferType == PDMACTASKFILETRANSFER_WRITE) 503 509 { 504 pEntry->fFlags &= ~PDMACFILECACHE_ENTRY_IS_DIRTY; 505 506 while (pCurr) 510 if (pEntry->fFlags & PDMACFILECACHE_ENTRY_IS_DEPRECATED) 507 511 { 508 AssertMsg(pCurr->fWrite, ("Completed write entries should never have read tasks attached\n")); 509 510 memcpy(pEntry->pbData + pCurr->uBufOffset, pCurr->pvBuf, pCurr->cbTransfer); 511 pEntry->fFlags |= PDMACFILECACHE_ENTRY_IS_DIRTY; 512 513 uint32_t uOld = ASMAtomicSubU32(&pCurr->pTask->cbTransferLeft, pCurr->cbTransfer); 514 AssertMsg(uOld >= pCurr->cbTransfer, ("New value would overflow\n")); 515 if (!(uOld - pCurr->cbTransfer) 516 && !ASMAtomicXchgBool(&pCurr->pTask->fCompleted, true)) 517 pdmR3AsyncCompletionCompleteTask(&pCurr->pTask->Core); 518 519 PPDMACFILETASKSEG pFree = pCurr; 520 pCurr = pCurr->pNext; 521 522 RTMemFree(pFree); 512 AssertMsg(!pCurr, ("The entry is deprecated but has waiting write segments attached\n")); 513 514 RTMemPageFree(pEntry->pbData); 515 pEntry->pbData = pEntry->pbDataReplace; 516 pEntry->pbDataReplace = NULL; 517 pEntry->fFlags &= ~PDMACFILECACHE_ENTRY_IS_DEPRECATED; 518 } 519 else 520 { 521 pEntry->fFlags &= ~PDMACFILECACHE_ENTRY_IS_DIRTY; 522 523 while (pCurr) 524 { 525 AssertMsg(pCurr->fWrite, ("Completed write entries should never have read tasks attached\n")); 526 527 memcpy(pEntry->pbData + pCurr->uBufOffset, pCurr->pvBuf, pCurr->cbTransfer); 528 pEntry->fFlags |= PDMACFILECACHE_ENTRY_IS_DIRTY; 529 530 uint32_t uOld = ASMAtomicSubU32(&pCurr->pTask->cbTransferLeft, pCurr->cbTransfer); 531 AssertMsg(uOld >= pCurr->cbTransfer, ("New value would overflow\n")); 532 if (!(uOld - pCurr->cbTransfer) 533 && !ASMAtomicXchgBool(&pCurr->pTask->fCompleted, true)) 534 pdmR3AsyncCompletionCompleteTask(&pCurr->pTask->Core); 535 536 PPDMACFILETASKSEG pFree = pCurr; 537 pCurr = pCurr->pNext; 538 539 RTMemFree(pFree); 540 } 523 541 } 524 542 } … … 854 872 return NULL; 855 873 856 pEntryNew->Core.Key = off; 857 pEntryNew->Core.KeyLast = off + cbData - 1; 858 pEntryNew->pEndpoint = pEndpoint; 859 pEntryNew->pCache = pCache; 860 pEntryNew->fFlags = 0; 861 pEntryNew->cRefs = 1; /* We are using it now. */ 862 pEntryNew->pList = NULL; 863 pEntryNew->cbData = cbData; 864 pEntryNew->pWaitingHead = NULL; 865 pEntryNew->pWaitingTail = NULL; 866 pEntryNew->pbData = (uint8_t *)RTMemPageAlloc(cbData); 874 pEntryNew->Core.Key = off; 875 pEntryNew->Core.KeyLast = off + cbData - 1; 876 pEntryNew->pEndpoint = pEndpoint; 877 pEntryNew->pCache = pCache; 878 pEntryNew->fFlags = 0; 879 pEntryNew->cRefs = 1; /* We are using it now. */ 880 pEntryNew->pList = NULL; 881 pEntryNew->cbData = cbData; 882 pEntryNew->pWaitingHead = NULL; 883 pEntryNew->pWaitingTail = NULL; 884 pEntryNew->pbDataReplace = NULL; 885 pEntryNew->pbData = (uint8_t *)RTMemPageAlloc(cbData); 867 886 868 887 if (RT_UNLIKELY(!pEntryNew->pbData)) … … 954 973 pbSegBuf += BytesTransfered; \ 955 974 } \ 956 while (0) ;975 while (0) 957 976 958 977 /** … … 1028 1047 || (pEntry->pList == &pCache->LruFrequentlyUsed)) 1029 1048 { 1030 LogFlow(("Fetching data for ghost entry %#p from file\n", pEntry)); 1031 1032 if (pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1033 PDMACFILECACHE_ENTRY_IO_IN_PROGRESS, 1034 PDMACFILECACHE_ENTRY_IS_DIRTY)) 1049 if(pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1050 PDMACFILECACHE_ENTRY_IS_DEPRECATED, 1051 0)) 1035 1052 { 1036 /* Entry didn't completed yet. Append to the list */ 1037 while (cbToRead) 1038 { 1039 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1040 1041 pSeg->pTask = pTask; 1042 pSeg->uBufOffset = OffDiff; 1043 pSeg->cbTransfer = RT_MIN(cbToRead, cbSegLeft); 1044 pSeg->pvBuf = pbSegBuf; 1045 pSeg->fWrite = false; 1046 1047 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1048 1049 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1050 1051 off += pSeg->cbTransfer; 1052 cbToRead -= pSeg->cbTransfer; 1053 OffDiff += pSeg->cbTransfer; 1054 } 1055 RTSemRWReleaseWrite(pEndpointCache->SemRWEntries); 1056 } 1057 else 1058 { 1059 /* Read as much as we can from the entry. */ 1053 /* Entry is deprecated. Read data from the new buffer. */ 1060 1054 while (cbToRead) 1061 1055 { 1062 1056 size_t cbCopy = RT_MIN(cbSegLeft, cbToRead); 1063 1057 1064 memcpy(pbSegBuf, pEntry->pbData + OffDiff, cbCopy);1058 memcpy(pbSegBuf, pEntry->pbDataReplace + OffDiff, cbCopy); 1065 1059 1066 1060 ADVANCE_SEGMENT_BUFFER(cbCopy); … … 1070 1064 OffDiff += cbCopy; 1071 1065 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1066 } 1067 } 1068 else 1069 { 1070 if (pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1071 PDMACFILECACHE_ENTRY_IO_IN_PROGRESS, 1072 PDMACFILECACHE_ENTRY_IS_DIRTY)) 1073 { 1074 /* Entry didn't completed yet. Append to the list */ 1075 while (cbToRead) 1076 { 1077 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1078 1079 pSeg->pTask = pTask; 1080 pSeg->uBufOffset = OffDiff; 1081 pSeg->cbTransfer = RT_MIN(cbToRead, cbSegLeft); 1082 pSeg->pvBuf = pbSegBuf; 1083 pSeg->fWrite = false; 1084 1085 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1086 1087 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1088 1089 off += pSeg->cbTransfer; 1090 cbToRead -= pSeg->cbTransfer; 1091 OffDiff += pSeg->cbTransfer; 1092 } 1093 RTSemRWReleaseWrite(pEndpointCache->SemRWEntries); 1094 } 1095 else 1096 { 1097 /* Read as much as we can from the entry. */ 1098 while (cbToRead) 1099 { 1100 size_t cbCopy = RT_MIN(cbSegLeft, cbToRead); 1101 1102 memcpy(pbSegBuf, pEntry->pbData + OffDiff, cbCopy); 1103 1104 ADVANCE_SEGMENT_BUFFER(cbCopy); 1105 1106 cbToRead -= cbCopy; 1107 off += cbCopy; 1108 OffDiff += cbCopy; 1109 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1110 } 1072 1111 } 1073 1112 } … … 1080 1119 else 1081 1120 { 1121 LogFlow(("Fetching data for ghost entry %#p from file\n", pEntry)); 1122 1082 1123 RTCritSectEnter(&pCache->CritSect); 1083 1124 pdmacFileCacheUpdate(pCache, pEntry); … … 1303 1344 || (pEntry->pList == &pCache->LruFrequentlyUsed)) 1304 1345 { 1305 1306 /* If the entry is dirty it must be also in progress now and we have to defer updating it again. */ 1346 /* Check if the buffer is deprecated. */ 1307 1347 if(pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1308 PDMACFILECACHE_ENTRY_IS_D IRTY,1348 PDMACFILECACHE_ENTRY_IS_DEPRECATED, 1309 1349 0)) 1310 1350 { 1311 1351 AssertMsg(pEntry->fFlags & PDMACFILECACHE_ENTRY_IO_IN_PROGRESS, 1312 ("Entry is dirty but not in progress\n")); 1313 1314 /* The data isn't written to the file yet */ 1352 ("Entry is deprecated but not in progress\n")); 1353 AssertPtr(pEntry->pbDataReplace); 1354 1355 LogFlow(("Writing to deprecated buffer of entry %#p\n", pEntry)); 1356 1357 /* Update the data from the write. */ 1315 1358 while (cbToWrite) 1316 1359 { 1317 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1318 1319 pSeg->pTask = pTask; 1320 pSeg->uBufOffset = OffDiff; 1321 pSeg->cbTransfer = RT_MIN(cbToWrite, cbSegLeft); 1322 pSeg->pvBuf = pbSegBuf; 1323 pSeg->fWrite = true; 1324 1325 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1326 1327 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1328 1329 off += pSeg->cbTransfer; 1330 OffDiff += pSeg->cbTransfer; 1331 cbToWrite -= pSeg->cbTransfer; 1360 size_t cbCopy = RT_MIN(cbSegLeft, cbToWrite); 1361 1362 memcpy(pEntry->pbDataReplace + OffDiff, pbSegBuf, cbCopy); 1363 1364 ADVANCE_SEGMENT_BUFFER(cbCopy); 1365 1366 cbToWrite-= cbCopy; 1367 off += cbCopy; 1368 OffDiff += cbCopy; 1369 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1332 1370 } 1333 STAM_COUNTER_INC(&pEndpointCache->StatWriteDeferred);1334 1371 RTSemRWReleaseWrite(pEndpointCache->SemRWEntries); 1335 1372 } 1336 1373 else 1337 1374 { 1338 /* 1339 * Check if a read is in progress for this entry. 1340 * We have to defer processing in that case. 1341 */ 1375 /* If the entry is dirty it must be also in progress now and we have to defer updating it again. */ 1342 1376 if(pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1343 PDMACFILECACHE_ENTRY_I O_IN_PROGRESS,1377 PDMACFILECACHE_ENTRY_IS_DIRTY, 1344 1378 0)) 1345 1379 { 1346 while (cbToWrite) 1380 AssertMsg(pEntry->fFlags & PDMACFILECACHE_ENTRY_IO_IN_PROGRESS, 1381 ("Entry is dirty but not in progress\n")); 1382 Assert(!pEntry->pbDataReplace); 1383 1384 /* Deprecate the current buffer. */ 1385 if (!pEntry->pWaitingHead) 1386 pEntry->pbDataReplace = (uint8_t *)RTMemPageAlloc(pEntry->cbData); 1387 1388 /* If we are out of memory or have waiting segments 1389 * defer the write. */ 1390 if (!pEntry->pbDataReplace || pEntry->pWaitingHead) 1347 1391 { 1348 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1349 1350 pSeg->pTask = pTask; 1351 pSeg->uBufOffset = OffDiff; 1352 pSeg->cbTransfer = RT_MIN(cbToWrite, cbSegLeft); 1353 pSeg->pvBuf = pbSegBuf; 1354 pSeg->fWrite = true; 1355 1356 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1357 1358 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1359 1360 off += pSeg->cbTransfer; 1361 OffDiff += pSeg->cbTransfer; 1362 cbToWrite -= pSeg->cbTransfer; 1392 /* The data isn't written to the file yet */ 1393 while (cbToWrite) 1394 { 1395 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1396 1397 pSeg->pTask = pTask; 1398 pSeg->uBufOffset = OffDiff; 1399 pSeg->cbTransfer = RT_MIN(cbToWrite, cbSegLeft); 1400 pSeg->pvBuf = pbSegBuf; 1401 pSeg->fWrite = true; 1402 1403 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1404 1405 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1406 1407 off += pSeg->cbTransfer; 1408 OffDiff += pSeg->cbTransfer; 1409 cbToWrite -= pSeg->cbTransfer; 1410 } 1411 STAM_COUNTER_INC(&pEndpointCache->StatWriteDeferred); 1363 1412 } 1364 STAM_COUNTER_INC(&pEndpointCache->StatWriteDeferred); 1413 else 1414 { 1415 LogFlow(("Deprecating buffer for entry %#p\n", pEntry)); 1416 pEntry->fFlags |= PDMACFILECACHE_ENTRY_IS_DEPRECATED; 1417 1418 #if 1 1419 /* Copy the data before the update. */ 1420 if (OffDiff) 1421 memcpy(pEntry->pbDataReplace, pEntry->pbData, OffDiff); 1422 1423 /* Copy data behind the update. */ 1424 if ((pEntry->cbData - OffDiff - cbToWrite) > 0) 1425 memcpy(pEntry->pbDataReplace + OffDiff + cbToWrite, 1426 pEntry->pbData + OffDiff + cbToWrite, 1427 (pEntry->cbData - OffDiff - cbToWrite)); 1428 #else 1429 /* A safer method but probably slower. */ 1430 memcpy(pEntry->pbDataReplace, pEntry->pbData, pEntry->cbData); 1431 #endif 1432 1433 /* Update the data from the write. */ 1434 while (cbToWrite) 1435 { 1436 size_t cbCopy = RT_MIN(cbSegLeft, cbToWrite); 1437 1438 memcpy(pEntry->pbDataReplace + OffDiff, pbSegBuf, cbCopy); 1439 1440 ADVANCE_SEGMENT_BUFFER(cbCopy); 1441 1442 cbToWrite-= cbCopy; 1443 off += cbCopy; 1444 OffDiff += cbCopy; 1445 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1446 } 1447 1448 /* We are done here. A new write is initiated if the current request completes. */ 1449 } 1450 1365 1451 RTSemRWReleaseWrite(pEndpointCache->SemRWEntries); 1366 1452 } 1367 1453 else 1368 1454 { 1369 /* Write as much as we can into the entry and update the file. */ 1370 while (cbToWrite) 1455 /* 1456 * Check if a read is in progress for this entry. 1457 * We have to defer processing in that case. 1458 */ 1459 if(pdmacFileEpCacheEntryFlagIsSetClearAcquireLock(pEndpointCache, pEntry, 1460 PDMACFILECACHE_ENTRY_IO_IN_PROGRESS, 1461 0)) 1371 1462 { 1372 size_t cbCopy = RT_MIN(cbSegLeft, cbToWrite); 1373 1374 memcpy(pEntry->pbData + OffDiff, pbSegBuf, cbCopy); 1375 1376 ADVANCE_SEGMENT_BUFFER(cbCopy); 1377 1378 cbToWrite-= cbCopy; 1379 off += cbCopy; 1380 OffDiff += cbCopy; 1381 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1463 while (cbToWrite) 1464 { 1465 PPDMACFILETASKSEG pSeg = (PPDMACFILETASKSEG)RTMemAllocZ(sizeof(PDMACFILETASKSEG)); 1466 1467 pSeg->pTask = pTask; 1468 pSeg->uBufOffset = OffDiff; 1469 pSeg->cbTransfer = RT_MIN(cbToWrite, cbSegLeft); 1470 pSeg->pvBuf = pbSegBuf; 1471 pSeg->fWrite = true; 1472 1473 ADVANCE_SEGMENT_BUFFER(pSeg->cbTransfer); 1474 1475 pdmacFileEpCacheEntryAddWaitingSegment(pEntry, pSeg); 1476 1477 off += pSeg->cbTransfer; 1478 OffDiff += pSeg->cbTransfer; 1479 cbToWrite -= pSeg->cbTransfer; 1480 } 1481 STAM_COUNTER_INC(&pEndpointCache->StatWriteDeferred); 1482 RTSemRWReleaseWrite(pEndpointCache->SemRWEntries); 1382 1483 } 1383 1384 pEntry->fFlags |= PDMACFILECACHE_ENTRY_IS_DIRTY; 1385 pdmacFileCacheWriteToEndpoint(pEntry); 1484 else 1485 { 1486 /* Write as much as we can into the entry and update the file. */ 1487 while (cbToWrite) 1488 { 1489 size_t cbCopy = RT_MIN(cbSegLeft, cbToWrite); 1490 1491 memcpy(pEntry->pbData + OffDiff, pbSegBuf, cbCopy); 1492 1493 ADVANCE_SEGMENT_BUFFER(cbCopy); 1494 1495 cbToWrite-= cbCopy; 1496 off += cbCopy; 1497 OffDiff += cbCopy; 1498 ASMAtomicSubS32(&pTask->cbTransferLeft, cbCopy); 1499 } 1500 1501 pEntry->fFlags |= PDMACFILECACHE_ENTRY_IS_DIRTY; 1502 pdmacFileCacheWriteToEndpoint(pEntry); 1503 } 1386 1504 } 1505 1506 /* Move this entry to the top position */ 1507 RTCritSectEnter(&pCache->CritSect); 1508 pdmacFileCacheEntryAddToList(&pCache->LruFrequentlyUsed, pEntry); 1509 RTCritSectLeave(&pCache->CritSect); 1387 1510 } 1388 1389 /* Move this entry to the top position */1390 RTCritSectEnter(&pCache->CritSect);1391 pdmacFileCacheEntryAddToList(&pCache->LruFrequentlyUsed, pEntry);1392 RTCritSectLeave(&pCache->CritSect);1393 1511 } 1394 1512 else -
trunk/src/VBox/VMM/PDMAsyncCompletionFileInternal.h
r24278 r24356 229 229 /** Pointer to the memory containing the data. */ 230 230 uint8_t *pbData; 231 /** Pointer to the buffer replacing the current one 232 * if the deprecated flag is set. */ 233 uint8_t *pbDataReplace; 231 234 /** Head of list of tasks waiting for this one to finish. */ 232 235 PPDMACFILETASKSEG pWaitingHead; … … 237 240 #define PDMACFILECACHE_ENTRY_IO_IN_PROGRESS RT_BIT(0) 238 241 /** Entry is locked and thus not evictable. */ 239 #define PDMACFILECACHE_ENTRY_LOCKED RT_BIT(1)242 #define PDMACFILECACHE_ENTRY_LOCKED RT_BIT(1) 240 243 /** Entry is dirty */ 241 #define PDMACFILECACHE_ENTRY_IS_DIRTY RT_BIT(2) 244 #define PDMACFILECACHE_ENTRY_IS_DIRTY RT_BIT(2) 245 /** The current buffer used for the entry is deprecated. 246 * The new one is available and will be replaced as soon as the file update 247 * completed. 248 */ 249 #define PDMACFILECACHE_ENTRY_IS_DEPRECATED RT_BIT(3) 242 250 /** Entry is not evictable. */ 243 251 #define PDMACFILECACHE_NOT_EVICTABLE (PDMACFILECACHE_ENTRY_LOCKED | PDMACFILECACHE_IO_IN_PROGRESS)
Note:
See TracChangeset
for help on using the changeset viewer.