@@ -256,6 +256,7 @@ of effective system reads for a given file with a code like
256
256
#include " TFriendElement.h"
257
257
#include " TFile.h"
258
258
#include " TMath.h"
259
+ #include " TVirtualPerfStats.h"
259
260
#include < limits.h>
260
261
261
262
Int_t TTreeCache::fgLearnEntries = 100 ;
@@ -714,6 +715,10 @@ TBranch *TTreeCache::CalculateMissEntries(Long64_t pos, Int_t len, Bool_t all)
714
715
Bool_t found_request = kFALSE ;
715
716
TBranch *resultBranch = nullptr ;
716
717
Long64_t entry = fTree ->GetReadEntry ();
718
+
719
+ std::vector<std::pair<TBranch*,Int_t>> basketsInfo;
720
+ auto perfStats = GetTree ()->GetPerfStats ();
721
+
717
722
// printf("Will search %d branches for basket at %ld.\n", count, pos);
718
723
for (int i = 0 ; i < count; i++) {
719
724
TBranch *b =
@@ -730,12 +735,30 @@ TBranch *TTreeCache::CalculateMissEntries(Long64_t pos, Int_t len, Bool_t all)
730
735
}
731
736
// At this point, we are ready to push back a new offset
732
737
fMissCache ->fEntries .emplace_back (std::move (iopos));
738
+
739
+ if (R__unlikely (perfStats)) {
740
+ Int_t blistsize = b->GetWriteBasket ();
741
+ Int_t basketNumber = -1 ;
742
+ for (Int_t bn = 0 ; bn < blistsize; ++bn) {
743
+ if (iopos.fPos == b->GetBasketSeek (bn)) {
744
+ basketNumber = bn;
745
+ break ;
746
+ }
747
+ }
748
+ if (basketNumber >= 0 )
749
+ basketsInfo.emplace_back (b, basketNumber);
750
+ }
733
751
}
734
752
if (R__unlikely (!found_request)) {
735
753
// We have gone through all the branches in this file and the requested basket
736
754
// doesn't appear to be in any of them. Likely a logic error / bug.
737
755
fMissCache ->fEntries .clear ();
738
756
}
757
+ if (R__unlikely (perfStats)) {
758
+ for (auto &info : basketsInfo) {
759
+ perfStats->SetLoadedMiss (info.first , info.second );
760
+ }
761
+ }
739
762
return resultBranch;
740
763
}
741
764
@@ -1011,6 +1034,7 @@ Bool_t TTreeCache::FillBuffer()
1011
1034
Int_t prevNtot;
1012
1035
Int_t minBasket = 0 ; // We will use this to avoid re-checking the first baskets in the 2nd (or more) run in the while loop.
1013
1036
Long64_t maxReadEntry = minEntry; // If we are stopped before the end of the 2nd pass, this marker will where we need to start next time.
1037
+ auto perfStats = GetTree ()->GetPerfStats ();
1014
1038
do {
1015
1039
prevNtot = fNtotCurrentBuf ;
1016
1040
Int_t nextMinBasket = INT_MAX;
@@ -1101,6 +1125,11 @@ Bool_t TTreeCache::FillBuffer()
1101
1125
}
1102
1126
}
1103
1127
}
1128
+
1129
+ if (R__unlikely (perfStats)) {
1130
+ perfStats->SetLoaded (b, j);
1131
+ }
1132
+
1104
1133
if (fEnablePrefetching ){
1105
1134
if (fFirstBuffer ) {
1106
1135
TFileCacheRead::Prefetch (pos,len);
@@ -1337,23 +1366,47 @@ Int_t TTreeCache::ReadBufferNormal(char *buf, Long64_t pos, Int_t len){
1337
1366
return 1 ;
1338
1367
}
1339
1368
1369
+ static const auto recordMiss = [](TVirtualPerfStats *perfStats, TObjArray *branches, Bool_t bufferFilled, Long64_t basketpos) {
1370
+ if (gDebug > 6 )
1371
+ ::Info (" TTreeCache::ReadBufferNormal" , " Cache miss after an %s FillBuffer: pos=%lld" , bufferFilled ? " active" : " inactive" , basketpos);
1372
+ for (Int_t i=0 ; i<branches->GetEntries (); ++i) {
1373
+ TBranch *b = (TBranch*)branches->UncheckedAt (i);
1374
+ Int_t blistsize = b->GetListOfBaskets ()->GetSize ();
1375
+ for (Int_t j = 0 ; j < blistsize; ++j) {
1376
+ if (basketpos == b->GetBasketSeek (j)) {
1377
+ if (gDebug > 6 )
1378
+ ::Info (" TTreeCache::ReadBufferNormal" ," Missing basket: %d for %s" , j, b->GetName ());
1379
+ perfStats->SetMissed (b, j);
1380
+ }
1381
+ }
1382
+ }
1383
+ };
1384
+
1340
1385
// not found in cache. Do we need to fill the cache?
1341
1386
Bool_t bufferFilled = FillBuffer();
1342
1387
if (bufferFilled) {
1343
1388
Int_t res = TFileCacheRead::ReadBuffer (buf,pos,len);
1344
1389
1345
1390
if (res == 1 )
1346
1391
fNReadOk ++;
1347
- else if (res == 0 )
1392
+ else if (res == 0 ) {
1348
1393
fNReadMiss ++;
1394
+ auto perfStats = GetTree ()->GetPerfStats ();
1395
+ if (perfStats)
1396
+ recordMiss (perfStats, fBranches , bufferFilled, pos);
1397
+ }
1349
1398
1350
1399
return res;
1351
1400
}
1401
+
1352
1402
if (CheckMissCache (buf, pos, len)) {
1353
1403
return 1 ;
1354
1404
}
1355
1405
1356
1406
fNReadMiss ++;
1407
+ auto perfStats = GetTree ()->GetPerfStats ();
1408
+ if (perfStats)
1409
+ recordMiss (perfStats, fBranches , bufferFilled, pos);
1357
1410
1358
1411
return 0 ;
1359
1412
}
0 commit comments