| OLD | NEW |
| 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 #include "vm/globals.h" | 5 #include "vm/globals.h" |
| 6 #if defined(TARGET_ARCH_ARM) | 6 #if defined(TARGET_ARCH_ARM) |
| 7 | 7 |
| 8 #include "vm/assembler.h" | 8 #include "vm/assembler.h" |
| 9 #include "vm/code_generator.h" | 9 #include "vm/code_generator.h" |
| 10 #include "vm/cpu.h" | 10 #include "vm/cpu.h" |
| (...skipping 2013 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2024 } | 2024 } |
| 2025 | 2025 |
| 2026 | 2026 |
| 2027 // Called from megamorphic calls. | 2027 // Called from megamorphic calls. |
| 2028 // R0: receiver | 2028 // R0: receiver |
| 2029 // R9: MegamorphicCache (preserved) | 2029 // R9: MegamorphicCache (preserved) |
| 2030 // Passed to target: | 2030 // Passed to target: |
| 2031 // CODE_REG: target Code | 2031 // CODE_REG: target Code |
| 2032 // R4: arguments descriptor | 2032 // R4: arguments descriptor |
| 2033 void StubCode::GenerateMegamorphicCallStub(Assembler* assembler) { | 2033 void StubCode::GenerateMegamorphicCallStub(Assembler* assembler) { |
| 2034 __ NoMonomorphicCheckedEntry(); | |
| 2035 | |
| 2036 __ LoadTaggedClassIdMayBeSmi(R0, R0); | 2034 __ LoadTaggedClassIdMayBeSmi(R0, R0); |
| 2037 // R0: receiver cid as Smi. | 2035 // R0: receiver cid as Smi. |
| 2038 __ ldr(R2, FieldAddress(R9, MegamorphicCache::buckets_offset())); | 2036 __ ldr(R2, FieldAddress(R9, MegamorphicCache::buckets_offset())); |
| 2039 __ ldr(R1, FieldAddress(R9, MegamorphicCache::mask_offset())); | 2037 __ ldr(R1, FieldAddress(R9, MegamorphicCache::mask_offset())); |
| 2040 // R2: cache buckets array. | 2038 // R2: cache buckets array. |
| 2041 // R1: mask. | 2039 // R1: mask. |
| 2042 | 2040 |
| 2043 // Compute the table index. | 2041 // Compute the table index. |
| 2044 ASSERT(MegamorphicCache::kSpreadFactor == 7); | 2042 ASSERT(MegamorphicCache::kSpreadFactor == 7); |
| 2045 // Use reverse substract to multiply with 7 == 8 - 1. | 2043 // Use reverse substract to multiply with 7 == 8 - 1. |
| (...skipping 35 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2081 } | 2079 } |
| 2082 | 2080 |
| 2083 | 2081 |
| 2084 // Called from switchable IC calls. | 2082 // Called from switchable IC calls. |
| 2085 // R0: receiver | 2083 // R0: receiver |
| 2086 // R9: ICData (preserved) | 2084 // R9: ICData (preserved) |
| 2087 // Passed to target: | 2085 // Passed to target: |
| 2088 // CODE_REG: target Code object | 2086 // CODE_REG: target Code object |
| 2089 // R4: arguments descriptor | 2087 // R4: arguments descriptor |
| 2090 void StubCode::GenerateICCallThroughFunctionStub(Assembler* assembler) { | 2088 void StubCode::GenerateICCallThroughFunctionStub(Assembler* assembler) { |
| 2091 __ NoMonomorphicCheckedEntry(); | |
| 2092 | |
| 2093 Label loop, found, miss; | 2089 Label loop, found, miss; |
| 2094 __ ldr(R4, FieldAddress(R9, ICData::arguments_descriptor_offset())); | 2090 __ ldr(R4, FieldAddress(R9, ICData::arguments_descriptor_offset())); |
| 2095 __ ldr(R8, FieldAddress(R9, ICData::ic_data_offset())); | 2091 __ ldr(R8, FieldAddress(R9, ICData::ic_data_offset())); |
| 2096 __ AddImmediate(R8, R8, Array::data_offset() - kHeapObjectTag); | 2092 __ AddImmediate(R8, R8, Array::data_offset() - kHeapObjectTag); |
| 2097 // R8: first IC entry | 2093 // R8: first IC entry |
| 2098 __ LoadTaggedClassIdMayBeSmi(R1, R0); | 2094 __ LoadTaggedClassIdMayBeSmi(R1, R0); |
| 2099 // R1: receiver cid as Smi | 2095 // R1: receiver cid as Smi |
| 2100 | 2096 |
| 2101 __ Bind(&loop); | 2097 __ Bind(&loop); |
| 2102 __ ldr(R2, Address(R8, 0)); | 2098 __ ldr(R2, Address(R8, 0)); |
| (...skipping 15 matching lines...) Expand all Loading... |
| 2118 | 2114 |
| 2119 __ Bind(&miss); | 2115 __ Bind(&miss); |
| 2120 __ LoadIsolate(R2); | 2116 __ LoadIsolate(R2); |
| 2121 __ ldr(CODE_REG, Address(R2, Isolate::ic_miss_code_offset())); | 2117 __ ldr(CODE_REG, Address(R2, Isolate::ic_miss_code_offset())); |
| 2122 __ ldr(R1, FieldAddress(CODE_REG, Code::entry_point_offset())); | 2118 __ ldr(R1, FieldAddress(CODE_REG, Code::entry_point_offset())); |
| 2123 __ bx(R1); | 2119 __ bx(R1); |
| 2124 } | 2120 } |
| 2125 | 2121 |
| 2126 | 2122 |
| 2127 void StubCode::GenerateICCallThroughCodeStub(Assembler* assembler) { | 2123 void StubCode::GenerateICCallThroughCodeStub(Assembler* assembler) { |
| 2128 __ NoMonomorphicCheckedEntry(); | |
| 2129 | |
| 2130 Label loop, found, miss; | 2124 Label loop, found, miss; |
| 2131 __ ldr(R4, FieldAddress(R9, ICData::arguments_descriptor_offset())); | 2125 __ ldr(R4, FieldAddress(R9, ICData::arguments_descriptor_offset())); |
| 2132 __ ldr(R8, FieldAddress(R9, ICData::ic_data_offset())); | 2126 __ ldr(R8, FieldAddress(R9, ICData::ic_data_offset())); |
| 2133 __ AddImmediate(R8, R8, Array::data_offset() - kHeapObjectTag); | 2127 __ AddImmediate(R8, R8, Array::data_offset() - kHeapObjectTag); |
| 2134 // R8: first IC entry | 2128 // R8: first IC entry |
| 2135 __ LoadTaggedClassIdMayBeSmi(R1, R0); | 2129 __ LoadTaggedClassIdMayBeSmi(R1, R0); |
| 2136 // R1: receiver cid as Smi | 2130 // R1: receiver cid as Smi |
| 2137 | 2131 |
| 2138 __ Bind(&loop); | 2132 __ Bind(&loop); |
| 2139 __ ldr(R2, Address(R8, 0)); | 2133 __ ldr(R2, Address(R8, 0)); |
| (...skipping 18 matching lines...) Expand all Loading... |
| 2158 __ ldr(CODE_REG, Address(R2, Isolate::ic_miss_code_offset())); | 2152 __ ldr(CODE_REG, Address(R2, Isolate::ic_miss_code_offset())); |
| 2159 __ ldr(R1, FieldAddress(CODE_REG, Code::entry_point_offset())); | 2153 __ ldr(R1, FieldAddress(CODE_REG, Code::entry_point_offset())); |
| 2160 __ bx(R1); | 2154 __ bx(R1); |
| 2161 } | 2155 } |
| 2162 | 2156 |
| 2163 | 2157 |
| 2164 // Called from switchable IC calls. | 2158 // Called from switchable IC calls. |
| 2165 // R0: receiver | 2159 // R0: receiver |
| 2166 // R9: UnlinkedCall | 2160 // R9: UnlinkedCall |
| 2167 void StubCode::GenerateUnlinkedCallStub(Assembler* assembler) { | 2161 void StubCode::GenerateUnlinkedCallStub(Assembler* assembler) { |
| 2168 __ NoMonomorphicCheckedEntry(); | |
| 2169 | |
| 2170 __ EnterStubFrame(); | 2162 __ EnterStubFrame(); |
| 2171 __ Push(R0); // Preserve receiver. | 2163 __ Push(R0); // Preserve receiver. |
| 2172 | 2164 |
| 2173 __ LoadImmediate(IP, 0); | 2165 __ LoadImmediate(IP, 0); |
| 2174 __ Push(IP); // Result slot | 2166 __ Push(IP); // Result slot |
| 2175 __ Push(R0); // Arg0: Receiver | 2167 __ Push(R0); // Arg0: Receiver |
| 2176 __ Push(R9); // Arg1: UnlinkedCall | 2168 __ Push(R9); // Arg1: UnlinkedCall |
| 2177 __ CallRuntime(kUnlinkedCallRuntimeEntry, 2); | 2169 __ CallRuntime(kUnlinkedCallRuntimeEntry, 2); |
| 2178 __ Drop(2); | 2170 __ Drop(2); |
| 2179 __ Pop(R9); // result = IC | 2171 __ Pop(R9); // result = IC |
| 2180 | 2172 |
| 2181 __ Pop(R0); // Restore receiver. | 2173 __ Pop(R0); // Restore receiver. |
| 2182 __ LeaveStubFrame(); | 2174 __ LeaveStubFrame(); |
| 2183 | 2175 |
| 2184 __ ldr(CODE_REG, Address(THR, Thread::ic_lookup_through_code_stub_offset())); | 2176 __ ldr(CODE_REG, Address(THR, Thread::ic_lookup_through_code_stub_offset())); |
| 2185 __ ldr(R1, FieldAddress(CODE_REG, Code::checked_entry_point_offset())); | 2177 __ ldr(R1, FieldAddress(CODE_REG, Code::checked_entry_point_offset())); |
| 2186 __ bx(R1); | 2178 __ bx(R1); |
| 2187 } | 2179 } |
| 2188 | 2180 |
| 2189 | 2181 |
| 2190 // Called from switchable IC calls. | 2182 // Called from switchable IC calls. |
| 2191 // R0: receiver | 2183 // R0: receiver |
| 2192 // R9: SingleTargetCache | 2184 // R9: SingleTargetCache |
| 2193 // Passed to target: | 2185 // Passed to target: |
| 2194 // CODE_REG: target Code object | 2186 // CODE_REG: target Code object |
| 2195 void StubCode::GenerateSingleTargetCallStub(Assembler* assembler) { | 2187 void StubCode::GenerateSingleTargetCallStub(Assembler* assembler) { |
| 2196 __ NoMonomorphicCheckedEntry(); | |
| 2197 | |
| 2198 Label miss; | 2188 Label miss; |
| 2199 | |
| 2200 __ LoadClassIdMayBeSmi(R1, R0); | 2189 __ LoadClassIdMayBeSmi(R1, R0); |
| 2201 __ ldrh(R2, FieldAddress(R9, SingleTargetCache::lower_limit_offset())); | 2190 __ ldrh(R2, FieldAddress(R9, SingleTargetCache::lower_limit_offset())); |
| 2202 __ ldrh(R3, FieldAddress(R9, SingleTargetCache::upper_limit_offset())); | 2191 __ ldrh(R3, FieldAddress(R9, SingleTargetCache::upper_limit_offset())); |
| 2203 | 2192 |
| 2204 __ cmp(R1, Operand(R2)); | 2193 __ cmp(R1, Operand(R2)); |
| 2205 __ b(&miss, LT); | 2194 __ b(&miss, LT); |
| 2206 __ cmp(R1, Operand(R3)); | 2195 __ cmp(R1, Operand(R3)); |
| 2207 __ b(&miss, GT); | 2196 __ b(&miss, GT); |
| 2208 | 2197 |
| 2209 __ ldr(R1, FieldAddress(R9, SingleTargetCache::entry_point_offset())); | 2198 __ ldr(R1, FieldAddress(R9, SingleTargetCache::entry_point_offset())); |
| (...skipping 43 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2253 } | 2242 } |
| 2254 | 2243 |
| 2255 | 2244 |
| 2256 void StubCode::GenerateFrameAwaitingMaterializationStub(Assembler* assembler) { | 2245 void StubCode::GenerateFrameAwaitingMaterializationStub(Assembler* assembler) { |
| 2257 __ bkpt(0); | 2246 __ bkpt(0); |
| 2258 } | 2247 } |
| 2259 | 2248 |
| 2260 } // namespace dart | 2249 } // namespace dart |
| 2261 | 2250 |
| 2262 #endif // defined TARGET_ARCH_ARM | 2251 #endif // defined TARGET_ARCH_ARM |
| OLD | NEW |