[llvm-mca] Allow the definition of multiple register files.

This is a refactoring in preparation for other two changes that will allow
scheduling models to define multiple register files. This is the first step
towards fixing PR36662.

class RegisterFile (in Dispatch.h) now can emulate multiple register files.
Internally, it tracks the number of available physical registers in each
register file (described by class RegisterFileInfo).

Each register file is associated to a list of MCRegisterClass indices. Knowing
the register class indices allows to map physical registers to register files.

The long term goal is to allow processor models to optionally specify how many
register files are implemented via tablegen.

Differential Revision: https://reviews.llvm.org/D44488

llvm-svn: 327798
diff --git a/llvm/tools/llvm-mca/Dispatch.h b/llvm/tools/llvm-mca/Dispatch.h
index b342976..5a17b4d 100644
--- a/llvm/tools/llvm-mca/Dispatch.h
+++ b/llvm/tools/llvm-mca/Dispatch.h
@@ -28,73 +28,127 @@
 class Scheduler;
 class Backend;
 
-/// \brief Keeps track of register definitions.
-///
-/// This class tracks register definitions, and performs register renaming
-/// to break anti dependencies.
-/// By default, there is no limit in the number of register aliases which
-/// can be created for the purpose of register renaming. However, users can
-/// specify at object construction time a limit in the number of temporary
-/// registers which can be used by the register renaming logic.
+/// \brief Manages hardware register files, and tracks data dependencies
+/// between registers.
 class RegisterFile {
   const llvm::MCRegisterInfo &MRI;
-  // Currently used mappings and maximum used mappings.
-  // These are to generate statistics only.
-  unsigned NumUsedMappings;
-  unsigned MaxUsedMappings;
-  // Total number of mappings created over time.
-  unsigned TotalMappingsCreated;
 
-  // The maximum number of register aliases which can be used by the
-  // register renamer. Defaut value for this field is zero.
-  // A value of zero for this field means that there is no limit in the
-  // amount of register mappings which can be created. That is equivalent
-  // to having a theoretically infinite number of temporary registers.
-  unsigned TotalMappings;
+  // Each register file is described by an instance of RegisterMappingTracker.
+  // RegisterMappingTracker tracks the number of register mappings dynamically
+  // allocated during the execution.
+  struct RegisterMappingTracker {
+    // Total number of register mappings that are available for register
+    // renaming. A value of zero for this field means: this register file has
+    // an unbounded number of registers.
+    const unsigned TotalMappings;
+    // Number of mappings that are currently in use.
+    unsigned NumUsedMappings;
+    // Maximum number of register mappings used.
+    unsigned MaxUsedMappings;
+    // Total number of mappings allocated during the entire execution.
+    unsigned TotalMappingsCreated;
 
-  // This map contains an entry for every physical register.
-  // A register index is used as a key value to access a WriteState.
-  // This is how we track RAW dependencies for dispatched
-  // instructions. For every register, we track the last seen write only.
-  // This assumes that all writes fully update both super and sub registers.
-  // We need a flag in MCInstrDesc to check if a write also updates super
-  // registers. We can then have a extra tablegen flag to set for instructions.
-  // This is a separate patch on its own.
-  std::vector<WriteState *> RegisterMappings;
-  // Assumptions are:
-  //  a) a false dependencies is always removed by the register renamer.
-  //  b) the register renamer can create an "infinite" number of mappings.
-  // Since we track the number of mappings created, in future we may
-  // introduce constraints on the number of mappings that can be created.
-  // For example, the maximum number of registers that are available for
-  // register renaming purposes may default to the size of the register file.
+    RegisterMappingTracker(unsigned NumMappings)
+        : TotalMappings(NumMappings), NumUsedMappings(0), MaxUsedMappings(0),
+          TotalMappingsCreated(0) {}
+  };
 
-  // In future, we can extend this design to allow multiple register files, and
-  // apply different restrictions on the register mappings and the number of
-  // temporary registers used by mappings.
+  // This is where information related to the various register files is kept.
+  // This set always contains at least one register file at index #0. That
+  // register file "sees" all the physical registers declared by the target, and
+  // (by default) it allows an unbound number of mappings.
+  // Users can limit the number of mappings that can be created by register file
+  // #0 through the command line flag `-register-file-size`.
+  llvm::SmallVector<RegisterMappingTracker, 4> RegisterFiles;
+
+  // RegisterMapping objects are mainly used to track physical register
+  // definitions. A WriteState object describes a register definition, and it is
+  // used to track RAW dependencies (see Instruction.h).  A RegisterMapping
+  // object also specifies the set of register files.  The mapping between
+  // physreg and register files is done using a "register file mask".
+  //
+  // A register file mask identifies a set of register files. Each bit of the
+  // mask representation references a specific register file.
+  // For example:
+  //    0b0001     -->  Register file #0
+  //    0b0010     -->  Register file #1
+  //    0b0100     -->  Register file #2
+  //
+  // Note that this implementation allows register files to overlap.
+  // The maximum number of register files allowed by this implementation is 32.
+  using RegisterMapping = std::pair<WriteState *, unsigned>;
+
+  // This map contains one entry for each physical register defined by the
+  // processor scheduling model.
+  std::vector<RegisterMapping> RegisterMappings;
+
+  // This method creates a new RegisterMappingTracker for a register file that
+  // contains all the physical registers specified by the register classes in
+  // the 'RegisterClasses' set.
+  //
+  // The long term goal is to let scheduling models optionally describe register
+  // files via tablegen definitions. This is still a work in progress.
+  // For example, here is how a tablegen definition for a x86 FP register file
+  // that features AVX might look like:
+  //
+  //    def FPRegisterFile : RegisterFile<[VR128RegClass, VR256RegClass], 60>
+  //
+  // Here FPRegisterFile contains all the registers defined by register class
+  // VR128RegClass and VR256RegClass. FPRegisterFile implements 60
+  // registers which can be used for register renaming purpose.
+  //
+  // The list of register classes is then converted by the tablegen backend into
+  // a list of register class indices. That list, along with the number of
+  // available mappings, is then used to create a new RegisterMappingTracker.
+  void addRegisterFile(llvm::ArrayRef<unsigned> RegisterClasses,
+                       unsigned NumTemps);
+
+  // Allocates a new register mapping in every register file specified by the
+  // register file mask. This method is called from addRegisterMapping.
+  void createNewMappings(unsigned RegisterFileMask);
+
+  // Removes a previously allocated mapping from each register file in the
+  // RegisterFileMask set. This method is called from invalidateRegisterMapping.
+  void removeMappings(unsigned RegisterFileMask);
 
 public:
-  RegisterFile(const llvm::MCRegisterInfo &mri, unsigned Mappings = 0)
-      : MRI(mri), NumUsedMappings(0), MaxUsedMappings(0),
-        TotalMappingsCreated(0), TotalMappings(Mappings),
-        RegisterMappings(MRI.getNumRegs(), nullptr) {}
+  RegisterFile(const llvm::MCRegisterInfo &mri, unsigned TempRegs = 0)
+      : MRI(mri), RegisterMappings(MRI.getNumRegs(), {nullptr, 0U}) {
+    addRegisterFile({}, TempRegs);
+    // TODO: teach the scheduling models how to specify multiple register files.
+  }
 
   // Creates a new register mapping for RegID.
-  // This reserves a temporary register in the register file.
+  // This reserves a microarchitectural register in every register file that
+  // contains RegID.
   void addRegisterMapping(WriteState &WS);
 
   // Invalidates register mappings associated to the input WriteState object.
-  // This releases temporary registers in the register file.
+  // This releases previously allocated mappings for the physical register
+  // associated to the WriteState.
   void invalidateRegisterMapping(const WriteState &WS);
 
-  bool isAvailable(unsigned NumRegWrites);
+  // Checks if there are enough microarchitectural registers in the register
+  // files.  Returns a "response mask" where each bit is the response from a
+  // RegisterMappingTracker.
+  // For example: if all register files are available, then the response mask
+  // is a bitmask of all zeroes. If Instead register file #1 is not available,
+  // then the response mask is 0b10.
+  unsigned isAvailable(const llvm::ArrayRef<unsigned> Regs) const;
   void collectWrites(llvm::SmallVectorImpl<WriteState *> &Writes,
                      unsigned RegID) const;
   void updateOnRead(ReadState &RS, unsigned RegID);
-  unsigned getMaxUsedRegisterMappings() const { return MaxUsedMappings; }
-  unsigned getTotalRegisterMappingsCreated() const {
-    return TotalMappingsCreated;
+  unsigned getMaxUsedRegisterMappings(unsigned RegisterFileIndex) const {
+    assert(RegisterFileIndex < getNumRegisterFiles() &&
+           "Invalid register file index!");
+    return RegisterFiles[RegisterFileIndex].MaxUsedMappings;
   }
+  unsigned getTotalRegisterMappingsCreated(unsigned RegisterFileIndex) const {
+    assert(RegisterFileIndex < getNumRegisterFiles() &&
+           "Invalid register file index!");
+    return RegisterFiles[RegisterFileIndex].TotalMappingsCreated;
+  }
+  unsigned getNumRegisterFiles() const { return RegisterFiles.size(); }
 
 #ifndef NDEBUG
   void dump() const;
@@ -291,11 +345,11 @@
   unsigned getNumDispatchGroupStalls() const {
     return DispatchStalls[DS_DISPATCH_GROUP_RESTRICTION];
   }
-  unsigned getMaxUsedRegisterMappings() const {
-    return RAT->getMaxUsedRegisterMappings();
+  unsigned getMaxUsedRegisterMappings(unsigned RegFileIndex = 0) const {
+    return RAT->getMaxUsedRegisterMappings(RegFileIndex);
   }
-  unsigned getTotalRegisterMappingsCreated() const {
-    return RAT->getTotalRegisterMappingsCreated();
+  unsigned getTotalRegisterMappingsCreated(unsigned RegFileIndex = 0) const {
+    return RAT->getTotalRegisterMappingsCreated(RegFileIndex);
   }
   void addNewRegisterMapping(WriteState &WS) { RAT->addRegisterMapping(WS); }