/* * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016 SAP SE. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. * */ // Major contributions by JL, LS #include "precompiled.hpp" #include "asm/macroAssembler.inline.hpp" #include "memory/resourceArea.hpp" #include "nativeInst_s390.hpp" #include "oops/oop.inline.hpp" #include "runtime/handles.hpp" #include "runtime/sharedRuntime.hpp" #include "runtime/stubRoutines.hpp" #include "utilities/ostream.hpp" #ifdef COMPILER1 #include "c1/c1_Runtime1.hpp" #endif #define LUCY_DBG //------------------------------------- // N a t i v e I n s t r u c t i o n //------------------------------------- // Define this switch to prevent identity updates. // In high-concurrency scenarios, it is beneficial to prevent // identity updates. It has a positive effect on cache line steals. // and invalidations. // Test runs of JVM98, JVM2008, and JBB2005 show a very low frequency // of identity updates. Detection is therefore disabled. #undef SUPPRESS_IDENTITY_UPDATE void NativeInstruction::verify() { // Make sure code pattern is actually an instruction address. // Do not allow: // - NULL // - any address in first page (0x0000 .. 0x0fff) // - odd address (will cause a "specification exception") address addr = addr_at(0); if ((addr == 0) || (((unsigned long)addr & ~0x0fff) == 0) || ((intptr_t)addr & 1) != 0) { tty->print_cr(INTPTR_FORMAT ": bad instruction address", p2i(addr)); fatal("not an instruction address"); } } // Print location and value (hex representation) of current NativeInstruction void NativeInstruction::print(const char* msg) const { int len = Assembler::instr_len(addr_at(0)); if (msg == NULL) { // Output line without trailing blanks. switch (len) { case 2: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x", p2i(addr_at(0)), len, halfword_at(0)); break; case 4: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x %4.4x", p2i(addr_at(0)), len, halfword_at(0), halfword_at(2)); break; case 6: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x %4.4x %4.4x", p2i(addr_at(0)), len, halfword_at(0), halfword_at(2), halfword_at(4)); break; default: // Never reached. instr_len() always returns one of the above values. Keep the compiler happy. ShouldNotReachHere(); break; } } else { // Output line with filler blanks to have msg aligned. switch (len) { case 2: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x %s", p2i(addr_at(0)), len, halfword_at(0), msg); break; case 4: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x %4.4x %s", p2i(addr_at(0)), len, halfword_at(0), halfword_at(2), msg); break; case 6: tty->print_cr(INTPTR_FORMAT "(len=%d): %4.4x %4.4x %4.4x %s", p2i(addr_at(0)), len, halfword_at(0), halfword_at(2), halfword_at(4), msg); break; default: // Never reached. instr_len() always returns one of the above values. Keep the compiler happy. ShouldNotReachHere(); break; } } } void NativeInstruction::print() const { print(NULL); } // Hex-Dump of storage around current NativeInstruction. Also try disassembly. void NativeInstruction::dump(const unsigned int range, const char* msg) const { Assembler::dump_code_range(tty, addr_at(0), range, (msg == NULL) ? "":msg); } void NativeInstruction::dump(const unsigned int range) const { dump(range, NULL); } void NativeInstruction::dump() const { dump(32, NULL); } void NativeInstruction::set_halfword_at(int offset, short i) { address addr = addr_at(offset); #ifndef SUPPRESS_IDENTITY_UPDATE *(short*)addr = i; #else if (*(short*)addr != i) { *(short*)addr = i; } #endif ICache::invalidate_word(addr); } void NativeInstruction::set_word_at(int offset, int i) { address addr = addr_at(offset); #ifndef SUPPRESS_IDENTITY_UPDATE *(int*)addr = i; #else if (*(int*)addr != i) { *(int*)addr = i; } #endif ICache::invalidate_word(addr); } void NativeInstruction::set_jlong_at(int offset, jlong i) { address addr = addr_at(offset); #ifndef SUPPRESS_IDENTITY_UPDATE *(jlong*)addr = i; #else if (*(jlong*)addr != i) { *(jlong*)addr = i; } #endif // Don't need to invalidate 2 words here, because // the flush instruction operates on doublewords. ICache::invalidate_word(addr); } #undef SUPPRESS_IDENTITY_UPDATE //------------------------------------------------------------ int NativeInstruction::illegal_instruction() { return 0; } bool NativeInstruction::is_illegal() { // An instruction with main opcode 0x00 (leftmost byte) is not a valid instruction // (and will never be) and causes a SIGILL where the pc points to the next instruction. // The caller of this method wants to know if such a situation exists at the current pc. // // The result of this method is unsharp with respect to the following facts: // - Stepping backwards in the instruction stream is not possible on z/Architecture. // - z/Architecture instructions are 2, 4, or 6 bytes in length. // - The instruction length is coded in the leftmost two bits of the main opcode. // - The result is exact if the caller knows by some other means that the // instruction is of length 2. // // If this method returns false, then the 2-byte instruction at *-2 is not a 0x00 opcode. // If this method returns true, then the 2-byte instruction at *-2 is a 0x00 opcode. return halfword_at(-2) == illegal_instruction(); } // We use an illtrap for marking a method as not_entrant or zombie. bool NativeInstruction::is_sigill_zombie_not_entrant() { if (!is_illegal()) return false; // Just a quick path. // One-sided error of is_illegal tolerable here // (see implementation of is_illegal() for details). CodeBlob* cb = CodeCache::find_blob_unsafe(addr_at(0)); if (cb == NULL || !cb->is_nmethod()) { return false; } nmethod *nm = (nmethod *)cb; // This method is not_entrant or zombie if the illtrap instruction // is located at the verified entry point. // BE AWARE: the current pc (this) points to the instruction after the // "illtrap" location. address sig_addr = ((address) this) - 2; return nm->verified_entry_point() == sig_addr; } bool NativeInstruction::is_jump() { unsigned long inst; Assembler::get_instruction((address)this, &inst); return MacroAssembler::is_branch_pcrelative_long(inst); } //--------------------------------------------------- // N a t i v e I l l e g a l I n s t r u c t i o n //--------------------------------------------------- void NativeIllegalInstruction::insert(address code_pos) { NativeIllegalInstruction* nii = (NativeIllegalInstruction*) nativeInstruction_at(code_pos); nii->set_halfword_at(0, illegal_instruction()); } //----------------------- // N a t i v e C a l l //----------------------- void NativeCall::verify() { if (NativeCall::is_call_at(addr_at(0))) return; fatal("this is not a `NativeCall' site"); } address NativeCall::destination() const { if (MacroAssembler::is_call_far_pcrelative(instruction_address())) { address here = addr_at(MacroAssembler::nop_size()); return MacroAssembler::get_target_addr_pcrel(here); } return (address)((NativeMovConstReg *)this)->data(); } // Similar to replace_mt_safe, but just changes the destination. The // important thing is that free-running threads are able to execute this // call instruction at all times. Thus, the displacement field must be // 4-byte-aligned. We enforce this on z/Architecture by inserting a nop // instruction in front of 'brasl' when needed. // // Used in the runtime linkage of calls; see class CompiledIC. void NativeCall::set_destination_mt_safe(address dest) { if (MacroAssembler::is_call_far_pcrelative(instruction_address())) { address iaddr = addr_at(MacroAssembler::nop_size()); // Ensure that patching is atomic hence mt safe. assert(((long)addr_at(MacroAssembler::call_far_pcrelative_size()) & (call_far_pcrelative_displacement_alignment-1)) == 0, "constant must be 4-byte aligned"); set_word_at(MacroAssembler::call_far_pcrelative_size() - 4, Assembler::z_pcrel_off(dest, iaddr)); } else { assert(MacroAssembler::is_load_const_from_toc(instruction_address()), "unsupported instruction"); nativeMovConstReg_at(instruction_address())->set_data(((intptr_t)dest)); } } //----------------------------- // N a t i v e F a r C a l l //----------------------------- void NativeFarCall::verify() { NativeInstruction::verify(); if (NativeFarCall::is_far_call_at(addr_at(0))) return; fatal("not a NativeFarCall"); } address NativeFarCall::destination() { assert(MacroAssembler::is_call_far_patchable_at((address)this), "unexpected call type"); address ctable = NULL; return MacroAssembler::get_dest_of_call_far_patchable_at((address)this, ctable); } // Handles both patterns of patchable far calls. void NativeFarCall::set_destination(address dest, int toc_offset) { address inst_addr = (address)this; // Set new destination (implementation of call may change here). assert(MacroAssembler::is_call_far_patchable_at(inst_addr), "unexpected call type"); if (!MacroAssembler::is_call_far_patchable_pcrelative_at(inst_addr)) { address ctable = CodeCache::find_blob(inst_addr)->ctable_begin(); // Need distance of TOC entry from current instruction. toc_offset = (ctable + toc_offset) - inst_addr; // Call is via constant table entry. MacroAssembler::set_dest_of_call_far_patchable_at(inst_addr, dest, toc_offset); } else { // Here, we have a pc-relative call (brasl). // Be aware: dest may have moved in this case, so really patch the displacement, // when necessary! // This while loop will also consume the nop which always preceeds a call_far_pcrelative. // We need to revert this after the loop. Pc-relative calls are always assumed to have a leading nop. unsigned int nop_sz = MacroAssembler::nop_size(); unsigned int nop_bytes = 0; while(MacroAssembler::is_z_nop(inst_addr+nop_bytes)) { nop_bytes += nop_sz; } if (nop_bytes > 0) { inst_addr += nop_bytes - nop_sz; } assert(MacroAssembler::is_call_far_pcrelative(inst_addr), "not a pc-relative call"); address target = MacroAssembler::get_target_addr_pcrel(inst_addr + nop_sz); if (target != dest) { NativeCall *call = nativeCall_at(inst_addr); call->set_destination_mt_safe(dest); } } } //------------------------------------- // N a t i v e M o v C o n s t R e g //------------------------------------- // Do not use an assertion here. Let clients decide whether they only // want this when assertions are enabled. void NativeMovConstReg::verify() { address loc = addr_at(0); // This while loop will also consume the nop which always preceeds a // call_far_pcrelative. We need to revert this after the // loop. Pc-relative calls are always assumed to have a leading nop. unsigned int nop_sz = MacroAssembler::nop_size(); unsigned int nop_bytes = 0; while(MacroAssembler::is_z_nop(loc+nop_bytes)) { nop_bytes += nop_sz; } if (nop_bytes > 0) { if (MacroAssembler::is_call_far_pcrelative(loc+nop_bytes-nop_sz)) return; loc += nop_bytes; } if (!MacroAssembler::is_load_const_from_toc(loc) && // Load const from TOC. !MacroAssembler::is_load_const(loc) && // Load const inline. !MacroAssembler::is_load_narrow_oop(loc) && // Load narrow oop. !MacroAssembler::is_load_narrow_klass(loc) && // Load narrow Klass ptr. !MacroAssembler::is_compare_immediate_narrow_oop(loc) && // Compare immediate narrow. !MacroAssembler::is_compare_immediate_narrow_klass(loc) && // Compare immediate narrow. !MacroAssembler::is_pcrelative_instruction(loc)) { // Just to make it run. tty->cr(); tty->print_cr("NativeMovConstReg::verify(): verifying addr %p(0x%x), %d leading nops", loc, *(uint*)loc, nop_bytes/nop_sz); tty->cr(); ((NativeMovConstReg*)loc)->dump(64, "NativeMovConstReg::verify()"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #endif fatal("this is not a `NativeMovConstReg' site"); } } address NativeMovConstReg::next_instruction_address(int offset) const { address inst_addr = addr_at(offset); // Load address (which is a constant) pc-relative. if (MacroAssembler::is_load_addr_pcrel(inst_addr)) { return addr_at(offset+MacroAssembler::load_addr_pcrel_size()); } // Load constant from TOC. if (MacroAssembler::is_load_const_from_toc(inst_addr)) { return addr_at(offset+MacroAssembler::load_const_from_toc_size()); } // Load constant inline. if (MacroAssembler::is_load_const(inst_addr)) { return addr_at(offset+MacroAssembler::load_const_size()); } // Load constant narrow inline. if (MacroAssembler::is_load_narrow_oop(inst_addr)) { return addr_at(offset+MacroAssembler::load_narrow_oop_size()); } if (MacroAssembler::is_load_narrow_klass(inst_addr)) { return addr_at(offset+MacroAssembler::load_narrow_klass_size()); } // Compare constant narrow inline. if (MacroAssembler::is_compare_immediate_narrow_oop(inst_addr)) { return addr_at(offset+MacroAssembler::compare_immediate_narrow_oop_size()); } if (MacroAssembler::is_compare_immediate_narrow_klass(inst_addr)) { return addr_at(offset+MacroAssembler::compare_immediate_narrow_klass_size()); } if (MacroAssembler::is_call_far_patchable_pcrelative_at(inst_addr)) { return addr_at(offset+MacroAssembler::call_far_patchable_size()); } if (MacroAssembler::is_pcrelative_instruction(inst_addr)) { return addr_at(offset+Assembler::instr_len(inst_addr)); } ((NativeMovConstReg*)inst_addr)->dump(64, "NativeMovConstReg site is not recognized as such"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #else guarantee(false, "Not a NativeMovConstReg site"); #endif return NULL; } intptr_t NativeMovConstReg::data() const { address loc = addr_at(0); if (MacroAssembler::is_load_const(loc)) { return MacroAssembler::get_const(loc); } else if (MacroAssembler::is_load_narrow_oop(loc) || MacroAssembler::is_compare_immediate_narrow_oop(loc) || MacroAssembler::is_load_narrow_klass(loc) || MacroAssembler::is_compare_immediate_narrow_klass(loc)) { ((NativeMovConstReg*)loc)->dump(32, "NativeMovConstReg::data(): cannot extract data from narrow ptr (oop or klass)"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #else ShouldNotReachHere(); #endif return *(intptr_t *)NULL; } else { // Otherwise, assume data resides in TOC. Is asserted in called method. return MacroAssembler::get_const_from_toc(loc); } } // Patch in a new constant. // // There are situations where we have multiple (hopefully two at most) // relocations connected to one instruction. Loading an oop from CP // using pcrelative addressing would one such example. Here we have an // oop relocation, modifying the oop itself, and an internal word relocation, // modifying the relative address. // // NativeMovConstReg::set_data is then called once for each relocation. To be // able to distinguish between the relocations, we use a rather dirty hack: // // All calls that deal with an internal word relocation to fix their relative // address are on a faked, odd instruction address. The instruction can be // found on the next lower, even address. // // All other calls are "normal", i.e. on even addresses. address NativeMovConstReg::set_data_plain(intptr_t src, CodeBlob *cb) { unsigned long x = (unsigned long)src; address loc = instruction_address(); address next_address; if (MacroAssembler::is_load_addr_pcrel(loc)) { MacroAssembler::patch_target_addr_pcrel(loc, (address)src); ICache::invalidate_range(loc, MacroAssembler::load_addr_pcrel_size()); next_address = next_instruction_address(); } else if (MacroAssembler::is_load_const_from_toc(loc)) { // Load constant from TOC. MacroAssembler::set_const_in_toc(loc, src, cb); next_address = next_instruction_address(); } else if (MacroAssembler::is_load_const(loc)) { // Not mt safe, ok in methods like CodeBuffer::copy_code(). MacroAssembler::patch_const(loc, x); ICache::invalidate_range(loc, MacroAssembler::load_const_size()); next_address = next_instruction_address(); } // cOops else if (MacroAssembler::is_load_narrow_oop(loc)) { MacroAssembler::patch_load_narrow_oop(loc, (oop) (void*) x); ICache::invalidate_range(loc, MacroAssembler::load_narrow_oop_size()); next_address = next_instruction_address(); } // compressed klass ptrs else if (MacroAssembler::is_load_narrow_klass(loc)) { MacroAssembler::patch_load_narrow_klass(loc, (Klass*)x); ICache::invalidate_range(loc, MacroAssembler::load_narrow_klass_size()); next_address = next_instruction_address(); } // cOops else if (MacroAssembler::is_compare_immediate_narrow_oop(loc)) { MacroAssembler::patch_compare_immediate_narrow_oop(loc, (oop) (void*) x); ICache::invalidate_range(loc, MacroAssembler::compare_immediate_narrow_oop_size()); next_address = next_instruction_address(); } // compressed klass ptrs else if (MacroAssembler::is_compare_immediate_narrow_klass(loc)) { MacroAssembler::patch_compare_immediate_narrow_klass(loc, (Klass*)x); ICache::invalidate_range(loc, MacroAssembler::compare_immediate_narrow_klass_size()); next_address = next_instruction_address(); } else if (MacroAssembler::is_call_far_patchable_pcrelative_at(loc)) { assert(ShortenBranches, "Wait a minute! A pc-relative call w/o ShortenBranches?"); // This NativeMovConstReg site does not need to be patched. It was // patched when it was converted to a call_pcrelative site // before. The value of the src argument is not related to the // branch target. next_address = next_instruction_address(); } else { tty->print_cr("WARNING: detected an unrecognized code pattern at loc = %p -> 0x%8.8x %8.8x", loc, *((unsigned int*)loc), *((unsigned int*)(loc+4))); next_address = next_instruction_address(); // Failure should be handled in next_instruction_address(). #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #endif } return next_address; } // Divided up in set_data_plain() which patches the instruction in the // code stream and set_data() which additionally patches the oop pool // if necessary. void NativeMovConstReg::set_data(intptr_t src) { // Also store the value into an oop_Relocation cell, if any. CodeBlob *cb = CodeCache::find_blob(instruction_address()); address next_address = set_data_plain(src, cb); relocInfo::update_oop_pool(instruction_address(), next_address, (address)src, cb); } void NativeMovConstReg::set_narrow_oop(intptr_t data) { const address start = addr_at(0); int range = 0; if (MacroAssembler::is_load_narrow_oop(start)) { range = MacroAssembler::patch_load_narrow_oop(start, cast_to_oop (data)); } else if (MacroAssembler::is_compare_immediate_narrow_oop(start)) { range = MacroAssembler::patch_compare_immediate_narrow_oop(start, cast_to_oop (data)); } else { fatal("this is not a `NativeMovConstReg::narrow_oop' site"); } ICache::invalidate_range(start, range); } // Compressed klass ptrs. patch narrow klass constant. void NativeMovConstReg::set_narrow_klass(intptr_t data) { const address start = addr_at(0); int range = 0; if (MacroAssembler::is_load_narrow_klass(start)) { range = MacroAssembler::patch_load_narrow_klass(start, (Klass*)data); } else if (MacroAssembler::is_compare_immediate_narrow_klass(start)) { range = MacroAssembler::patch_compare_immediate_narrow_klass(start, (Klass*)data); } else { fatal("this is not a `NativeMovConstReg::narrow_klass' site"); } ICache::invalidate_range(start, range); } void NativeMovConstReg::set_pcrel_addr(intptr_t newTarget, CompiledMethod *passed_nm /* = NULL */, bool copy_back_to_oop_pool) { address next_address; address loc = addr_at(0); if (MacroAssembler::is_load_addr_pcrel(loc)) { address oldTarget = MacroAssembler::get_target_addr_pcrel(loc); MacroAssembler::patch_target_addr_pcrel(loc, (address)newTarget); ICache::invalidate_range(loc, MacroAssembler::load_addr_pcrel_size()); next_address = loc + MacroAssembler::load_addr_pcrel_size(); } else if (MacroAssembler::is_load_const_from_toc_pcrelative(loc) ) { // Load constant from TOC. address oldTarget = MacroAssembler::get_target_addr_pcrel(loc); MacroAssembler::patch_target_addr_pcrel(loc, (address)newTarget); ICache::invalidate_range(loc, MacroAssembler::load_const_from_toc_size()); next_address = loc + MacroAssembler::load_const_from_toc_size(); } else if (MacroAssembler::is_call_far_patchable_pcrelative_at(loc)) { assert(ShortenBranches, "Wait a minute! A pc-relative call w/o ShortenBranches?"); next_address = next_instruction_address(); } else { assert(false, "Not a NativeMovConstReg site for set_pcrel_addr"); next_address = next_instruction_address(); // Failure should be handled in next_instruction_address(). } if (copy_back_to_oop_pool) { if (relocInfo::update_oop_pool(instruction_address(), next_address, (address)newTarget, NULL)) { ((NativeMovConstReg*)instruction_address())->dump(64, "NativeMovConstReg::set_pcrel_addr(): found oop reloc for pcrel_addr"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #else assert(false, "Ooooops: found oop reloc for pcrel_addr"); #endif } } } void NativeMovConstReg::set_pcrel_data(intptr_t newData, CompiledMethod *passed_nm /* = NULL */, bool copy_back_to_oop_pool) { address next_address; address loc = addr_at(0); if (MacroAssembler::is_load_const_from_toc(loc) ) { // Load constant from TOC. // Offset is +/- 2**32 -> use long. long offset = MacroAssembler::get_load_const_from_toc_offset(loc); address target = MacroAssembler::get_target_addr_pcrel(loc); intptr_t oldData = *(intptr_t*)target; if (oldData != newData) { // Update only if data changes. Prevents cache invalidation. *(intptr_t *)(target) = newData; } // ICache::invalidate_range(target, sizeof(unsigned long)); // No ICache invalidate for CP data. next_address = loc + MacroAssembler::load_const_from_toc_size(); } else if (MacroAssembler::is_call_far_pcrelative(loc)) { ((NativeMovConstReg*)loc)->dump(64, "NativeMovConstReg::set_pcrel_data() has a problem: setting data for a pc-relative call?"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #else assert(false, "Ooooops: setting data for a pc-relative call"); #endif next_address = next_instruction_address(); } else { assert(false, "Not a NativeMovConstReg site for set_pcrel_data"); next_address = next_instruction_address(); // Failure should be handled in next_instruction_address(). } if (copy_back_to_oop_pool) { if (relocInfo::update_oop_pool(instruction_address(), next_address, (address)newData, NULL)) { ((NativeMovConstReg*)instruction_address())->dump(64, "NativeMovConstReg::set_pcrel_data(): found oop reloc for pcrel_data"); #ifdef LUCY_DBG VM_Version::z_SIGSEGV(); #else assert(false, "Ooooops: found oop reloc for pcrel_data"); #endif } } } #ifdef COMPILER1 //-------------------------------- // N a t i v e M o v R e g M e m //-------------------------------- void NativeMovRegMem::verify() { address l1 = addr_at(0); address l2 = addr_at(MacroAssembler::load_const_size()); if (!MacroAssembler::is_load_const(l1)) { tty->cr(); tty->print_cr("NativeMovRegMem::verify(): verifying addr " PTR_FORMAT, p2i(l1)); tty->cr(); ((NativeMovRegMem*)l1)->dump(64, "NativeMovConstReg::verify()"); fatal("this is not a `NativeMovRegMem' site"); } unsigned long inst1; Assembler::get_instruction(l2, &inst1); if (!Assembler::is_z_lb(inst1) && !Assembler::is_z_llgh(inst1) && !Assembler::is_z_lh(inst1) && !Assembler::is_z_l(inst1) && !Assembler::is_z_llgf(inst1) && !Assembler::is_z_lg(inst1) && !Assembler::is_z_le(inst1) && !Assembler::is_z_ld(inst1) && !Assembler::is_z_stc(inst1) && !Assembler::is_z_sth(inst1) && !Assembler::is_z_st(inst1) && !UseCompressedOops && !Assembler::is_z_stg(inst1) && !Assembler::is_z_ste(inst1) && !Assembler::is_z_std(inst1)) { tty->cr(); tty->print_cr("NativeMovRegMem::verify(): verifying addr " PTR_FORMAT ": wrong or missing load or store at " PTR_FORMAT, p2i(l1), p2i(l2)); tty->cr(); ((NativeMovRegMem*)l1)->dump(64, "NativeMovConstReg::verify()"); fatal("this is not a `NativeMovRegMem' site"); } } #endif // COMPILER1 //----------------------- // N a t i v e J u m p //----------------------- void NativeJump::verify() { if (NativeJump::is_jump_at(addr_at(0))) return; fatal("this is not a `NativeJump' site"); } // Patch atomically with an illtrap. void NativeJump::patch_verified_entry(address entry, address verified_entry, address dest) { ResourceMark rm; int code_size = 2; CodeBuffer cb(verified_entry, code_size + 1); MacroAssembler* a = new MacroAssembler(&cb); #ifdef COMPILER2 assert(dest == SharedRuntime::get_handle_wrong_method_stub(), "expected fixed destination of patch"); #endif a->z_illtrap(); ICache::invalidate_range(verified_entry, code_size); } #undef LUCY_DBG //------------------------------------- // N a t i v e G e n e r a l J u m p //------------------------------------- #ifndef PRODUCT void NativeGeneralJump::verify() { unsigned long inst; Assembler::get_instruction((address)this, &inst); assert(MacroAssembler::is_branch_pcrelative_long(inst), "not a general jump instruction"); } #endif void NativeGeneralJump::insert_unconditional(address code_pos, address entry) { uint64_t instr = BRCL_ZOPC | Assembler::uimm4(Assembler::bcondAlways, 8, 48) | Assembler::simm32(RelAddr::pcrel_off32(entry, code_pos), 16, 48); *(uint64_t*) code_pos = (instr << 16); // Must shift into big end, then the brcl will be written to code_pos. ICache::invalidate_range(code_pos, instruction_size); } void NativeGeneralJump::replace_mt_safe(address instr_addr, address code_buffer) { assert(((intptr_t)instr_addr & (BytesPerWord-1)) == 0, "requirement for mt safe patching"); // Bytes_after_jump cannot change, because we own the Patching_lock. assert(Patching_lock->owned_by_self(), "must hold lock to patch instruction"); intptr_t bytes_after_jump = (*(intptr_t*)instr_addr) & 0x000000000000ffffL; // 2 bytes after jump. intptr_t load_const_bytes = (*(intptr_t*)code_buffer) & 0xffffffffffff0000L; *(intptr_t*)instr_addr = load_const_bytes | bytes_after_jump; ICache::invalidate_range(instr_addr, 6); }