Go to the documentation of this file.
47 #include "arch/types.hh"
48 #include "arch/utility.hh"
50 #include "config/the_isa.hh"
51 #include "debug/Branch.hh"
55 numThreads(params.numThreads),
57 BTB(params.BTBEntries,
62 iPred(params.indirectBranchPred),
64 instShiftAmt(params.instShiftAmt)
71 :
Stats::Group(parent),
74 "Number of conditional branches predicted"),
76 "Number of conditional branches incorrect"),
81 "Number of times the RAS was used to get a target."),
83 "Number of incorrect RAS predictions."),
85 "Number of indirect predictor lookups."),
89 "Number of mispredicted indirect branches.")
115 for (M5_VAR_USED
const auto& ph :
predHist)
128 bool pred_taken =
false;
134 void *bp_history = NULL;
135 void *indirect_history = NULL;
138 DPRINTF(Branch,
"[tid:%i] [sn:%llu] "
139 "Unconditional control\n",
146 pred_taken =
lookup(tid,
pc.instAddr(), bp_history);
148 DPRINTF(Branch,
"[tid:%i] [sn:%llu] "
149 "Branch predictor predicted %i for PC %s\n",
150 tid, seqNum, pred_taken,
pc);
153 const bool orig_pred_taken = pred_taken;
158 DPRINTF(Branch,
"[tid:%i] [sn:%llu] "
159 "Creating prediction history "
160 "for PC %s\n", tid, seqNum,
pc);
163 bp_history, indirect_history, tid, inst);
182 DPRINTF(Branch,
"[tid:%i] [sn:%llu] Instruction %s is a return, "
183 "RAS predicted target: %s, RAS index: %i\n",
184 tid, seqNum,
pc, target, predict_record.
RASIndex);
196 "[tid:%i] [sn:%llu] Instruction %s was a call, adding "
197 "%s to the RAS index: %i\n",
198 tid, seqNum,
pc,
pc,
RAS[tid].topIdx());
209 "[tid:%i] [sn:%llu] Instruction %s predicted "
211 tid, seqNum,
pc, target);
213 DPRINTF(Branch,
"[tid:%i] [sn:%llu] BTB doesn't have a "
214 "valid entry\n",tid,seqNum);
223 "[tid:%i] [sn:%llu] btbUpdate "
240 "[tid:%i] [sn:%llu] "
241 "Instruction %s predicted "
242 "indirect target is %s\n",
243 tid, seqNum,
pc, target);
249 "[tid:%i] [sn:%llu] "
250 "Instruction %s no indirect "
271 predict_record.
target = target.instAddr();
284 predHist[tid].push_front(predict_record);
287 "[tid:%i] [sn:%llu] History entry added. "
288 "predHist.size(): %i\n",
297 DPRINTF(Branch,
"[tid:%i] Committing branches until "
298 "sn:%llu]\n", tid, done_sn);
301 predHist[tid].back().seqNum <= done_sn) {
305 predHist[tid].back().bpHistory,
false,
326 while (!pred_hist.empty() &&
327 pred_hist.front().seqNum > squashed_sn) {
328 if (pred_hist.front().usedRAS) {
329 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu]"
330 " Restoring top of RAS to: %i,"
331 " target: %s\n", tid, squashed_sn,
332 pred_hist.front().RASIndex, pred_hist.front().RASTarget);
334 RAS[tid].restore(pred_hist.front().RASIndex,
335 pred_hist.front().RASTarget);
336 }
else if (pred_hist.front().wasCall && pred_hist.front().pushedRAS) {
338 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu] Squashing"
339 " Call [sn:%llu] PC: %s Popping RAS\n", tid, squashed_sn,
340 pred_hist.front().seqNum, pred_hist.front().pc);
345 squash(tid, pred_hist.front().bpHistory);
350 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu] "
351 "Removing history for [sn:%llu] "
352 "PC %#x\n", tid, squashed_sn, pred_hist.front().seqNum,
353 pred_hist.front().pc);
355 pred_hist.pop_front();
357 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu] predHist.size(): %i\n",
358 tid, squashed_sn,
predHist[tid].size());
383 DPRINTF(Branch,
"[tid:%i] Squashing from sequence number %i, "
384 "setting target to %s\n", tid, squashed_sn, corrTarget);
392 if (!pred_hist.empty()) {
394 auto hist_it = pred_hist.begin();
399 if (pred_hist.front().seqNum != squashed_sn) {
400 DPRINTF(Branch,
"Front sn %i != Squash sn %i\n",
401 pred_hist.front().seqNum, squashed_sn);
403 assert(pred_hist.front().seqNum == squashed_sn);
407 if ((*hist_it).usedRAS) {
410 "[tid:%i] [squash sn:%llu] Incorrect RAS [sn:%llu]\n",
411 tid, squashed_sn, hist_it->seqNum);
424 pred_hist.front().predTaken = actually_taken;
425 pred_hist.front().target = corrTarget.instAddr();
427 update(tid, (*hist_it).pc, actually_taken,
428 pred_hist.front().bpHistory,
true, pred_hist.front().inst,
429 corrTarget.instAddr());
433 pred_hist.front().indirectHistory, actually_taken);
436 if (actually_taken) {
437 if (hist_it->wasReturn && !hist_it->usedRAS) {
438 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu] "
439 "Incorrectly predicted "
440 "return [sn:%llu] PC: %#x\n", tid, squashed_sn,
444 hist_it->usedRAS =
true;
446 if (hist_it->wasIndirect) {
450 hist_it->seqNum, pred_hist.front().indirectHistory,
454 DPRINTF(Branch,
"[tid:%i] [squash sn:%llu] "
455 "BTB Update called for [sn:%llu] "
456 "PC %#x\n", tid, squashed_sn,
457 hist_it->seqNum, hist_it->pc);
459 BTB.
update((*hist_it).pc, corrTarget, tid);
463 if (hist_it->usedRAS) {
465 "[tid:%i] [squash sn:%llu] Incorrectly predicted "
466 "return [sn:%llu] PC: %#x Restoring RAS\n", tid,
468 hist_it->seqNum, hist_it->pc);
470 "[tid:%i] [squash sn:%llu] Restoring top of RAS "
471 "to: %i, target: %s\n", tid, squashed_sn,
472 hist_it->RASIndex, hist_it->RASTarget);
473 RAS[tid].restore(hist_it->RASIndex, hist_it->RASTarget);
474 hist_it->usedRAS =
false;
475 }
else if (hist_it->wasCall && hist_it->pushedRAS) {
478 "[tid:%i] [squash sn:%llu] "
479 "Incorrectly predicted "
480 "Call [sn:%llu] PC: %s Popping RAS\n",
482 hist_it->seqNum, hist_it->pc);
484 hist_it->pushedRAS =
false;
488 DPRINTF(Branch,
"[tid:%i] [sn:%llu] pred_hist empty, can't "
489 "update\n", tid, squashed_sn);
499 auto pred_hist_it = ph.begin();
501 cprintf(
"predHist[%i].size(): %i\n",
i++, ph.size());
503 while (pred_hist_it != ph.end()) {
504 cprintf(
"sn:%llu], PC:%#x, tid:%i, predTaken:%i, "
506 pred_hist_it->seqNum, pred_hist_it->pc,
507 pred_hist_it->tid, pred_hist_it->predTaken,
508 pred_hist_it->bpHistory);
virtual void updateDirectionInfo(ThreadID tid, bool actually_taken)=0
bool wasReturn
Whether or not the instruction was a return.
void regProbePoints() override
Register probe points for this object.
bool isDirectCtrl() const
virtual void genIndirectInfo(ThreadID tid, void *&indirect_history)=0
PCState buildRetPC(const PCState &curPC, const PCState &callPC)
ProbePoints::PMUUPtr ppMisses
Miss-predicted branches.
virtual void recordTarget(InstSeqNum seq_num, void *indirect_history, const TheISA::PCState &target, ThreadID tid)=0
std::vector< ReturnAddrStack > RAS
The per-thread return address stack.
int16_t ThreadID
Thread index/ID type.
virtual void recordIndirect(Addr br_addr, Addr tgt_addr, InstSeqNum seq_num, ThreadID tid)=0
Stats::Scalar lookups
Stat for number of BP lookups.
Stats::Scalar indirectMisses
Stat for the number of indirect target misses.
TheISA::PCState RASTarget
The RAS target (only valid if a return).
IndirectPredictor * iPred
The indirect target predictor.
ProbePointArg generates a point for the class of Arg.
Addr target
Target of the branch.
void drainSanityCheck() const
Perform sanity checks after a drain.
virtual void btbUpdate(ThreadID tid, Addr instPC, void *&bp_history)=0
If a branch is not taken, because the BTB address is invalid or missing, this function sets the appro...
Stats::Scalar condPredicted
Stat for number of conditional branches predicted.
virtual void commit(InstSeqNum seq_num, ThreadID tid, void *indirect_history)=0
ProbePoints::PMUUPtr pmuProbePoint(const char *name)
Helper method to instantiate probe points belonging to this object.
BPredUnitStats(Stats::Group *parent)
void update(const InstSeqNum &done_sn, ThreadID tid)
Tells the branch predictor to commit any updates until the given sequence number.
void advancePC(PCState &pc, const StaticInstPtr &inst)
virtual bool lookup(ThreadID tid, Addr instPC, void *&bp_history)=0
Looks up a given PC in the BP to see if it is taken or not taken.
virtual void changeDirectionPrediction(ThreadID tid, void *indirect_history, bool actually_taken)=0
Stats::Scalar indirectHits
Stat for the number of indirect target hits.
Stats::Scalar condIncorrect
Stat for number of conditional branches predicted incorrectly.
void update(Addr instPC, const TheISA::PCState &targetPC, ThreadID tid)
Updates the BTB with the target of a branch.
Stats::Scalar BTBHits
Stat for number of BTB hits.
void cprintf(const char *format, const Args &...args)
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Stats::Scalar RASUsed
Stat for number of times the RAS is used to get a target.
Stats::Scalar indirectMispredicted
Stat for the number of indirect target mispredictions.
Stats::Scalar indirectLookups
Stat for the number of indirect target lookups.
bool valid(Addr instPC, ThreadID tid)
Checks if a branch is in the BTB.
Stats::Scalar RASIncorrect
Stat for number of times the RAS is incorrect.
virtual bool lookup(Addr br_addr, TheISA::PCState &br_target, ThreadID tid)=0
virtual void uncondBranch(ThreadID tid, Addr pc, void *&bp_history)=0
BPredUnit::BPredUnitStats stats
std::vector< History > predHist
The per-thread predictor history.
BPredUnit(const Params &p)
bool usedRAS
Whether or not the RAS was used.
virtual void squash(InstSeqNum seq_num, ThreadID tid)=0
bool wasIndirect
Wether this instruction was an indirect branch.
ProbeManager * getProbeManager()
Get the probe manager for this object.
std::unique_ptr< PMU > PMUUPtr
virtual const std::string name() const
unsigned RASIndex
The RAS index of the instruction (only valid if a call).
bool wasCall
Whether or not the instruction was a call.
BranchPredictorParams Params
Derived & precision(int _precision)
Set the precision and marks this stat to print at the end of simulation.
GenericISA::DelaySlotPCState< MachInst > PCState
Stats::Scalar BTBLookups
Stat for number of BTB lookups.
Stats::Formula BTBHitRatio
Stat for the ratio between BTB hits and BTB lookups.
bool predict(const StaticInstPtr &inst, const InstSeqNum &seqNum, TheISA::PCState &pc, ThreadID tid)
Predicts whether or not the instruction is a taken branch, and the target of the branch if it is take...
TheISA::PCState lookup(Addr instPC, ThreadID tid)
Looks up an address in the BTB.
const Params & params() const
ProbePoints::PMUUPtr ppBranches
Branches seen by the branch predictor.
virtual void deleteIndirectInfo(ThreadID tid, void *indirect_history)=0
void squash(const InstSeqNum &squashed_sn, ThreadID tid)
Squashes all outstanding updates until a given sequence number.
bool predTaken
Whether or not it was predicted taken.
bool isUncondCtrl() const
Abstract superclass for simulation objects.
Generated on Tue Mar 23 2021 19:41:25 for gem5 by doxygen 1.8.17