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" // Needed here to get TARGET_ARCH_ARM. | 5 #include "vm/globals.h" // Needed here to get TARGET_ARCH_ARM. |
6 #if defined(TARGET_ARCH_ARM) | 6 #if defined(TARGET_ARCH_ARM) |
7 | 7 |
| 8 #include "vm/constants_arm.h" |
| 9 #include "vm/cpu.h" |
8 #include "vm/instructions.h" | 10 #include "vm/instructions.h" |
9 #include "vm/object.h" | 11 #include "vm/object.h" |
10 | 12 |
11 namespace dart { | 13 namespace dart { |
12 | 14 |
13 uword InstructionPattern::Back(int n) const { | 15 CallPattern::CallPattern(uword pc, const Code& code) |
| 16 : end_(reinterpret_cast<uword*>(pc)), |
| 17 pool_index_(DecodePoolIndex()), |
| 18 object_pool_(Array::Handle(code.ObjectPool())) { } |
| 19 |
| 20 |
| 21 uword CallPattern::Back(int n) const { |
14 ASSERT(n > 0); | 22 ASSERT(n > 0); |
15 return *(end_ - n); | 23 return *(end_ - n); |
16 } | 24 } |
17 | 25 |
18 | 26 |
19 CallPattern::CallPattern(uword pc, const Code& code) | |
20 : InstructionPattern(pc), | |
21 pool_index_(DecodePoolIndex()), | |
22 object_pool_(Array::Handle(code.ObjectPool())) { } | |
23 | |
24 | |
25 int CallPattern::DecodePoolIndex() { | 27 int CallPattern::DecodePoolIndex() { |
26 ASSERT(Back(1) == 0xe12fff3e); // Last instruction: blx lr | 28 ASSERT(Back(1) == 0xe12fff3e); // Last instruction: blx lr |
27 // Decode the second to last instruction. | 29 // Decode the second to last instruction. |
28 uword instr = Back(2); | 30 uword instr = Back(2); |
29 int offset = 0; | 31 int offset = 0; |
30 if ((instr & 0xfffff000) == 0xe59ae000) { // ldr lr, [pp, #+offset] | 32 if ((instr & 0xfffff000) == 0xe59ae000) { // ldr lr, [pp, #+offset] |
31 offset = instr & 0xfff; | 33 offset = instr & 0xfff; |
32 } else { | 34 } else { |
33 ASSERT((instr & 0xfffff000) == 0xe59ee000); // ldr lr, [lr, #+offset] | 35 ASSERT((instr & 0xfffff000) == 0xe59ee000); // ldr lr, [lr, #+offset] |
34 offset = instr & 0xfff; | 36 offset = instr & 0xfff; |
(...skipping 28 matching lines...) Expand all Loading... |
63 // The address is stored in the object array as a RawSmi. | 65 // The address is stored in the object array as a RawSmi. |
64 return reinterpret_cast<uword>(target_address.raw()); | 66 return reinterpret_cast<uword>(target_address.raw()); |
65 } | 67 } |
66 | 68 |
67 | 69 |
68 void CallPattern::SetTargetAddress(uword target_address) const { | 70 void CallPattern::SetTargetAddress(uword target_address) const { |
69 ASSERT(Utils::IsAligned(target_address, 4)); | 71 ASSERT(Utils::IsAligned(target_address, 4)); |
70 // The address is stored in the object array as a RawSmi. | 72 // The address is stored in the object array as a RawSmi. |
71 const Smi& smi = Smi::Handle(reinterpret_cast<RawSmi*>(target_address)); | 73 const Smi& smi = Smi::Handle(reinterpret_cast<RawSmi*>(target_address)); |
72 object_pool_.SetAt(pool_index_, smi); | 74 object_pool_.SetAt(pool_index_, smi); |
| 75 // No need to flush the instruction cache, since the code is not modified. |
73 } | 76 } |
74 | 77 |
75 | 78 |
| 79 JumpPattern::JumpPattern(uword pc) : pc_(pc) { } |
| 80 |
| 81 |
76 bool JumpPattern::IsValid() const { | 82 bool JumpPattern::IsValid() const { |
77 UNIMPLEMENTED(); | 83 Instr* movw = Instr::At(pc_ + (0 * Instr::kInstrSize)); // movw ip, target_lo |
78 return false; | 84 Instr* movt = Instr::At(pc_ + (1 * Instr::kInstrSize)); // movw ip, target_lo |
| 85 Instr* bxip = Instr::At(pc_ + (2 * Instr::kInstrSize)); // bx ip |
| 86 return (movw->InstructionBits() & 0xfff0f000) == 0xe300c000 && |
| 87 (movt->InstructionBits() & 0xfff0f000) == 0xe340c000 && |
| 88 (bxip->InstructionBits() & 0xffffffff) == 0xe12fff1c; |
79 } | 89 } |
80 | 90 |
81 | 91 |
82 uword JumpPattern::TargetAddress() const { | 92 uword JumpPattern::TargetAddress() const { |
83 UNIMPLEMENTED(); | 93 Instr* movw = Instr::At(pc_ + (0 * Instr::kInstrSize)); // movw ip, target_lo |
84 return 0; | 94 Instr* movt = Instr::At(pc_ + (1 * Instr::kInstrSize)); // movw ip, target_lo |
| 95 uint16_t target_lo = movw->MovwField(); |
| 96 uint16_t target_hi = movt->MovwField(); |
| 97 return (target_hi << 16) | target_lo; |
85 } | 98 } |
86 | 99 |
87 | 100 |
88 void JumpPattern::SetTargetAddress(uword target) const { | 101 void JumpPattern::SetTargetAddress(uword target_address) const { |
89 UNIMPLEMENTED(); | 102 uint16_t target_lo = target_address & 0xffff; |
| 103 uint16_t target_hi = target_address >> 16; |
| 104 uword movw = 0xe300c000 | ((target_lo >> 12) << 16) | (target_lo & 0xfff); |
| 105 uword movt = 0xe340c000 | ((target_hi >> 12) << 16) | (target_hi & 0xfff); |
| 106 *reinterpret_cast<uword*>(pc_ + (0 * Instr::kInstrSize)) = movw; |
| 107 *reinterpret_cast<uword*>(pc_ + (1 * Instr::kInstrSize)) = movt; |
| 108 CPU::FlushICache(pc_, 2 * Instr::kInstrSize); |
90 } | 109 } |
91 | 110 |
92 } // namespace dart | 111 } // namespace dart |
93 | 112 |
94 #endif // defined TARGET_ARCH_ARM | 113 #endif // defined TARGET_ARCH_ARM |
95 | 114 |
OLD | NEW |