|
1 | 1 | /*
|
2 |
| - * Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved. |
| 2 | + * Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved. |
3 | 3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
4 | 4 | *
|
5 | 5 | * This code is free software; you can redistribute it and/or modify it
|
|
23 | 23 | */
|
24 | 24 |
|
25 | 25 | #include "precompiled.hpp"
|
| 26 | +#include "asm/assembler.inline.hpp" |
| 27 | +#include "code/codeBlob.hpp" |
| 28 | +#include "code/nativeInst.hpp" |
| 29 | +#include "code/nmethod.hpp" |
26 | 30 | #include "gc/shared/barrierSetNMethod.hpp"
|
27 | 31 | #include "utilities/debug.hpp"
|
28 | 32 |
|
| 33 | +class NativeMethodBarrier: public NativeInstruction { |
| 34 | + private: |
| 35 | + static const int PATCHABLE_INSTRUCTION_OFFSET = 3*6; // bytes |
| 36 | + |
| 37 | + address get_barrier_start_address() const { |
| 38 | + return NativeInstruction::addr_at(0); |
| 39 | + } |
| 40 | + |
| 41 | + address get_patchable_data_address() const { |
| 42 | + address inst_addr = get_barrier_start_address() + PATCHABLE_INSTRUCTION_OFFSET; |
| 43 | + |
| 44 | + debug_only(Assembler::is_z_cfi(*((long*)inst_addr))); |
| 45 | + return inst_addr + 2; |
| 46 | + } |
| 47 | + |
| 48 | + public: |
| 49 | + static const int BARRIER_TOTAL_LENGTH = PATCHABLE_INSTRUCTION_OFFSET + 2*6 + 2; // bytes |
| 50 | + |
| 51 | + int get_guard_value() const { |
| 52 | + address data_addr = get_patchable_data_address(); |
| 53 | + // Return guard instruction value |
| 54 | + return *((int32_t*)data_addr); |
| 55 | + } |
| 56 | + |
| 57 | + void set_guard_value(int value) { |
| 58 | + int32_t* data_addr = (int32_t*)get_patchable_data_address(); |
| 59 | + |
| 60 | + // Set guard instruction value |
| 61 | + *data_addr = value; |
| 62 | + } |
| 63 | + |
| 64 | + #ifdef ASSERT |
| 65 | + void verify() const { |
| 66 | + int offset = 0; // bytes |
| 67 | + const address start = get_barrier_start_address(); |
| 68 | + |
| 69 | + MacroAssembler::is_load_const(/* address */ start + offset); // two instructions |
| 70 | + offset += Assembler::instr_len(&start[offset]); |
| 71 | + offset += Assembler::instr_len(&start[offset]); |
| 72 | + |
| 73 | + Assembler::is_z_lg(*((long*)(start + offset))); |
| 74 | + offset += Assembler::instr_len(&start[offset]); |
| 75 | + |
| 76 | + Assembler::is_z_cfi(*((long*)(start + offset))); |
| 77 | + offset += Assembler::instr_len(&start[offset]); |
| 78 | + |
| 79 | + Assembler::is_z_larl(*((long*)(start + offset))); |
| 80 | + offset += Assembler::instr_len(&start[offset]); |
| 81 | + |
| 82 | + Assembler::is_z_bcr(*((long*)(start + offset))); |
| 83 | + offset += Assembler::instr_len(&start[offset]); |
| 84 | + |
| 85 | + assert(offset == BARRIER_TOTAL_LENGTH, "check offset == barrier length constant"); |
| 86 | + } |
| 87 | + #endif |
| 88 | + |
| 89 | +}; |
| 90 | + |
| 91 | +static NativeMethodBarrier* get_nmethod_barrier(nmethod* nm) { |
| 92 | + address barrier_address = nm->code_begin() + nm->frame_complete_offset() - NativeMethodBarrier::BARRIER_TOTAL_LENGTH; |
| 93 | + auto barrier = reinterpret_cast<NativeMethodBarrier*>(barrier_address); |
| 94 | + |
| 95 | + debug_only(barrier->verify()); |
| 96 | + return barrier; |
| 97 | +} |
| 98 | + |
29 | 99 | void BarrierSetNMethod::deoptimize(nmethod* nm, address* return_address_ptr) {
|
30 |
| - ShouldNotReachHere(); |
| 100 | + // Not required on s390 as a valid backchain is present |
| 101 | + return; |
| 102 | +} |
| 103 | + |
| 104 | +void BarrierSetNMethod::arm(nmethod* nm, int arm_value) { |
| 105 | + if (!supports_entry_barrier(nm)) { |
| 106 | + return; |
| 107 | + } |
| 108 | + |
| 109 | + NativeMethodBarrier* barrier = get_nmethod_barrier(nm); |
| 110 | + barrier->set_guard_value(arm_value); |
31 | 111 | }
|
32 | 112 |
|
33 | 113 | void BarrierSetNMethod::disarm(nmethod* nm) {
|
34 |
| - ShouldNotReachHere(); |
| 114 | + if (!supports_entry_barrier(nm)) { |
| 115 | + return; |
| 116 | + } |
| 117 | + |
| 118 | + NativeMethodBarrier* barrier = get_nmethod_barrier(nm); |
| 119 | + barrier->set_guard_value(disarmed_value()); |
35 | 120 | }
|
36 | 121 |
|
37 | 122 | bool BarrierSetNMethod::is_armed(nmethod* nm) {
|
38 |
| - ShouldNotReachHere(); |
39 |
| - return false; |
| 123 | + if (!supports_entry_barrier(nm)) { |
| 124 | + return false; |
| 125 | + } |
| 126 | + |
| 127 | + NativeMethodBarrier* barrier = get_nmethod_barrier(nm); |
| 128 | + return barrier->get_guard_value() != disarmed_value(); |
40 | 129 | }
|
0 commit comments