50 #include "config/the_isa.hh"
51 #include "debug/Branch.hh"
56 namespace branch_prediction
61 numThreads(params.numThreads),
63 BTB(params.BTBEntries,
68 iPred(params.indirectBranchPred),
70 instShiftAmt(params.instShiftAmt)
77 : statistics::
Group(parent),
78 ADD_STAT(lookups, statistics::units::Count::get(),
79 "Number of BP lookups"),
80 ADD_STAT(condPredicted, statistics::units::Count::get(),
81 "Number of conditional branches predicted"),
82 ADD_STAT(condIncorrect, statistics::units::Count::get(),
83 "Number of conditional branches incorrect"),
84 ADD_STAT(BTBLookups, statistics::units::Count::get(),
85 "Number of BTB lookups"),
86 ADD_STAT(BTBHits, statistics::units::Count::get(),
"Number of BTB hits"),
87 ADD_STAT(BTBHitRatio, statistics::units::Ratio::get(),
"BTB Hit Ratio",
88 BTBHits / BTBLookups),
89 ADD_STAT(RASUsed, statistics::units::Count::get(),
90 "Number of times the RAS was used to get a target."),
91 ADD_STAT(RASIncorrect, statistics::units::Count::get(),
92 "Number of incorrect RAS predictions."),
93 ADD_STAT(indirectLookups, statistics::units::Count::get(),
94 "Number of indirect predictor lookups."),
95 ADD_STAT(indirectHits, statistics::units::Count::get(),
96 "Number of indirect target hits."),
97 ADD_STAT(indirectMisses, statistics::units::Count::get(),
98 "Number of indirect misses."),
99 ADD_STAT(indirectMispredicted, statistics::units::Count::get(),
100 "Number of mispredicted indirect branches.")
126 for ([[maybe_unused]]
const auto& ph :
predHist)
139 bool pred_taken =
false;
140 std::unique_ptr<PCStateBase> target(
pc.clone());
145 void *bp_history = NULL;
146 void *indirect_history = NULL;
149 DPRINTF(
Branch,
"[tid:%i] [sn:%llu] Unconditional control\n",
156 pred_taken =
lookup(tid,
pc.instAddr(), bp_history);
159 "Branch predictor predicted %i for PC %s\n",
160 tid, seqNum, pred_taken,
pc);
163 const bool orig_pred_taken = pred_taken;
169 "[tid:%i] [sn:%llu] Creating prediction history for PC %s\n",
173 bp_history, indirect_history, tid, inst);
193 DPRINTF(
Branch,
"[tid:%i] [sn:%llu] Instruction %s is a return, "
194 "RAS predicted target: %s, RAS index: %i\n",
195 tid, seqNum,
pc, *target, predict_record.
RASIndex);
207 "[tid:%i] [sn:%llu] Instruction %s was a call, adding "
208 "%s to the RAS index: %i\n",
209 tid, seqNum,
pc,
pc,
RAS[tid].topIdx());
220 "[tid:%i] [sn:%llu] Instruction %s predicted "
222 tid, seqNum,
pc, *target);
225 "valid entry\n", tid, seqNum);
234 "[tid:%i] [sn:%llu] btbUpdate "
251 "[tid:%i] [sn:%llu] Instruction %s predicted "
252 "indirect target is %s\n",
253 tid, seqNum,
pc, *target);
259 "[tid:%i] [sn:%llu] Instruction %s no indirect "
280 predict_record.
target = target->instAddr();
293 predHist[tid].push_front(predict_record);
296 "[tid:%i] [sn:%llu] History entry added. "
297 "predHist.size(): %i\n",
307 "sn:%llu]\n", tid, done_sn);
310 predHist[tid].back().seqNum <= done_sn) {
314 predHist[tid].back().bpHistory,
false,
335 while (!pred_hist.empty() &&
336 pred_hist.front().seqNum > squashed_sn) {
337 if (pred_hist.front().usedRAS) {
338 if (pred_hist.front().RASTarget !=
nullptr) {
340 " Restoring top of RAS to: %i,"
341 " target: %s\n", tid, squashed_sn,
342 pred_hist.front().RASIndex,
343 *pred_hist.front().RASTarget);
347 " Restoring top of RAS to: %i,"
348 " target: INVALID_TARGET\n", tid, squashed_sn,
349 pred_hist.front().RASIndex);
352 RAS[tid].restore(pred_hist.front().RASIndex,
353 pred_hist.front().RASTarget.get());
354 }
else if (pred_hist.front().wasCall && pred_hist.front().pushedRAS) {
357 " Call [sn:%llu] PC: %s Popping RAS\n", tid, squashed_sn,
358 pred_hist.front().seqNum, pred_hist.front().pc);
363 squash(tid, pred_hist.front().bpHistory);
369 "Removing history for [sn:%llu] "
370 "PC %#x\n", tid, squashed_sn, pred_hist.front().seqNum,
371 pred_hist.front().pc);
373 pred_hist.pop_front();
375 DPRINTF(
Branch,
"[tid:%i] [squash sn:%llu] predHist.size(): %i\n",
376 tid, squashed_sn,
predHist[tid].size());
401 DPRINTF(
Branch,
"[tid:%i] Squashing from sequence number %i, "
402 "setting target to %s\n", tid, squashed_sn, corr_target);
410 if (!pred_hist.empty()) {
412 auto hist_it = pred_hist.begin();
417 if (pred_hist.front().seqNum != squashed_sn) {
419 pred_hist.front().seqNum, squashed_sn);
421 assert(pred_hist.front().seqNum == squashed_sn);
425 if ((*hist_it).usedRAS) {
428 "[tid:%i] [squash sn:%llu] Incorrect RAS [sn:%llu]\n",
429 tid, squashed_sn, hist_it->seqNum);
442 pred_hist.front().predTaken = actually_taken;
443 pred_hist.front().target = corr_target.
instAddr();
445 update(tid, (*hist_it).pc, actually_taken,
446 pred_hist.front().bpHistory,
true, pred_hist.front().inst,
451 pred_hist.front().indirectHistory, actually_taken);
454 if (actually_taken) {
455 if (hist_it->wasReturn && !hist_it->usedRAS) {
457 "Incorrectly predicted "
458 "return [sn:%llu] PC: %#x\n", tid, squashed_sn,
462 hist_it->usedRAS =
true;
464 if (hist_it->wasIndirect) {
468 hist_it->seqNum, pred_hist.front().indirectHistory,
473 "BTB Update called for [sn:%llu] "
474 "PC %#x\n", tid, squashed_sn,
475 hist_it->seqNum, hist_it->pc);
477 BTB.
update(hist_it->pc, corr_target, tid);
481 if (hist_it->usedRAS) {
483 "[tid:%i] [squash sn:%llu] Incorrectly predicted "
484 "return [sn:%llu] PC: %#x Restoring RAS\n", tid,
486 hist_it->seqNum, hist_it->pc);
488 "[tid:%i] [squash sn:%llu] Restoring top of RAS "
489 "to: %i, target: %s\n", tid, squashed_sn,
490 hist_it->RASIndex, *hist_it->RASTarget);
491 RAS[tid].restore(hist_it->RASIndex, hist_it->RASTarget.get());
492 hist_it->usedRAS =
false;
493 }
else if (hist_it->wasCall && hist_it->pushedRAS) {
496 "[tid:%i] [squash sn:%llu] "
497 "Incorrectly predicted "
498 "Call [sn:%llu] PC: %s Popping RAS\n",
500 hist_it->seqNum, hist_it->pc);
502 hist_it->pushedRAS =
false;
507 "update\n", tid, squashed_sn);
517 auto pred_hist_it = ph.begin();
519 cprintf(
"predHist[%i].size(): %i\n",
i++, ph.size());
521 while (pred_hist_it != ph.end()) {
522 cprintf(
"sn:%llu], PC:%#x, tid:%i, predTaken:%i, "
524 pred_hist_it->seqNum, pred_hist_it->pc,
525 pred_hist_it->tid, pred_hist_it->predTaken,
526 pred_hist_it->bpHistory);