1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef ART_COMPILER_OPTIMIZING_NODES_X86_H_ 18 #define ART_COMPILER_OPTIMIZING_NODES_X86_H_ 19 20 namespace art HIDDEN { 21 22 // Compute the address of the method for X86 Constant area support. 23 class HX86ComputeBaseMethodAddress final : public HExpression<0> { 24 public: 25 // Treat the value as an int32_t, but it is really a 32 bit native pointer. HX86ComputeBaseMethodAddress()26 HX86ComputeBaseMethodAddress() 27 : HExpression(kX86ComputeBaseMethodAddress, 28 DataType::Type::kInt32, 29 SideEffects::None(), 30 kNoDexPc) { 31 } 32 CanBeMoved()33 bool CanBeMoved() const override { return true; } 34 35 DECLARE_INSTRUCTION(X86ComputeBaseMethodAddress); 36 37 protected: 38 DEFAULT_COPY_CONSTRUCTOR(X86ComputeBaseMethodAddress); 39 }; 40 41 // Load a constant value from the constant table. 42 class HX86LoadFromConstantTable final : public HExpression<2> { 43 public: HX86LoadFromConstantTable(HX86ComputeBaseMethodAddress * method_base,HConstant * constant)44 HX86LoadFromConstantTable(HX86ComputeBaseMethodAddress* method_base, 45 HConstant* constant) 46 : HExpression(kX86LoadFromConstantTable, 47 constant->GetType(), 48 SideEffects::None(), 49 kNoDexPc) { 50 SetRawInputAt(0, method_base); 51 SetRawInputAt(1, constant); 52 } 53 GetBaseMethodAddress()54 HX86ComputeBaseMethodAddress* GetBaseMethodAddress() const { 55 return InputAt(0)->AsX86ComputeBaseMethodAddress(); 56 } 57 GetConstant()58 HConstant* GetConstant() const { 59 return InputAt(1)->AsConstant(); 60 } 61 62 DECLARE_INSTRUCTION(X86LoadFromConstantTable); 63 64 protected: 65 DEFAULT_COPY_CONSTRUCTOR(X86LoadFromConstantTable); 66 }; 67 68 // Version of HNeg with access to the constant table for FP types. 69 class HX86FPNeg final : public HExpression<2> { 70 public: HX86FPNeg(DataType::Type result_type,HInstruction * input,HX86ComputeBaseMethodAddress * method_base,uint32_t dex_pc)71 HX86FPNeg(DataType::Type result_type, 72 HInstruction* input, 73 HX86ComputeBaseMethodAddress* method_base, 74 uint32_t dex_pc) 75 : HExpression(kX86FPNeg, result_type, SideEffects::None(), dex_pc) { 76 DCHECK(DataType::IsFloatingPointType(result_type)); 77 SetRawInputAt(0, input); 78 SetRawInputAt(1, method_base); 79 } 80 GetBaseMethodAddress()81 HX86ComputeBaseMethodAddress* GetBaseMethodAddress() const { 82 return InputAt(1)->AsX86ComputeBaseMethodAddress(); 83 } 84 85 DECLARE_INSTRUCTION(X86FPNeg); 86 87 protected: 88 DEFAULT_COPY_CONSTRUCTOR(X86FPNeg); 89 }; 90 91 // X86 version of HPackedSwitch that holds a pointer to the base method address. 92 class HX86PackedSwitch final : public HExpression<2> { 93 public: HX86PackedSwitch(int32_t start_value,int32_t num_entries,HInstruction * input,HX86ComputeBaseMethodAddress * method_base,uint32_t dex_pc)94 HX86PackedSwitch(int32_t start_value, 95 int32_t num_entries, 96 HInstruction* input, 97 HX86ComputeBaseMethodAddress* method_base, 98 uint32_t dex_pc) 99 : HExpression(kX86PackedSwitch, SideEffects::None(), dex_pc), 100 start_value_(start_value), 101 num_entries_(num_entries) { 102 SetRawInputAt(0, input); 103 SetRawInputAt(1, method_base); 104 } 105 IsControlFlow()106 bool IsControlFlow() const override { return true; } 107 GetStartValue()108 int32_t GetStartValue() const { return start_value_; } 109 GetNumEntries()110 int32_t GetNumEntries() const { return num_entries_; } 111 GetBaseMethodAddress()112 HX86ComputeBaseMethodAddress* GetBaseMethodAddress() const { 113 return InputAt(1)->AsX86ComputeBaseMethodAddress(); 114 } 115 GetDefaultBlock()116 HBasicBlock* GetDefaultBlock() const { 117 // Last entry is the default block. 118 return GetBlock()->GetSuccessors()[num_entries_]; 119 } 120 121 DECLARE_INSTRUCTION(X86PackedSwitch); 122 123 protected: 124 DEFAULT_COPY_CONSTRUCTOR(X86PackedSwitch); 125 126 private: 127 const int32_t start_value_; 128 const int32_t num_entries_; 129 }; 130 131 class HX86AndNot final : public HBinaryOperation { 132 public: 133 HX86AndNot(DataType::Type result_type, 134 HInstruction* left, 135 HInstruction* right, 136 uint32_t dex_pc = kNoDexPc) HBinaryOperation(kX86AndNot,result_type,left,right,SideEffects::None (),dex_pc)137 : HBinaryOperation(kX86AndNot, result_type, left, right, SideEffects::None(), dex_pc) { 138 } 139 IsCommutative()140 bool IsCommutative() const override { return false; } 141 Compute(T x,T y)142 template <typename T> static T Compute(T x, T y) { return ~x & y; } 143 Evaluate(HIntConstant * x,HIntConstant * y)144 HConstant* Evaluate(HIntConstant* x, HIntConstant* y) const override { 145 return GetBlock()->GetGraph()->GetIntConstant( 146 Compute(x->GetValue(), y->GetValue()), GetDexPc()); 147 } Evaluate(HLongConstant * x,HLongConstant * y)148 HConstant* Evaluate(HLongConstant* x, HLongConstant* y) const override { 149 return GetBlock()->GetGraph()->GetLongConstant( 150 Compute(x->GetValue(), y->GetValue()), GetDexPc()); 151 } Evaluate(HFloatConstant * x,HFloatConstant * y)152 HConstant* Evaluate([[maybe_unused]] HFloatConstant* x, 153 [[maybe_unused]] HFloatConstant* y) const override { 154 LOG(FATAL) << DebugName() << " is not defined for float values"; 155 UNREACHABLE(); 156 } Evaluate(HDoubleConstant * x,HDoubleConstant * y)157 HConstant* Evaluate([[maybe_unused]] HDoubleConstant* x, 158 [[maybe_unused]] HDoubleConstant* y) const override { 159 LOG(FATAL) << DebugName() << " is not defined for double values"; 160 UNREACHABLE(); 161 } 162 163 DECLARE_INSTRUCTION(X86AndNot); 164 165 protected: 166 DEFAULT_COPY_CONSTRUCTOR(X86AndNot); 167 }; 168 169 class HX86MaskOrResetLeastSetBit final : public HUnaryOperation { 170 public: 171 HX86MaskOrResetLeastSetBit(DataType::Type result_type, InstructionKind op, 172 HInstruction* input, uint32_t dex_pc = kNoDexPc) HUnaryOperation(kX86MaskOrResetLeastSetBit,result_type,input,dex_pc)173 : HUnaryOperation(kX86MaskOrResetLeastSetBit, result_type, input, dex_pc), 174 op_kind_(op) { 175 DCHECK_EQ(result_type, DataType::Kind(input->GetType())); 176 DCHECK(op == HInstruction::kAnd || op == HInstruction::kXor) << op; 177 } 178 template <typename T> 179 auto Compute(T x) const -> decltype(x & (x-1)) { 180 static_assert(std::is_same<decltype(x & (x-1)), decltype(x ^(x-1))>::value, 181 "Inconsistent bitwise types"); 182 switch (op_kind_) { 183 case HInstruction::kAnd: 184 return x & (x-1); 185 case HInstruction::kXor: 186 return x ^ (x-1); 187 default: 188 LOG(FATAL) << "Unreachable"; 189 UNREACHABLE(); 190 } 191 } 192 Evaluate(HIntConstant * x)193 HConstant* Evaluate(HIntConstant* x) const override { 194 return GetBlock()->GetGraph()->GetIntConstant(Compute(x->GetValue()), GetDexPc()); 195 } Evaluate(HLongConstant * x)196 HConstant* Evaluate(HLongConstant* x) const override { 197 return GetBlock()->GetGraph()->GetLongConstant(Compute(x->GetValue()), GetDexPc()); 198 } Evaluate(HFloatConstant * x)199 HConstant* Evaluate([[maybe_unused]] HFloatConstant* x) const override { 200 LOG(FATAL) << DebugName() << "is not defined for float values"; 201 UNREACHABLE(); 202 } Evaluate(HDoubleConstant * x)203 HConstant* Evaluate([[maybe_unused]] HDoubleConstant* x) const override { 204 LOG(FATAL) << DebugName() << "is not defined for double values"; 205 UNREACHABLE(); 206 } GetOpKind()207 InstructionKind GetOpKind() const { return op_kind_; } 208 209 DECLARE_INSTRUCTION(X86MaskOrResetLeastSetBit); 210 211 protected: 212 const InstructionKind op_kind_; 213 214 DEFAULT_COPY_CONSTRUCTOR(X86MaskOrResetLeastSetBit); 215 }; 216 217 } // namespace art 218 219 #endif // ART_COMPILER_OPTIMIZING_NODES_X86_H_ 220