| // Copyright (c) 2013, the Dart project authors.  Please see the AUTHORS file | 
 | // for details. All rights reserved. Use of this source code is governed by a | 
 | // BSD-style license that can be found in the LICENSE file. | 
 |  | 
 | #include "vm/globals.h"  // Needed here to get TARGET_ARCH_ARM. | 
 | #if defined(TARGET_ARCH_ARM) | 
 |  | 
 | #include "vm/code_patcher.h" | 
 |  | 
 | #include "vm/instructions.h" | 
 | #include "vm/object.h" | 
 |  | 
 | namespace dart { | 
 |  | 
 | CodePtr CodePatcher::GetStaticCallTargetAt(uword return_address, | 
 |                                            const Code& code) { | 
 |   ASSERT(code.ContainsInstructionAt(return_address)); | 
 |   CallPattern call(return_address, code); | 
 |   return call.TargetCode(); | 
 | } | 
 |  | 
 | void CodePatcher::PatchStaticCallAt(uword return_address, | 
 |                                     const Code& code, | 
 |                                     const Code& new_target) { | 
 |   ASSERT(code.ContainsInstructionAt(return_address)); | 
 |   CallPattern call(return_address, code); | 
 |   call.SetTargetCode(new_target); | 
 | } | 
 |  | 
 | void CodePatcher::InsertDeoptimizationCallAt(uword start) { | 
 |   UNREACHABLE(); | 
 | } | 
 |  | 
 | CodePtr CodePatcher::GetInstanceCallAt(uword return_address, | 
 |                                        const Code& caller_code, | 
 |                                        Object* data) { | 
 |   ASSERT(caller_code.ContainsInstructionAt(return_address)); | 
 |   ICCallPattern call(return_address, caller_code); | 
 |   if (data != nullptr) { | 
 |     *data = call.Data(); | 
 |   } | 
 |   return call.TargetCode(); | 
 | } | 
 |  | 
 | void CodePatcher::PatchInstanceCallAt(uword return_address, | 
 |                                       const Code& caller_code, | 
 |                                       const Object& data, | 
 |                                       const Code& target) { | 
 |   auto thread = Thread::Current(); | 
 |   thread->isolate_group()->RunWithStoppedMutators([&]() { | 
 |     PatchInstanceCallAtWithMutatorsStopped(thread, return_address, caller_code, | 
 |                                            data, target); | 
 |   }); | 
 | } | 
 |  | 
 | void CodePatcher::PatchInstanceCallAtWithMutatorsStopped( | 
 |     Thread* thread, | 
 |     uword return_address, | 
 |     const Code& caller_code, | 
 |     const Object& data, | 
 |     const Code& target) { | 
 |   ASSERT(caller_code.ContainsInstructionAt(return_address)); | 
 |   ICCallPattern call(return_address, caller_code); | 
 |   call.SetData(data); | 
 |   call.SetTargetCode(target); | 
 | } | 
 |  | 
 | FunctionPtr CodePatcher::GetUnoptimizedStaticCallAt(uword return_address, | 
 |                                                     const Code& caller_code, | 
 |                                                     ICData* ic_data_result) { | 
 |   ASSERT(caller_code.ContainsInstructionAt(return_address)); | 
 |   ICCallPattern static_call(return_address, caller_code); | 
 |   ICData& ic_data = ICData::Handle(); | 
 |   ic_data ^= static_call.Data(); | 
 |   if (ic_data_result != nullptr) { | 
 |     *ic_data_result = ic_data.ptr(); | 
 |   } | 
 |   return ic_data.GetTargetAt(0); | 
 | } | 
 |  | 
 | void CodePatcher::PatchSwitchableCallAt(uword return_address, | 
 |                                         const Code& caller_code, | 
 |                                         const Object& data, | 
 |                                         const Code& target) { | 
 |   auto thread = Thread::Current(); | 
 |   // Ensure all threads are suspended as we update data and target pair. | 
 |   thread->isolate_group()->RunWithStoppedMutators([&]() { | 
 |     PatchSwitchableCallAtWithMutatorsStopped(thread, return_address, | 
 |                                              caller_code, data, target); | 
 |   }); | 
 | } | 
 |  | 
 | void CodePatcher::PatchSwitchableCallAtWithMutatorsStopped( | 
 |     Thread* thread, | 
 |     uword return_address, | 
 |     const Code& caller_code, | 
 |     const Object& data, | 
 |     const Code& target) { | 
 |   if (FLAG_precompiled_mode) { | 
 |     BareSwitchableCallPattern call(return_address); | 
 |     call.SetData(data); | 
 |     call.SetTarget(target); | 
 |   } else { | 
 |     SwitchableCallPattern call(return_address, caller_code); | 
 |     call.SetData(data); | 
 |     call.SetTarget(target); | 
 |   } | 
 | } | 
 |  | 
 | uword CodePatcher::GetSwitchableCallTargetEntryAt(uword return_address, | 
 |                                                   const Code& caller_code) { | 
 |   if (FLAG_precompiled_mode) { | 
 |     BareSwitchableCallPattern call(return_address); | 
 |     return call.target_entry(); | 
 |   } else { | 
 |     SwitchableCallPattern call(return_address, caller_code); | 
 |     return call.target_entry(); | 
 |   } | 
 | } | 
 |  | 
 | ObjectPtr CodePatcher::GetSwitchableCallDataAt(uword return_address, | 
 |                                                const Code& caller_code) { | 
 |   if (FLAG_precompiled_mode) { | 
 |     BareSwitchableCallPattern call(return_address); | 
 |     return call.data(); | 
 |   } else { | 
 |     SwitchableCallPattern call(return_address, caller_code); | 
 |     return call.data(); | 
 |   } | 
 | } | 
 |  | 
 | void CodePatcher::PatchNativeCallAt(uword return_address, | 
 |                                     const Code& code, | 
 |                                     NativeFunction target, | 
 |                                     const Code& trampoline) { | 
 |   Thread::Current()->isolate_group()->RunWithStoppedMutators([&]() { | 
 |     ASSERT(code.ContainsInstructionAt(return_address)); | 
 |     NativeCallPattern call(return_address, code); | 
 |     call.set_target(trampoline); | 
 |     call.set_native_function(target); | 
 |   }); | 
 | } | 
 |  | 
 | CodePtr CodePatcher::GetNativeCallAt(uword return_address, | 
 |                                      const Code& code, | 
 |                                      NativeFunction* target) { | 
 |   ASSERT(code.ContainsInstructionAt(return_address)); | 
 |   NativeCallPattern call(return_address, code); | 
 |   *target = call.native_function(); | 
 |   return call.target(); | 
 | } | 
 |  | 
 | }  // namespace dart | 
 |  | 
 | #endif  // defined TARGET_ARCH_ARM |