1*f5c631daSSadaf Ebrahimi // Copyright 2018, VIXL authors 2*f5c631daSSadaf Ebrahimi // All rights reserved. 3*f5c631daSSadaf Ebrahimi // 4*f5c631daSSadaf Ebrahimi // Redistribution and use in source and binary forms, with or without 5*f5c631daSSadaf Ebrahimi // modification, are permitted provided that the following conditions are met: 6*f5c631daSSadaf Ebrahimi // 7*f5c631daSSadaf Ebrahimi // * Redistributions of source code must retain the above copyright notice, 8*f5c631daSSadaf Ebrahimi // this list of conditions and the following disclaimer. 9*f5c631daSSadaf Ebrahimi // * Redistributions in binary form must reproduce the above copyright notice, 10*f5c631daSSadaf Ebrahimi // this list of conditions and the following disclaimer in the documentation 11*f5c631daSSadaf Ebrahimi // and/or other materials provided with the distribution. 12*f5c631daSSadaf Ebrahimi // * Neither the name of Arm Limited nor the names of its contributors may be 13*f5c631daSSadaf Ebrahimi // used to endorse or promote products derived from this software without 14*f5c631daSSadaf Ebrahimi // specific prior written permission. 15*f5c631daSSadaf Ebrahimi // 16*f5c631daSSadaf Ebrahimi // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS CONTRIBUTORS "AS IS" AND 17*f5c631daSSadaf Ebrahimi // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 18*f5c631daSSadaf Ebrahimi // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 19*f5c631daSSadaf Ebrahimi // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE 20*f5c631daSSadaf Ebrahimi // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21*f5c631daSSadaf Ebrahimi // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 22*f5c631daSSadaf Ebrahimi // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 23*f5c631daSSadaf Ebrahimi // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 24*f5c631daSSadaf Ebrahimi // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 25*f5c631daSSadaf Ebrahimi // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 26*f5c631daSSadaf Ebrahimi 27*f5c631daSSadaf Ebrahimi #ifndef VIXL_AARCH64_CPU_FEATURES_AUDITOR_AARCH64_H_ 28*f5c631daSSadaf Ebrahimi #define VIXL_AARCH64_CPU_FEATURES_AUDITOR_AARCH64_H_ 29*f5c631daSSadaf Ebrahimi 30*f5c631daSSadaf Ebrahimi #include <functional> 31*f5c631daSSadaf Ebrahimi #include <iostream> 32*f5c631daSSadaf Ebrahimi #include <unordered_map> 33*f5c631daSSadaf Ebrahimi 34*f5c631daSSadaf Ebrahimi #include "cpu-features.h" 35*f5c631daSSadaf Ebrahimi #include "decoder-aarch64.h" 36*f5c631daSSadaf Ebrahimi #include "decoder-visitor-map-aarch64.h" 37*f5c631daSSadaf Ebrahimi 38*f5c631daSSadaf Ebrahimi namespace vixl { 39*f5c631daSSadaf Ebrahimi namespace aarch64 { 40*f5c631daSSadaf Ebrahimi 41*f5c631daSSadaf Ebrahimi // This visitor records the CPU features that each decoded instruction requires. 42*f5c631daSSadaf Ebrahimi // It provides: 43*f5c631daSSadaf Ebrahimi // - the set of CPU features required by the most recently decoded instruction, 44*f5c631daSSadaf Ebrahimi // - a cumulative set of encountered CPU features, 45*f5c631daSSadaf Ebrahimi // - an optional list of 'available' CPU features. 46*f5c631daSSadaf Ebrahimi // 47*f5c631daSSadaf Ebrahimi // Primarily, this allows the Disassembler and Simulator to share the same CPU 48*f5c631daSSadaf Ebrahimi // features logic. However, it can be used standalone to scan code blocks for 49*f5c631daSSadaf Ebrahimi // CPU features. 50*f5c631daSSadaf Ebrahimi class CPUFeaturesAuditor : public DecoderVisitor { 51*f5c631daSSadaf Ebrahimi public: 52*f5c631daSSadaf Ebrahimi // Construction arguments: 53*f5c631daSSadaf Ebrahimi // - If a decoder is specified, the CPUFeaturesAuditor automatically 54*f5c631daSSadaf Ebrahimi // registers itself as a visitor. Otherwise, this can be done manually. 55*f5c631daSSadaf Ebrahimi // 56*f5c631daSSadaf Ebrahimi // - If an `available` features list is provided, it is used as a hint in 57*f5c631daSSadaf Ebrahimi // cases where instructions may be provided by multiple separate features. 58*f5c631daSSadaf Ebrahimi // An example of this is FP&SIMD loads and stores: some of these are used 59*f5c631daSSadaf Ebrahimi // in both FP and integer SIMD code. If exactly one of those features is 60*f5c631daSSadaf Ebrahimi // in `available` when one of these instructions is encountered, then the 61*f5c631daSSadaf Ebrahimi // auditor will record that feature. Otherwise, it will record _both_ 62*f5c631daSSadaf Ebrahimi // features. 63*f5c631daSSadaf Ebrahimi explicit CPUFeaturesAuditor( 64*f5c631daSSadaf Ebrahimi Decoder* decoder, const CPUFeatures& available = CPUFeatures::None()) available_(available)65*f5c631daSSadaf Ebrahimi : available_(available), decoder_(decoder) { 66*f5c631daSSadaf Ebrahimi if (decoder_ != NULL) decoder_->AppendVisitor(this); 67*f5c631daSSadaf Ebrahimi } 68*f5c631daSSadaf Ebrahimi 69*f5c631daSSadaf Ebrahimi explicit CPUFeaturesAuditor( 70*f5c631daSSadaf Ebrahimi const CPUFeatures& available = CPUFeatures::None()) available_(available)71*f5c631daSSadaf Ebrahimi : available_(available), decoder_(NULL) {} 72*f5c631daSSadaf Ebrahimi ~CPUFeaturesAuditor()73*f5c631daSSadaf Ebrahimi virtual ~CPUFeaturesAuditor() { 74*f5c631daSSadaf Ebrahimi if (decoder_ != NULL) decoder_->RemoveVisitor(this); 75*f5c631daSSadaf Ebrahimi } 76*f5c631daSSadaf Ebrahimi ResetSeenFeatures()77*f5c631daSSadaf Ebrahimi void ResetSeenFeatures() { 78*f5c631daSSadaf Ebrahimi seen_ = CPUFeatures::None(); 79*f5c631daSSadaf Ebrahimi last_instruction_ = CPUFeatures::None(); 80*f5c631daSSadaf Ebrahimi } 81*f5c631daSSadaf Ebrahimi 82*f5c631daSSadaf Ebrahimi // Query or set available CPUFeatures. GetAvailableFeatures()83*f5c631daSSadaf Ebrahimi const CPUFeatures& GetAvailableFeatures() const { return available_; } SetAvailableFeatures(const CPUFeatures & available)84*f5c631daSSadaf Ebrahimi void SetAvailableFeatures(const CPUFeatures& available) { 85*f5c631daSSadaf Ebrahimi available_ = available; 86*f5c631daSSadaf Ebrahimi } 87*f5c631daSSadaf Ebrahimi 88*f5c631daSSadaf Ebrahimi // Query CPUFeatures seen since construction (or the last call to `Reset()`). GetSeenFeatures()89*f5c631daSSadaf Ebrahimi const CPUFeatures& GetSeenFeatures() const { return seen_; } 90*f5c631daSSadaf Ebrahimi 91*f5c631daSSadaf Ebrahimi // Query CPUFeatures from the last instruction visited by this auditor. GetInstructionFeatures()92*f5c631daSSadaf Ebrahimi const CPUFeatures& GetInstructionFeatures() const { 93*f5c631daSSadaf Ebrahimi return last_instruction_; 94*f5c631daSSadaf Ebrahimi } 95*f5c631daSSadaf Ebrahimi InstructionIsAvailable()96*f5c631daSSadaf Ebrahimi bool InstructionIsAvailable() const { 97*f5c631daSSadaf Ebrahimi return available_.Has(last_instruction_); 98*f5c631daSSadaf Ebrahimi } 99*f5c631daSSadaf Ebrahimi 100*f5c631daSSadaf Ebrahimi // The common CPUFeatures interface operates on the available_ list. GetCPUFeatures()101*f5c631daSSadaf Ebrahimi CPUFeatures* GetCPUFeatures() { return &available_; } SetCPUFeatures(const CPUFeatures & available)102*f5c631daSSadaf Ebrahimi void SetCPUFeatures(const CPUFeatures& available) { 103*f5c631daSSadaf Ebrahimi SetAvailableFeatures(available); 104*f5c631daSSadaf Ebrahimi } 105*f5c631daSSadaf Ebrahimi 106*f5c631daSSadaf Ebrahimi virtual void Visit(Metadata* metadata, 107*f5c631daSSadaf Ebrahimi const Instruction* instr) VIXL_OVERRIDE; 108*f5c631daSSadaf Ebrahimi 109*f5c631daSSadaf Ebrahimi private: 110*f5c631daSSadaf Ebrahimi class RecordInstructionFeaturesScope; 111*f5c631daSSadaf Ebrahimi 112*f5c631daSSadaf Ebrahimi #define DECLARE(A) virtual void Visit##A(const Instruction* instr); 113*f5c631daSSadaf Ebrahimi VISITOR_LIST(DECLARE) 114*f5c631daSSadaf Ebrahimi #undef DECLARE 115*f5c631daSSadaf Ebrahimi 116*f5c631daSSadaf Ebrahimi void LoadStoreHelper(const Instruction* instr); 117*f5c631daSSadaf Ebrahimi void LoadStorePairHelper(const Instruction* instr); 118*f5c631daSSadaf Ebrahimi 119*f5c631daSSadaf Ebrahimi CPUFeatures seen_; 120*f5c631daSSadaf Ebrahimi CPUFeatures last_instruction_; 121*f5c631daSSadaf Ebrahimi CPUFeatures available_; 122*f5c631daSSadaf Ebrahimi 123*f5c631daSSadaf Ebrahimi Decoder* decoder_; 124*f5c631daSSadaf Ebrahimi 125*f5c631daSSadaf Ebrahimi using FormToVisitorFnMap = std::unordered_map< 126*f5c631daSSadaf Ebrahimi uint32_t, 127*f5c631daSSadaf Ebrahimi std::function<void(CPUFeaturesAuditor*, const Instruction*)>>; 128*f5c631daSSadaf Ebrahimi static const FormToVisitorFnMap* GetFormToVisitorFnMap(); 129*f5c631daSSadaf Ebrahimi }; 130*f5c631daSSadaf Ebrahimi 131*f5c631daSSadaf Ebrahimi } // namespace aarch64 132*f5c631daSSadaf Ebrahimi } // namespace vixl 133*f5c631daSSadaf Ebrahimi 134*f5c631daSSadaf Ebrahimi #endif // VIXL_AARCH64_CPU_FEATURES_AUDITOR_AARCH64_H_ 135