src/hotspot/cpu/s390/sharedRuntime_s390.cpp
author chegar
Thu, 17 Oct 2019 20:54:25 +0100
branchdatagramsocketimpl-branch
changeset 58679 9c3209ff7550
parent 58678 9cf78a70fa4f
parent 58554 8c3c39710a08
permissions -rw-r--r--
datagramsocketimpl-branch: merge with default
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     1
/*
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 52460
diff changeset
     2
 * Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved.
58356
feff88c68082 8231448: s390 and ppc - replace JVM type comparisons to T_OBJECT and T_ARRAY with call to is_reference_type
mbaesken
parents: 57710
diff changeset
     3
 * Copyright (c) 2016, 2019, SAP SE. All rights reserved.
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     4
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     5
 *
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     6
 * This code is free software; you can redistribute it and/or modify it
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     7
 * under the terms of the GNU General Public License version 2 only, as
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     8
 * published by the Free Software Foundation.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
     9
 *
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    10
 * This code is distributed in the hope that it will be useful, but WITHOUT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    11
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    12
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    13
 * version 2 for more details (a copy is included in the LICENSE file that
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    14
 * accompanied this code).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    15
 *
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    16
 * You should have received a copy of the GNU General Public License version
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    17
 * 2 along with this work; if not, write to the Free Software Foundation,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    18
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    19
 *
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    20
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    21
 * or visit www.oracle.com if you need additional information or have any
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    22
 * questions.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    23
 *
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    24
 */
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    25
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    26
#include "precompiled.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    27
#include "asm/macroAssembler.inline.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    28
#include "code/debugInfoRec.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    29
#include "code/icBuffer.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    30
#include "code/vtableStubs.hpp"
49449
ef5d5d343e2a 8199263: Split interfaceSupport.hpp to not require including .inline.hpp files
coleenp
parents: 49368
diff changeset
    31
#include "gc/shared/gcLocker.hpp"
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    32
#include "interpreter/interpreter.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    33
#include "interpreter/interp_masm.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    34
#include "memory/resourceArea.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    35
#include "oops/compiledICHolder.hpp"
58554
8c3c39710a08 8232151: Minimal VM build broken after JDK-8232050
jiefu
parents: 58356
diff changeset
    36
#include "oops/klass.inline.hpp"
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    37
#include "registerSaver_s390.hpp"
49449
ef5d5d343e2a 8199263: Split interfaceSupport.hpp to not require including .inline.hpp files
coleenp
parents: 49368
diff changeset
    38
#include "runtime/safepointMechanism.hpp"
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    39
#include "runtime/sharedRuntime.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    40
#include "runtime/vframeArray.hpp"
46625
edefffab74e2 8183552: Move align functions to align.hpp
stefank
parents: 46620
diff changeset
    41
#include "utilities/align.hpp"
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    42
#include "vmreg_s390.inline.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    43
#ifdef COMPILER1
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    44
#include "c1/c1_Runtime1.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    45
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    46
#ifdef COMPILER2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    47
#include "opto/ad.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    48
#include "opto/runtime.hpp"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    49
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    50
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    51
#ifdef PRODUCT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    52
#define __ masm->
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    53
#else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    54
#define __ (Verbose ? (masm->block_comment(FILE_AND_LINE),masm):masm)->
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    55
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    56
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    57
#define BLOCK_COMMENT(str) __ block_comment(str)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    58
#define BIND(label)        bind(label); BLOCK_COMMENT(#label ":")
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    59
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    60
#define RegisterSaver_LiveIntReg(regname) \
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    61
  { RegisterSaver::int_reg,   regname->encoding(), regname->as_VMReg() }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    62
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    63
#define RegisterSaver_LiveFloatReg(regname) \
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    64
  { RegisterSaver::float_reg, regname->encoding(), regname->as_VMReg() }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    65
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    66
// Registers which are not saved/restored, but still they have got a frame slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    67
// Used to get same frame size for RegisterSaver_LiveRegs and RegisterSaver_LiveRegsWithoutR2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    68
#define RegisterSaver_ExcludedIntReg(regname) \
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    69
  { RegisterSaver::excluded_reg, regname->encoding(), regname->as_VMReg() }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    70
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    71
// Registers which are not saved/restored, but still they have got a frame slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    72
// Used to get same frame size for RegisterSaver_LiveRegs and RegisterSaver_LiveRegsWithoutR2.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    73
#define RegisterSaver_ExcludedFloatReg(regname) \
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    74
  { RegisterSaver::excluded_reg, regname->encoding(), regname->as_VMReg() }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    75
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    76
static const RegisterSaver::LiveRegType RegisterSaver_LiveRegs[] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    77
  // Live registers which get spilled to the stack. Register positions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    78
  // in this array correspond directly to the stack layout.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    79
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    80
  // live float registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    81
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    82
  RegisterSaver_LiveFloatReg(Z_F0 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    83
  // RegisterSaver_ExcludedFloatReg(Z_F1 ), // scratch (Z_fscratch_1)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    84
  RegisterSaver_LiveFloatReg(Z_F2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    85
  RegisterSaver_LiveFloatReg(Z_F3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    86
  RegisterSaver_LiveFloatReg(Z_F4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    87
  RegisterSaver_LiveFloatReg(Z_F5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    88
  RegisterSaver_LiveFloatReg(Z_F6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    89
  RegisterSaver_LiveFloatReg(Z_F7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    90
  RegisterSaver_LiveFloatReg(Z_F8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    91
  RegisterSaver_LiveFloatReg(Z_F9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    92
  RegisterSaver_LiveFloatReg(Z_F10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    93
  RegisterSaver_LiveFloatReg(Z_F11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    94
  RegisterSaver_LiveFloatReg(Z_F12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    95
  RegisterSaver_LiveFloatReg(Z_F13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    96
  RegisterSaver_LiveFloatReg(Z_F14),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    97
  RegisterSaver_LiveFloatReg(Z_F15),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    98
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
    99
  // RegisterSaver_ExcludedIntReg(Z_R0), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   100
  // RegisterSaver_ExcludedIntReg(Z_R1), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   101
  RegisterSaver_LiveIntReg(Z_R2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   102
  RegisterSaver_LiveIntReg(Z_R3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   103
  RegisterSaver_LiveIntReg(Z_R4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   104
  RegisterSaver_LiveIntReg(Z_R5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   105
  RegisterSaver_LiveIntReg(Z_R6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   106
  RegisterSaver_LiveIntReg(Z_R7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   107
  RegisterSaver_LiveIntReg(Z_R8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   108
  RegisterSaver_LiveIntReg(Z_R9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   109
  RegisterSaver_LiveIntReg(Z_R10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   110
  RegisterSaver_LiveIntReg(Z_R11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   111
  RegisterSaver_LiveIntReg(Z_R12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   112
  RegisterSaver_LiveIntReg(Z_R13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   113
  // RegisterSaver_ExcludedIntReg(Z_R14), // return pc (Saved in caller frame.)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   114
  // RegisterSaver_ExcludedIntReg(Z_R15)  // stack pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   115
};
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   116
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   117
static const RegisterSaver::LiveRegType RegisterSaver_LiveIntRegs[] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   118
  // Live registers which get spilled to the stack. Register positions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   119
  // in this array correspond directly to the stack layout.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   120
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   121
  // live float registers: All excluded, but still they get a stack slot to get same frame size.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   122
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   123
  RegisterSaver_ExcludedFloatReg(Z_F0 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   124
  // RegisterSaver_ExcludedFloatReg(Z_F1 ), // scratch (Z_fscratch_1)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   125
  RegisterSaver_ExcludedFloatReg(Z_F2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   126
  RegisterSaver_ExcludedFloatReg(Z_F3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   127
  RegisterSaver_ExcludedFloatReg(Z_F4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   128
  RegisterSaver_ExcludedFloatReg(Z_F5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   129
  RegisterSaver_ExcludedFloatReg(Z_F6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   130
  RegisterSaver_ExcludedFloatReg(Z_F7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   131
  RegisterSaver_ExcludedFloatReg(Z_F8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   132
  RegisterSaver_ExcludedFloatReg(Z_F9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   133
  RegisterSaver_ExcludedFloatReg(Z_F10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   134
  RegisterSaver_ExcludedFloatReg(Z_F11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   135
  RegisterSaver_ExcludedFloatReg(Z_F12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   136
  RegisterSaver_ExcludedFloatReg(Z_F13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   137
  RegisterSaver_ExcludedFloatReg(Z_F14),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   138
  RegisterSaver_ExcludedFloatReg(Z_F15),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   139
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   140
  // RegisterSaver_ExcludedIntReg(Z_R0), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   141
  // RegisterSaver_ExcludedIntReg(Z_R1), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   142
  RegisterSaver_LiveIntReg(Z_R2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   143
  RegisterSaver_LiveIntReg(Z_R3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   144
  RegisterSaver_LiveIntReg(Z_R4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   145
  RegisterSaver_LiveIntReg(Z_R5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   146
  RegisterSaver_LiveIntReg(Z_R6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   147
  RegisterSaver_LiveIntReg(Z_R7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   148
  RegisterSaver_LiveIntReg(Z_R8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   149
  RegisterSaver_LiveIntReg(Z_R9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   150
  RegisterSaver_LiveIntReg(Z_R10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   151
  RegisterSaver_LiveIntReg(Z_R11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   152
  RegisterSaver_LiveIntReg(Z_R12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   153
  RegisterSaver_LiveIntReg(Z_R13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   154
  // RegisterSaver_ExcludedIntReg(Z_R14), // return pc (Saved in caller frame.)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   155
  // RegisterSaver_ExcludedIntReg(Z_R15)  // stack pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   156
};
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   157
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   158
static const RegisterSaver::LiveRegType RegisterSaver_LiveRegsWithoutR2[] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   159
  // Live registers which get spilled to the stack. Register positions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   160
  // in this array correspond directly to the stack layout.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   161
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   162
  // live float registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   163
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   164
  RegisterSaver_LiveFloatReg(Z_F0 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   165
  // RegisterSaver_ExcludedFloatReg(Z_F1 ), // scratch (Z_fscratch_1)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   166
  RegisterSaver_LiveFloatReg(Z_F2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   167
  RegisterSaver_LiveFloatReg(Z_F3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   168
  RegisterSaver_LiveFloatReg(Z_F4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   169
  RegisterSaver_LiveFloatReg(Z_F5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   170
  RegisterSaver_LiveFloatReg(Z_F6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   171
  RegisterSaver_LiveFloatReg(Z_F7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   172
  RegisterSaver_LiveFloatReg(Z_F8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   173
  RegisterSaver_LiveFloatReg(Z_F9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   174
  RegisterSaver_LiveFloatReg(Z_F10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   175
  RegisterSaver_LiveFloatReg(Z_F11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   176
  RegisterSaver_LiveFloatReg(Z_F12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   177
  RegisterSaver_LiveFloatReg(Z_F13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   178
  RegisterSaver_LiveFloatReg(Z_F14),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   179
  RegisterSaver_LiveFloatReg(Z_F15),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   180
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   181
  // RegisterSaver_ExcludedIntReg(Z_R0), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   182
  // RegisterSaver_ExcludedIntReg(Z_R1), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   183
  RegisterSaver_ExcludedIntReg(Z_R2), // Omit saving R2.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   184
  RegisterSaver_LiveIntReg(Z_R3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   185
  RegisterSaver_LiveIntReg(Z_R4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   186
  RegisterSaver_LiveIntReg(Z_R5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   187
  RegisterSaver_LiveIntReg(Z_R6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   188
  RegisterSaver_LiveIntReg(Z_R7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   189
  RegisterSaver_LiveIntReg(Z_R8 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   190
  RegisterSaver_LiveIntReg(Z_R9 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   191
  RegisterSaver_LiveIntReg(Z_R10),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   192
  RegisterSaver_LiveIntReg(Z_R11),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   193
  RegisterSaver_LiveIntReg(Z_R12),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   194
  RegisterSaver_LiveIntReg(Z_R13),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   195
  // RegisterSaver_ExcludedIntReg(Z_R14), // return pc (Saved in caller frame.)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   196
  // RegisterSaver_ExcludedIntReg(Z_R15)  // stack pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   197
};
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   198
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   199
// Live argument registers which get spilled to the stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   200
static const RegisterSaver::LiveRegType RegisterSaver_LiveArgRegs[] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   201
  RegisterSaver_LiveFloatReg(Z_FARG1),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   202
  RegisterSaver_LiveFloatReg(Z_FARG2),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   203
  RegisterSaver_LiveFloatReg(Z_FARG3),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   204
  RegisterSaver_LiveFloatReg(Z_FARG4),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   205
  RegisterSaver_LiveIntReg(Z_ARG1),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   206
  RegisterSaver_LiveIntReg(Z_ARG2),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   207
  RegisterSaver_LiveIntReg(Z_ARG3),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   208
  RegisterSaver_LiveIntReg(Z_ARG4),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   209
  RegisterSaver_LiveIntReg(Z_ARG5)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   210
};
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   211
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   212
static const RegisterSaver::LiveRegType RegisterSaver_LiveVolatileRegs[] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   213
  // Live registers which get spilled to the stack. Register positions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   214
  // in this array correspond directly to the stack layout.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   215
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   216
  // live float registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   217
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   218
  RegisterSaver_LiveFloatReg(Z_F0 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   219
  // RegisterSaver_ExcludedFloatReg(Z_F1 ), // scratch (Z_fscratch_1)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   220
  RegisterSaver_LiveFloatReg(Z_F2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   221
  RegisterSaver_LiveFloatReg(Z_F3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   222
  RegisterSaver_LiveFloatReg(Z_F4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   223
  RegisterSaver_LiveFloatReg(Z_F5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   224
  RegisterSaver_LiveFloatReg(Z_F6 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   225
  RegisterSaver_LiveFloatReg(Z_F7 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   226
  // RegisterSaver_LiveFloatReg(Z_F8 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   227
  // RegisterSaver_LiveFloatReg(Z_F9 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   228
  // RegisterSaver_LiveFloatReg(Z_F10), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   229
  // RegisterSaver_LiveFloatReg(Z_F11), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   230
  // RegisterSaver_LiveFloatReg(Z_F12), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   231
  // RegisterSaver_LiveFloatReg(Z_F13), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   232
  // RegisterSaver_LiveFloatReg(Z_F14), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   233
  // RegisterSaver_LiveFloatReg(Z_F15), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   234
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   235
  // RegisterSaver_ExcludedIntReg(Z_R0), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   236
  // RegisterSaver_ExcludedIntReg(Z_R1), // scratch
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   237
  RegisterSaver_LiveIntReg(Z_R2 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   238
  RegisterSaver_LiveIntReg(Z_R3 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   239
  RegisterSaver_LiveIntReg(Z_R4 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   240
  RegisterSaver_LiveIntReg(Z_R5 ),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   241
  // RegisterSaver_LiveIntReg(Z_R6 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   242
  // RegisterSaver_LiveIntReg(Z_R7 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   243
  // RegisterSaver_LiveIntReg(Z_R8 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   244
  // RegisterSaver_LiveIntReg(Z_R9 ), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   245
  // RegisterSaver_LiveIntReg(Z_R10), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   246
  // RegisterSaver_LiveIntReg(Z_R11), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   247
  // RegisterSaver_LiveIntReg(Z_R12), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   248
  // RegisterSaver_LiveIntReg(Z_R13), // non-volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   249
  // RegisterSaver_ExcludedIntReg(Z_R14), // return pc (Saved in caller frame.)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   250
  // RegisterSaver_ExcludedIntReg(Z_R15)  // stack pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   251
};
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   252
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   253
int RegisterSaver::live_reg_save_size(RegisterSet reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   254
  int reg_space = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   255
  switch (reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   256
    case all_registers:           reg_space = sizeof(RegisterSaver_LiveRegs); break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   257
    case all_registers_except_r2: reg_space = sizeof(RegisterSaver_LiveRegsWithoutR2); break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   258
    case all_integer_registers:   reg_space = sizeof(RegisterSaver_LiveIntRegs); break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   259
    case all_volatile_registers:  reg_space = sizeof(RegisterSaver_LiveVolatileRegs); break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   260
    case arg_registers:           reg_space = sizeof(RegisterSaver_LiveArgRegs); break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   261
    default: ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   262
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   263
  return (reg_space / sizeof(RegisterSaver::LiveRegType)) * reg_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   264
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   265
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   266
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   267
int RegisterSaver::live_reg_frame_size(RegisterSet reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   268
  return live_reg_save_size(reg_set) + frame::z_abi_160_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   269
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   270
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   271
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   272
// return_pc: Specify the register that should be stored as the return pc in the current frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   273
OopMap* RegisterSaver::save_live_registers(MacroAssembler* masm, RegisterSet reg_set, Register return_pc) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   274
  // Record volatile registers as callee-save values in an OopMap so
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   275
  // their save locations will be propagated to the caller frame's
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   276
  // RegisterMap during StackFrameStream construction (needed for
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   277
  // deoptimization; see compiledVFrame::create_stack_value).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   278
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   279
  // Calculate frame size.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   280
  const int frame_size_in_bytes  = live_reg_frame_size(reg_set);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   281
  const int frame_size_in_slots  = frame_size_in_bytes / sizeof(jint);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   282
  const int register_save_offset = frame_size_in_bytes - live_reg_save_size(reg_set);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   283
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   284
  // OopMap frame size is in c2 stack slots (sizeof(jint)) not bytes or words.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   285
  OopMap* map = new OopMap(frame_size_in_slots, 0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   286
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   287
  int regstosave_num = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   288
  const RegisterSaver::LiveRegType* live_regs = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   289
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   290
  switch (reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   291
    case all_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   292
      regstosave_num = sizeof(RegisterSaver_LiveRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   293
      live_regs      = RegisterSaver_LiveRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   294
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   295
    case all_registers_except_r2:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   296
      regstosave_num = sizeof(RegisterSaver_LiveRegsWithoutR2)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   297
      live_regs      = RegisterSaver_LiveRegsWithoutR2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   298
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   299
    case all_integer_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   300
      regstosave_num = sizeof(RegisterSaver_LiveIntRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   301
      live_regs      = RegisterSaver_LiveIntRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   302
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   303
    case all_volatile_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   304
      regstosave_num = sizeof(RegisterSaver_LiveVolatileRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   305
      live_regs      = RegisterSaver_LiveVolatileRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   306
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   307
    case arg_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   308
      regstosave_num = sizeof(RegisterSaver_LiveArgRegs)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   309
      live_regs      = RegisterSaver_LiveArgRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   310
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   311
    default: ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   312
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   313
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   314
  // Save return pc in old frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   315
  __ save_return_pc(return_pc);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   316
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   317
  // Push a new frame (includes stack linkage).
50094
2f79462aab9b 8201593: Print array length in ArrayIndexOutOfBoundsException.
goetz
parents: 49449
diff changeset
   318
  // Use return_pc as scratch for push_frame. Z_R0_scratch (the default) and Z_R1_scratch are
2f79462aab9b 8201593: Print array length in ArrayIndexOutOfBoundsException.
goetz
parents: 49449
diff changeset
   319
  // illegally used to pass parameters by RangeCheckStub::emit_code().
46726
7801367e3cc9 8180659: [s390] micro-optimization in resize_frame_absolute()
lucy
parents: 46721
diff changeset
   320
  __ push_frame(frame_size_in_bytes, return_pc);
7801367e3cc9 8180659: [s390] micro-optimization in resize_frame_absolute()
lucy
parents: 46721
diff changeset
   321
  // We have to restore return_pc right away.
7801367e3cc9 8180659: [s390] micro-optimization in resize_frame_absolute()
lucy
parents: 46721
diff changeset
   322
  // Nobody else will. Furthermore, return_pc isn't necessarily the default (Z_R14).
7801367e3cc9 8180659: [s390] micro-optimization in resize_frame_absolute()
lucy
parents: 46721
diff changeset
   323
  // Nobody else knows which register we saved.
7801367e3cc9 8180659: [s390] micro-optimization in resize_frame_absolute()
lucy
parents: 46721
diff changeset
   324
  __ z_lg(return_pc, _z_abi16(return_pc) + frame_size_in_bytes, Z_SP);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   325
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   326
  // Register save area in new frame starts above z_abi_160 area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   327
  int offset = register_save_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   328
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   329
  Register first = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   330
  Register last  = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   331
  int      first_offset = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   332
  bool     float_spilled = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   333
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   334
  for (int i = 0; i < regstosave_num; i++, offset += reg_size) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   335
    int reg_num  = live_regs[i].reg_num;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   336
    int reg_type = live_regs[i].reg_type;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   337
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   338
    switch (reg_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   339
      case RegisterSaver::int_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   340
        Register reg = as_Register(reg_num);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   341
        if (last != reg->predecessor()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   342
          if (first != noreg) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   343
            __ z_stmg(first, last, first_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   344
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   345
          first = reg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   346
          first_offset = offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   347
          DEBUG_ONLY(float_spilled = false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   348
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   349
        last = reg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   350
        assert(last != Z_R0, "r0 would require special treatment");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   351
        assert(!float_spilled, "for simplicity, do not mix up ints and floats in RegisterSaver_LiveRegs[]");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   352
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   353
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   354
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   355
      case RegisterSaver::excluded_reg: // Not saved/restored, but with dedicated slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   356
        continue; // Continue with next loop iteration.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   357
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   358
      case RegisterSaver::float_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   359
        FloatRegister freg = as_FloatRegister(reg_num);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   360
        __ z_std(freg, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   361
        DEBUG_ONLY(float_spilled = true);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   362
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   363
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   364
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   365
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   366
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   367
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   368
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   369
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   370
    // Second set_callee_saved is really a waste but we'll keep things as they were for now
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   371
    map->set_callee_saved(VMRegImpl::stack2reg(offset >> 2), live_regs[i].vmreg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   372
    map->set_callee_saved(VMRegImpl::stack2reg((offset + half_reg_size) >> 2), live_regs[i].vmreg->next());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   373
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   374
  assert(first != noreg, "Should spill at least one int reg.");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   375
  __ z_stmg(first, last, first_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   376
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   377
  // And we're done.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   378
  return map;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   379
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   380
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   381
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   382
// Generate the OopMap (again, regs where saved before).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   383
OopMap* RegisterSaver::generate_oop_map(MacroAssembler* masm, RegisterSet reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   384
  // Calculate frame size.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   385
  const int frame_size_in_bytes  = live_reg_frame_size(reg_set);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   386
  const int frame_size_in_slots  = frame_size_in_bytes / sizeof(jint);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   387
  const int register_save_offset = frame_size_in_bytes - live_reg_save_size(reg_set);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   388
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   389
  // OopMap frame size is in c2 stack slots (sizeof(jint)) not bytes or words.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   390
  OopMap* map = new OopMap(frame_size_in_slots, 0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   391
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   392
  int regstosave_num = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   393
  const RegisterSaver::LiveRegType* live_regs = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   394
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   395
  switch (reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   396
    case all_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   397
      regstosave_num = sizeof(RegisterSaver_LiveRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   398
      live_regs      = RegisterSaver_LiveRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   399
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   400
    case all_registers_except_r2:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   401
      regstosave_num = sizeof(RegisterSaver_LiveRegsWithoutR2)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   402
      live_regs      = RegisterSaver_LiveRegsWithoutR2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   403
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   404
    case all_integer_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   405
      regstosave_num = sizeof(RegisterSaver_LiveIntRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   406
      live_regs      = RegisterSaver_LiveIntRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   407
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   408
    case all_volatile_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   409
      regstosave_num = sizeof(RegisterSaver_LiveVolatileRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   410
      live_regs      = RegisterSaver_LiveVolatileRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   411
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   412
    case arg_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   413
      regstosave_num = sizeof(RegisterSaver_LiveArgRegs)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   414
      live_regs      = RegisterSaver_LiveArgRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   415
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   416
    default: ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   417
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   418
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   419
  // Register save area in new frame starts above z_abi_160 area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   420
  int offset = register_save_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   421
  for (int i = 0; i < regstosave_num; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   422
    if (live_regs[i].reg_type < RegisterSaver::excluded_reg) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   423
      map->set_callee_saved(VMRegImpl::stack2reg(offset>>2), live_regs[i].vmreg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   424
      map->set_callee_saved(VMRegImpl::stack2reg((offset + half_reg_size)>>2), live_regs[i].vmreg->next());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   425
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   426
    offset += reg_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   427
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   428
  return map;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   429
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   430
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   431
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   432
// Pop the current frame and restore all the registers that we saved.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   433
void RegisterSaver::restore_live_registers(MacroAssembler* masm, RegisterSet reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   434
  int offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   435
  const int register_save_offset = live_reg_frame_size(reg_set) - live_reg_save_size(reg_set);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   436
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   437
  Register first = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   438
  Register last = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   439
  int      first_offset = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   440
  bool     float_spilled = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   441
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   442
  int regstosave_num = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   443
  const RegisterSaver::LiveRegType* live_regs = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   444
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   445
  switch (reg_set) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   446
    case all_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   447
      regstosave_num = sizeof(RegisterSaver_LiveRegs)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   448
      live_regs      = RegisterSaver_LiveRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   449
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   450
    case all_registers_except_r2:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   451
      regstosave_num = sizeof(RegisterSaver_LiveRegsWithoutR2)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   452
      live_regs      = RegisterSaver_LiveRegsWithoutR2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   453
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   454
    case all_integer_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   455
      regstosave_num = sizeof(RegisterSaver_LiveIntRegs)/sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   456
      live_regs      = RegisterSaver_LiveIntRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   457
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   458
    case all_volatile_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   459
      regstosave_num = sizeof(RegisterSaver_LiveVolatileRegs)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   460
      live_regs      = RegisterSaver_LiveVolatileRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   461
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   462
    case arg_registers:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   463
      regstosave_num = sizeof(RegisterSaver_LiveArgRegs)/sizeof(RegisterSaver::LiveRegType);;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   464
      live_regs      = RegisterSaver_LiveArgRegs;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   465
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   466
    default: ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   467
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   468
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   469
  // Restore all registers (ints and floats).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   470
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   471
  // Register save area in new frame starts above z_abi_160 area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   472
  offset = register_save_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   473
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   474
  for (int i = 0; i < regstosave_num; i++, offset += reg_size) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   475
    int reg_num  = live_regs[i].reg_num;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   476
    int reg_type = live_regs[i].reg_type;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   477
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   478
    switch (reg_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   479
      case RegisterSaver::excluded_reg:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   480
        continue; // Continue with next loop iteration.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   481
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   482
      case RegisterSaver::int_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   483
        Register reg = as_Register(reg_num);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   484
        if (last != reg->predecessor()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   485
          if (first != noreg) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   486
            __ z_lmg(first, last, first_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   487
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   488
          first = reg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   489
          first_offset = offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   490
          DEBUG_ONLY(float_spilled = false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   491
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   492
        last = reg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   493
        assert(last != Z_R0, "r0 would require special treatment");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   494
        assert(!float_spilled, "for simplicity, do not mix up ints and floats in RegisterSaver_LiveRegs[]");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   495
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   496
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   497
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   498
      case RegisterSaver::float_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   499
        FloatRegister freg = as_FloatRegister(reg_num);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   500
        __ z_ld(freg, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   501
        DEBUG_ONLY(float_spilled = true);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   502
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   503
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   504
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   505
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   506
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   507
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   508
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   509
  assert(first != noreg, "Should spill at least one int reg.");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   510
  __ z_lmg(first, last, first_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   511
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   512
  // Pop the frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   513
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   514
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   515
  // Restore the flags.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   516
  __ restore_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   517
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   518
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   519
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   520
// Pop the current frame and restore the registers that might be holding a result.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   521
void RegisterSaver::restore_result_registers(MacroAssembler* masm) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   522
  int i;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   523
  int offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   524
  const int regstosave_num       = sizeof(RegisterSaver_LiveRegs) /
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   525
                                   sizeof(RegisterSaver::LiveRegType);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   526
  const int register_save_offset = live_reg_frame_size(all_registers) - live_reg_save_size(all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   527
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   528
  // Restore all result registers (ints and floats).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   529
  offset = register_save_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   530
  for (int i = 0; i < regstosave_num; i++, offset += reg_size) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   531
    int reg_num = RegisterSaver_LiveRegs[i].reg_num;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   532
    int reg_type = RegisterSaver_LiveRegs[i].reg_type;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   533
    switch (reg_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   534
      case RegisterSaver::excluded_reg:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   535
        continue; // Continue with next loop iteration.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   536
      case RegisterSaver::int_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   537
        if (as_Register(reg_num) == Z_RET) { // int result_reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   538
          __ z_lg(as_Register(reg_num), offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   539
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   540
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   541
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   542
      case RegisterSaver::float_reg: {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   543
        if (as_FloatRegister(reg_num) == Z_FRET) { // float result_reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   544
          __ z_ld(as_FloatRegister(reg_num), offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   545
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   546
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   547
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   548
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   549
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   550
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   551
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   552
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   553
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   554
size_t SharedRuntime::trampoline_size() {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   555
  return MacroAssembler::load_const_size() + 2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   556
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   557
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   558
void SharedRuntime::generate_trampoline(MacroAssembler *masm, address destination) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   559
  // Think about using pc-relative branch.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   560
  __ load_const(Z_R1_scratch, destination);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   561
  __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   562
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   563
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   564
// ---------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   565
void SharedRuntime::save_native_result(MacroAssembler * masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   566
                                       BasicType ret_type,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   567
                                       int frame_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   568
  Address memaddr(Z_SP, frame_slots * VMRegImpl::stack_slot_size);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   569
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   570
  switch (ret_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   571
    case T_BOOLEAN:  // Save shorter types as int. Do we need sign extension at restore??
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   572
    case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   573
    case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   574
    case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   575
    case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   576
      __ reg2mem_opt(Z_RET, memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   577
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   578
    case T_OBJECT:   // Save pointer types as long.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   579
    case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   580
    case T_ADDRESS:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   581
    case T_VOID:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   582
    case T_LONG:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   583
      __ reg2mem_opt(Z_RET, memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   584
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   585
    case T_FLOAT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   586
      __ freg2mem_opt(Z_FRET, memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   587
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   588
    case T_DOUBLE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   589
      __ freg2mem_opt(Z_FRET, memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   590
      break;
51966
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   591
    default:
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   592
      ShouldNotReachHere();
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   593
      break;
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   594
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   595
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   596
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   597
void SharedRuntime::restore_native_result(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   598
                                          BasicType       ret_type,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   599
                                          int             frame_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   600
  Address memaddr(Z_SP, frame_slots * VMRegImpl::stack_slot_size);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   601
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   602
  switch (ret_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   603
    case T_BOOLEAN:  // Restore shorter types as int. Do we need sign extension at restore??
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   604
    case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   605
    case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   606
    case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   607
    case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   608
      __ mem2reg_opt(Z_RET, memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   609
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   610
    case T_OBJECT:   // Restore pointer types as long.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   611
    case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   612
    case T_ADDRESS:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   613
    case T_VOID:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   614
    case T_LONG:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   615
      __ mem2reg_opt(Z_RET, memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   616
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   617
    case T_FLOAT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   618
      __ mem2freg_opt(Z_FRET, memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   619
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   620
    case T_DOUBLE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   621
      __ mem2freg_opt(Z_FRET, memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   622
      break;
51966
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   623
    default:
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   624
      ShouldNotReachHere();
2a2a78bd1fc3 8211145: [ppc] [s390]: Build fails due to -Werror=switch (introduced with JDK-8211029)
lucy
parents: 50094
diff changeset
   625
      break;
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   626
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   627
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   628
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   629
// ---------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   630
// Read the array of BasicTypes from a signature, and compute where the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   631
// arguments should go. Values in the VMRegPair regs array refer to 4-byte
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   632
// quantities. Values less than VMRegImpl::stack0 are registers, those above
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   633
// refer to 4-byte stack slots. All stack slots are based off of the stack pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   634
// as framesizes are fixed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   635
// VMRegImpl::stack0 refers to the first slot 0(sp).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   636
// VMRegImpl::stack0+1 refers to the memory word 4-byes higher. Registers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   637
// up to RegisterImpl::number_of_registers are the 64-bit integer registers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   638
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   639
// Note: the INPUTS in sig_bt are in units of Java argument words, which are
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   640
// either 32-bit or 64-bit depending on the build. The OUTPUTS are in 32-bit
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   641
// units regardless of build.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   642
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   643
// The Java calling convention is a "shifted" version of the C ABI.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   644
// By skipping the first C ABI register we can call non-static jni methods
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   645
// with small numbers of arguments without having to shuffle the arguments
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   646
// at all. Since we control the java ABI we ought to at least get some
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   647
// advantage out of it.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   648
int SharedRuntime::java_calling_convention(const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   649
                                           VMRegPair *regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   650
                                           int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   651
                                           int is_outgoing) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   652
  // c2c calling conventions for compiled-compiled calls.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   653
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   654
  // An int/float occupies 1 slot here.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   655
  const int inc_stk_for_intfloat   = 1; // 1 slots for ints and floats.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   656
  const int inc_stk_for_longdouble = 2; // 2 slots for longs and doubles.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   657
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   658
  const VMReg z_iarg_reg[5] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   659
    Z_R2->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   660
    Z_R3->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   661
    Z_R4->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   662
    Z_R5->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   663
    Z_R6->as_VMReg()
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   664
  };
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   665
  const VMReg z_farg_reg[4] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   666
    Z_F0->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   667
    Z_F2->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   668
    Z_F4->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   669
    Z_F6->as_VMReg()
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   670
  };
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   671
  const int z_num_iarg_registers = sizeof(z_iarg_reg) / sizeof(z_iarg_reg[0]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   672
  const int z_num_farg_registers = sizeof(z_farg_reg) / sizeof(z_farg_reg[0]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   673
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   674
  assert(RegisterImpl::number_of_arg_registers == z_num_iarg_registers, "iarg reg count mismatch");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   675
  assert(FloatRegisterImpl::number_of_arg_registers == z_num_farg_registers, "farg reg count mismatch");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   676
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   677
  int i;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   678
  int stk = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   679
  int ireg = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   680
  int freg = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   681
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   682
  for (int i = 0; i < total_args_passed; ++i) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   683
    switch (sig_bt[i]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   684
      case T_BOOLEAN:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   685
      case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   686
      case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   687
      case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   688
      case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   689
        if (ireg < z_num_iarg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   690
          // Put int/ptr in register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   691
          regs[i].set1(z_iarg_reg[ireg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   692
          ++ireg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   693
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   694
          // Put int/ptr on stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   695
          regs[i].set1(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   696
          stk += inc_stk_for_intfloat;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   697
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   698
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   699
      case T_LONG:
42874
973960866fa4 8170761: Buffer overrun in sharedRuntime_x86_64.cpp:477
jcm
parents: 42065
diff changeset
   700
        assert((i + 1) < total_args_passed && sig_bt[i+1] == T_VOID, "expecting half");
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   701
        if (ireg < z_num_iarg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   702
          // Put long in register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   703
          regs[i].set2(z_iarg_reg[ireg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   704
          ++ireg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   705
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   706
          // Put long on stack and align to 2 slots.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   707
          if (stk & 0x1) { ++stk; }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   708
          regs[i].set2(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   709
          stk += inc_stk_for_longdouble;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   710
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   711
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   712
      case T_OBJECT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   713
      case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   714
      case T_ADDRESS:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   715
        if (ireg < z_num_iarg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   716
          // Put ptr in register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   717
          regs[i].set2(z_iarg_reg[ireg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   718
          ++ireg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   719
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   720
          // Put ptr on stack and align to 2 slots, because
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   721
          // "64-bit pointers record oop-ishness on 2 aligned adjacent
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   722
          // registers." (see OopFlow::build_oop_map).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   723
          if (stk & 0x1) { ++stk; }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   724
          regs[i].set2(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   725
          stk += inc_stk_for_longdouble;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   726
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   727
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   728
      case T_FLOAT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   729
        if (freg < z_num_farg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   730
          // Put float in register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   731
          regs[i].set1(z_farg_reg[freg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   732
          ++freg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   733
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   734
          // Put float on stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   735
          regs[i].set1(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   736
          stk += inc_stk_for_intfloat;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   737
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   738
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   739
      case T_DOUBLE:
42874
973960866fa4 8170761: Buffer overrun in sharedRuntime_x86_64.cpp:477
jcm
parents: 42065
diff changeset
   740
        assert((i + 1) < total_args_passed && sig_bt[i+1] == T_VOID, "expecting half");
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   741
        if (freg < z_num_farg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   742
          // Put double in register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   743
          regs[i].set2(z_farg_reg[freg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   744
          ++freg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   745
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   746
          // Put double on stack and align to 2 slots.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   747
          if (stk & 0x1) { ++stk; }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   748
          regs[i].set2(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   749
          stk += inc_stk_for_longdouble;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   750
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   751
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   752
      case T_VOID:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   753
        assert(i != 0 && (sig_bt[i - 1] == T_LONG || sig_bt[i - 1] == T_DOUBLE), "expecting half");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   754
        // Do not count halves.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   755
        regs[i].set_bad();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   756
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   757
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   758
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   759
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   760
  }
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
   761
  return align_up(stk, 2);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   762
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   763
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   764
int SharedRuntime::c_calling_convention(const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   765
                                        VMRegPair *regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   766
                                        VMRegPair *regs2,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   767
                                        int total_args_passed) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   768
  assert(regs2 == NULL, "second VMRegPair array not used on this platform");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   769
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   770
  // Calling conventions for C runtime calls and calls to JNI native methods.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   771
  const VMReg z_iarg_reg[5] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   772
    Z_R2->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   773
    Z_R3->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   774
    Z_R4->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   775
    Z_R5->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   776
    Z_R6->as_VMReg()
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   777
  };
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   778
  const VMReg z_farg_reg[4] = {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   779
    Z_F0->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   780
    Z_F2->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   781
    Z_F4->as_VMReg(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   782
    Z_F6->as_VMReg()
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   783
  };
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   784
  const int z_num_iarg_registers = sizeof(z_iarg_reg) / sizeof(z_iarg_reg[0]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   785
  const int z_num_farg_registers = sizeof(z_farg_reg) / sizeof(z_farg_reg[0]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   786
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   787
  // Check calling conventions consistency.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   788
  assert(RegisterImpl::number_of_arg_registers == z_num_iarg_registers, "iarg reg count mismatch");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   789
  assert(FloatRegisterImpl::number_of_arg_registers == z_num_farg_registers, "farg reg count mismatch");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   790
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   791
  // Avoid passing C arguments in the wrong stack slots.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   792
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   793
  // 'Stk' counts stack slots. Due to alignment, 32 bit values occupy
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   794
  // 2 such slots, like 64 bit values do.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   795
  const int inc_stk_for_intfloat   = 2; // 2 slots for ints and floats.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   796
  const int inc_stk_for_longdouble = 2; // 2 slots for longs and doubles.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   797
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   798
  int i;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   799
  // Leave room for C-compatible ABI
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   800
  int stk = (frame::z_abi_160_size - frame::z_jit_out_preserve_size) / VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   801
  int freg = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   802
  int ireg = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   803
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   804
  // We put the first 5 arguments into registers and the rest on the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   805
  // stack. Float arguments are already in their argument registers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   806
  // due to c2c calling conventions (see calling_convention).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   807
  for (int i = 0; i < total_args_passed; ++i) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   808
    switch (sig_bt[i]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   809
      case T_BOOLEAN:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   810
      case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   811
      case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   812
      case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   813
      case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   814
        // Fall through, handle as long.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   815
      case T_LONG:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   816
      case T_OBJECT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   817
      case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   818
      case T_ADDRESS:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   819
      case T_METADATA:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   820
        // Oops are already boxed if required (JNI).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   821
        if (ireg < z_num_iarg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   822
          regs[i].set2(z_iarg_reg[ireg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   823
          ++ireg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   824
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   825
          regs[i].set2(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   826
          stk += inc_stk_for_longdouble;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   827
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   828
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   829
      case T_FLOAT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   830
        if (freg < z_num_farg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   831
          regs[i].set1(z_farg_reg[freg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   832
          ++freg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   833
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   834
          regs[i].set1(VMRegImpl::stack2reg(stk+1));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   835
          stk +=  inc_stk_for_intfloat;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   836
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   837
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   838
      case T_DOUBLE:
42874
973960866fa4 8170761: Buffer overrun in sharedRuntime_x86_64.cpp:477
jcm
parents: 42065
diff changeset
   839
        assert((i + 1) < total_args_passed && sig_bt[i+1] == T_VOID, "expecting half");
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   840
        if (freg < z_num_farg_registers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   841
          regs[i].set2(z_farg_reg[freg]);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   842
          ++freg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   843
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   844
          // Put double on stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   845
          regs[i].set2(VMRegImpl::stack2reg(stk));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   846
          stk += inc_stk_for_longdouble;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   847
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   848
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   849
      case T_VOID:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   850
        // Do not count halves.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   851
        regs[i].set_bad();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   852
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   853
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   854
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   855
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   856
  }
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
   857
  return align_up(stk, 2);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   858
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   859
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   860
////////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   861
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   862
//  Argument shufflers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   863
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   864
////////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   865
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   866
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   867
// The java_calling_convention describes stack locations as ideal slots on
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   868
// a frame with no abi restrictions. Since we must observe abi restrictions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   869
// (like the placement of the register window) the slots must be biased by
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   870
// the following value.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   871
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   872
static int reg2slot(VMReg r) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   873
  return r->reg2stack() + SharedRuntime::out_preserve_stack_slots();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   874
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   875
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   876
static int reg2offset(VMReg r) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   877
  return reg2slot(r) * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   878
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   879
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   880
static void verify_oop_args(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   881
                            int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   882
                            const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   883
                            const VMRegPair *regs) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   884
  if (!VerifyOops) { return; }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   885
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   886
  for (int i = 0; i < total_args_passed; i++) {
58356
feff88c68082 8231448: s390 and ppc - replace JVM type comparisons to T_OBJECT and T_ARRAY with call to is_reference_type
mbaesken
parents: 57710
diff changeset
   887
    if (is_reference_type(sig_bt[i])) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   888
      VMReg r = regs[i].first();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   889
      assert(r->is_valid(), "bad oop arg");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   890
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   891
      if (r->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   892
        __ z_lg(Z_R0_scratch,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   893
                Address(Z_SP, r->reg2stack() * VMRegImpl::stack_slot_size + wordSize));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   894
        __ verify_oop(Z_R0_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   895
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   896
        __ verify_oop(r->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   897
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   898
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   899
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   900
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   901
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   902
static void gen_special_dispatch(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   903
                                 int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   904
                                 vmIntrinsics::ID special_dispatch,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   905
                                 const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   906
                                 const VMRegPair *regs) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   907
  verify_oop_args(masm, total_args_passed, sig_bt, regs);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   908
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   909
  // Now write the args into the outgoing interpreter space.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   910
  bool     has_receiver   = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   911
  Register receiver_reg   = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   912
  int      member_arg_pos = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   913
  Register member_reg     = noreg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   914
  int      ref_kind       = MethodHandles::signature_polymorphic_intrinsic_ref_kind(special_dispatch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   915
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   916
  if (ref_kind != 0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   917
    member_arg_pos = total_args_passed - 1;  // trailing MemberName argument
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   918
    member_reg = Z_R9;                       // Known to be free at this point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   919
    has_receiver = MethodHandles::ref_kind_has_receiver(ref_kind);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   920
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   921
    guarantee(special_dispatch == vmIntrinsics::_invokeBasic, "special_dispatch=%d", special_dispatch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   922
    has_receiver = true;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   923
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   924
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   925
  if (member_reg != noreg) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   926
    // Load the member_arg into register, if necessary.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   927
    assert(member_arg_pos >= 0 && member_arg_pos < total_args_passed, "oob");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   928
    assert(sig_bt[member_arg_pos] == T_OBJECT, "dispatch argument must be an object");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   929
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   930
    VMReg r = regs[member_arg_pos].first();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   931
    assert(r->is_valid(), "bad member arg");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   932
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   933
    if (r->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   934
      __ z_lg(member_reg, Address(Z_SP, reg2offset(r)));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   935
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   936
      // No data motion is needed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   937
      member_reg = r->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   938
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   939
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   940
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   941
  if (has_receiver) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   942
    // Make sure the receiver is loaded into a register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   943
    assert(total_args_passed > 0, "oob");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   944
    assert(sig_bt[0] == T_OBJECT, "receiver argument must be an object");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   945
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   946
    VMReg r = regs[0].first();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   947
    assert(r->is_valid(), "bad receiver arg");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   948
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   949
    if (r->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   950
      // Porting note: This assumes that compiled calling conventions always
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   951
      // pass the receiver oop in a register. If this is not true on some
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   952
      // platform, pick a temp and load the receiver from stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   953
      assert(false, "receiver always in a register");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   954
      receiver_reg = Z_R13;  // Known to be free at this point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   955
      __ z_lg(receiver_reg, Address(Z_SP, reg2offset(r)));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   956
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   957
      // No data motion is needed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   958
      receiver_reg = r->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   959
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   960
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   961
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   962
  // Figure out which address we are really jumping to:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   963
  MethodHandles::generate_method_handle_dispatch(masm, special_dispatch,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   964
                                                 receiver_reg, member_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   965
                                                 /*for_compiler_entry:*/ true);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   966
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   967
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   968
////////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   969
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   970
//  Argument shufflers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   971
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   972
////////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   973
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   974
// Is the size of a vector size (in bytes) bigger than a size saved by default?
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   975
// 8 bytes registers are saved by default on z/Architecture.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   976
bool SharedRuntime::is_wide_vector(int size) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   977
  // Note, MaxVectorSize == 8 on this platform.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   978
  assert(size <= 8, "%d bytes vectors are not supported", size);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   979
  return size > 8;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   980
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   981
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   982
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   983
// An oop arg. Must pass a handle not the oop itself
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   984
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   985
static void object_move(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   986
                        OopMap *map,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   987
                        int oop_handle_offset,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   988
                        int framesize_in_slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   989
                        VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   990
                        VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   991
                        bool is_receiver,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   992
                        int *receiver_offset) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   993
  int frame_offset = framesize_in_slots*VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   994
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   995
  assert(!is_receiver || (is_receiver && (*receiver_offset == -1)), "only one receiving object per call, please.");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   996
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   997
  // Must pass a handle. First figure out the location we use as a handle.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   998
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
   999
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1000
    // Oop is already on the stack, put handle on stack or in register
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1001
    // If handle will be on the stack, use temp reg to calculate it.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1002
    Register rHandle = dst.first()->is_stack() ? Z_R1 : dst.first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1003
    Label    skip;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1004
    int      slot_in_older_frame = reg2slot(src.first());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1005
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1006
    guarantee(!is_receiver, "expecting receiver in register");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1007
    map->set_oop(VMRegImpl::stack2reg(slot_in_older_frame + framesize_in_slots));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1008
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1009
    __ add2reg(rHandle, reg2offset(src.first())+frame_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1010
    __ load_and_test_long(Z_R0, Address(rHandle));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1011
    __ z_brne(skip);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1012
    // Use a NULL handle if oop is NULL.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1013
    __ clear_reg(rHandle, true, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1014
    __ bind(skip);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1015
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1016
    // Copy handle to the right place (register or stack).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1017
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1018
      __ z_stg(rHandle, reg2offset(dst.first()), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1019
    } // else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1020
      // nothing to do. rHandle uses the correct register
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1021
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1022
    // Oop is passed in an input register. We must flush it to the stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1023
    const Register rOop = src.first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1024
    const Register rHandle = dst.first()->is_stack() ? Z_R1 : dst.first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1025
    int            oop_slot = (rOop->encoding()-Z_ARG1->encoding()) * VMRegImpl::slots_per_word + oop_handle_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1026
    int            oop_slot_offset = oop_slot*VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1027
    NearLabel skip;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1028
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1029
    if (is_receiver) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1030
      *receiver_offset = oop_slot_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1031
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1032
    map->set_oop(VMRegImpl::stack2reg(oop_slot));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1033
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1034
    // Flush Oop to stack, calculate handle.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1035
    __ z_stg(rOop, oop_slot_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1036
    __ add2reg(rHandle, oop_slot_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1037
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1038
    // If Oop == NULL, use a NULL handle.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1039
    __ compare64_and_branch(rOop, (RegisterOrConstant)0L, Assembler::bcondNotEqual, skip);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1040
    __ clear_reg(rHandle, true, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1041
    __ bind(skip);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1042
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1043
    // Copy handle to the right place (register or stack).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1044
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1045
      __ z_stg(rHandle, reg2offset(dst.first()), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1046
    } // else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1047
      // nothing to do here, since rHandle = dst.first()->as_Register in this case.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1048
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1049
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1050
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1051
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1052
// A float arg. May have to do float reg to int reg conversion
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1053
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1054
static void float_move(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1055
                       VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1056
                       VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1057
                       int framesize_in_slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1058
                       int workspace_slot_offset) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1059
  int frame_offset = framesize_in_slots * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1060
  int workspace_offset = workspace_slot_offset * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1061
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1062
  // We do not accept an argument in a VMRegPair to be spread over two slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1063
  // no matter what physical location (reg or stack) the slots may have.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1064
  // We just check for the unaccepted slot to be invalid.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1065
  assert(!src.second()->is_valid(), "float in arg spread over two slots");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1066
  assert(!dst.second()->is_valid(), "float out arg spread over two slots");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1067
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1068
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1069
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1070
      // stack -> stack. The easiest of the bunch.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1071
      __ z_mvc(Address(Z_SP, reg2offset(dst.first())),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1072
               Address(Z_SP, reg2offset(src.first()) + frame_offset), sizeof(float));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1073
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1074
      // stack to reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1075
      Address memaddr(Z_SP, reg2offset(src.first()) + frame_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1076
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1077
        __ mem2reg_opt(dst.first()->as_Register(), memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1078
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1079
        __ mem2freg_opt(dst.first()->as_FloatRegister(), memaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1080
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1081
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1082
  } else if (src.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1083
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1084
      // gpr -> stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1085
      __ reg2mem_opt(src.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1086
                     Address(Z_SP, reg2offset(dst.first()), false ));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1087
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1088
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1089
        // gpr -> gpr
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1090
        __ move_reg_if_needed(dst.first()->as_Register(), T_INT,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1091
                              src.first()->as_Register(), T_INT);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1092
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1093
        if (VM_Version::has_FPSupportEnhancements()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1094
          // gpr -> fpr. Exploit z10 capability of direct transfer.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1095
          __ z_ldgr(dst.first()->as_FloatRegister(), src.first()->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1096
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1097
          // gpr -> fpr. Use work space on stack to transfer data.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1098
          Address   stackaddr(Z_SP, workspace_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1099
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1100
          __ reg2mem_opt(src.first()->as_Register(), stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1101
          __ mem2freg_opt(dst.first()->as_FloatRegister(), stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1102
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1103
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1104
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1105
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1106
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1107
      // fpr -> stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1108
      __ freg2mem_opt(src.first()->as_FloatRegister(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1109
                      Address(Z_SP, reg2offset(dst.first())), false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1110
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1111
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1112
        if (VM_Version::has_FPSupportEnhancements()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1113
          // fpr -> gpr.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1114
          __ z_lgdr(dst.first()->as_Register(), src.first()->as_FloatRegister());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1115
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1116
          // fpr -> gpr. Use work space on stack to transfer data.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1117
          Address   stackaddr(Z_SP, workspace_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1118
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1119
          __ freg2mem_opt(src.first()->as_FloatRegister(), stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1120
          __ mem2reg_opt(dst.first()->as_Register(), stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1121
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1122
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1123
        // fpr -> fpr
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1124
        __ move_freg_if_needed(dst.first()->as_FloatRegister(), T_FLOAT,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1125
                               src.first()->as_FloatRegister(), T_FLOAT);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1126
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1127
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1128
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1129
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1130
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1131
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1132
// A double arg. May have to do double reg to long reg conversion
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1133
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1134
static void double_move(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1135
                        VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1136
                        VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1137
                        int framesize_in_slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1138
                        int workspace_slot_offset) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1139
  int frame_offset = framesize_in_slots*VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1140
  int workspace_offset = workspace_slot_offset*VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1141
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1142
  // Since src is always a java calling convention we know that the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1143
  // src pair is always either all registers or all stack (and aligned?)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1144
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1145
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1146
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1147
      // stack -> stack. The easiest of the bunch.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1148
      __ z_mvc(Address(Z_SP, reg2offset(dst.first())),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1149
               Address(Z_SP, reg2offset(src.first()) + frame_offset), sizeof(double));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1150
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1151
      // stack to reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1152
      Address stackaddr(Z_SP, reg2offset(src.first()) + frame_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1153
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1154
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1155
        __ mem2reg_opt(dst.first()->as_Register(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1156
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1157
        __ mem2freg_opt(dst.first()->as_FloatRegister(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1158
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1159
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1160
  } else if (src.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1161
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1162
      // gpr -> stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1163
      __ reg2mem_opt(src.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1164
                     Address(Z_SP, reg2offset(dst.first())));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1165
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1166
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1167
        // gpr -> gpr
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1168
        __ move_reg_if_needed(dst.first()->as_Register(), T_LONG,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1169
                              src.first()->as_Register(), T_LONG);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1170
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1171
        if (VM_Version::has_FPSupportEnhancements()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1172
          // gpr -> fpr. Exploit z10 capability of direct transfer.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1173
          __ z_ldgr(dst.first()->as_FloatRegister(), src.first()->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1174
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1175
          // gpr -> fpr. Use work space on stack to transfer data.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1176
          Address stackaddr(Z_SP, workspace_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1177
          __ reg2mem_opt(src.first()->as_Register(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1178
          __ mem2freg_opt(dst.first()->as_FloatRegister(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1179
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1180
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1181
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1182
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1183
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1184
      // fpr -> stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1185
      __ freg2mem_opt(src.first()->as_FloatRegister(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1186
                      Address(Z_SP, reg2offset(dst.first())));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1187
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1188
      if (dst.first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1189
        if (VM_Version::has_FPSupportEnhancements()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1190
          // fpr -> gpr. Exploit z10 capability of direct transfer.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1191
          __ z_lgdr(dst.first()->as_Register(), src.first()->as_FloatRegister());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1192
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1193
          // fpr -> gpr. Use work space on stack to transfer data.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1194
          Address stackaddr(Z_SP, workspace_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1195
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1196
          __ freg2mem_opt(src.first()->as_FloatRegister(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1197
          __ mem2reg_opt(dst.first()->as_Register(), stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1198
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1199
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1200
        // fpr -> fpr
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1201
        // In theory these overlap but the ordering is such that this is likely a nop.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1202
        __ move_freg_if_needed(dst.first()->as_FloatRegister(), T_DOUBLE,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1203
                               src.first()->as_FloatRegister(), T_DOUBLE);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1204
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1205
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1206
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1207
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1208
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1209
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1210
// A long arg.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1211
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1212
static void long_move(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1213
                      VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1214
                      VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1215
                      int framesize_in_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1216
  int frame_offset = framesize_in_slots*VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1217
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1218
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1219
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1220
      // stack -> stack. The easiest of the bunch.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1221
      __ z_mvc(Address(Z_SP, reg2offset(dst.first())),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1222
               Address(Z_SP, reg2offset(src.first()) + frame_offset), sizeof(long));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1223
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1224
      // stack to reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1225
      assert(dst.first()->is_Register(), "long dst value must be in GPR");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1226
      __ mem2reg_opt(dst.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1227
                      Address(Z_SP, reg2offset(src.first()) + frame_offset));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1228
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1229
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1230
    // reg to reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1231
    assert(src.first()->is_Register(), "long src value must be in GPR");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1232
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1233
      // reg -> stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1234
      __ reg2mem_opt(src.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1235
                     Address(Z_SP, reg2offset(dst.first())));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1236
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1237
      // reg -> reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1238
      assert(dst.first()->is_Register(), "long dst value must be in GPR");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1239
      __ move_reg_if_needed(dst.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1240
                            T_LONG, src.first()->as_Register(), T_LONG);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1241
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1242
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1243
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1244
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1245
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1246
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1247
// A int-like arg.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1248
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1249
// On z/Architecture we will store integer like items to the stack as 64 bit
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1250
// items, according to the z/Architecture ABI, even though Java would only store
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1251
// 32 bits for a parameter.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1252
// We do sign extension for all base types. That is ok since the only
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1253
// unsigned base type is T_CHAR, and T_CHAR uses only 16 bits of an int.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1254
// Sign extension 32->64 bit will thus not affect the value.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1255
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1256
static void move32_64(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1257
                      VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1258
                      VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1259
                      int framesize_in_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1260
  int frame_offset = framesize_in_slots * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1261
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1262
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1263
    Address memaddr(Z_SP, reg2offset(src.first()) + frame_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1264
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1265
      // stack -> stack. MVC not posible due to sign extension.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1266
      Address firstaddr(Z_SP, reg2offset(dst.first()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1267
      __ mem2reg_signed_opt(Z_R0_scratch, memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1268
      __ reg2mem_opt(Z_R0_scratch, firstaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1269
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1270
      // stack -> reg, sign extended
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1271
      __ mem2reg_signed_opt(dst.first()->as_Register(), memaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1272
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1273
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1274
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1275
      // reg -> stack, sign extended
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1276
      Address firstaddr(Z_SP, reg2offset(dst.first()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1277
      __ z_lgfr(src.first()->as_Register(), src.first()->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1278
      __ reg2mem_opt(src.first()->as_Register(), firstaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1279
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1280
      // reg -> reg, sign extended
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1281
      __ z_lgfr(dst.first()->as_Register(), src.first()->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1282
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1283
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1284
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1285
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1286
static void save_or_restore_arguments(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1287
                                      const int stack_slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1288
                                      const int total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1289
                                      const int arg_save_area,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1290
                                      OopMap *map,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1291
                                      VMRegPair *in_regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1292
                                      BasicType *in_sig_bt) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1293
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1294
  // If map is non-NULL then the code should store the values,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1295
  // otherwise it should load them.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1296
  int slot = arg_save_area;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1297
  // Handle double words first.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1298
  for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1299
    if (in_regs[i].first()->is_FloatRegister() && in_sig_bt[i] == T_DOUBLE) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1300
      int offset = slot * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1301
      slot += VMRegImpl::slots_per_word;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1302
      assert(slot <= stack_slots, "overflow (after DOUBLE stack slot)");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1303
      const FloatRegister   freg = in_regs[i].first()->as_FloatRegister();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1304
      Address   stackaddr(Z_SP, offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1305
      if (map != NULL) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1306
        __ freg2mem_opt(freg, stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1307
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1308
        __ mem2freg_opt(freg, stackaddr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1309
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1310
    } else if (in_regs[i].first()->is_Register() &&
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1311
               (in_sig_bt[i] == T_LONG || in_sig_bt[i] == T_ARRAY)) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1312
      int offset = slot * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1313
      const Register   reg = in_regs[i].first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1314
      if (map != NULL) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1315
        __ z_stg(reg, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1316
        if (in_sig_bt[i] == T_ARRAY) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1317
          map->set_oop(VMRegImpl::stack2reg(slot));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1318
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1319
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1320
        __ z_lg(reg, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1321
      }
47795
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1322
      slot += VMRegImpl::slots_per_word;
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1323
      assert(slot <= stack_slots, "overflow (after LONG/ARRAY stack slot)");
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1324
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1325
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1326
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1327
  // Save or restore single word registers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1328
  for (int i = 0; i < total_in_args; i++) {
47795
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1329
    if (in_regs[i].first()->is_Register()) {
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1330
      int offset = slot * VMRegImpl::stack_slot_size;
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1331
      // Value lives in an input register. Save it on stack.
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1332
      switch (in_sig_bt[i]) {
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1333
        case T_BOOLEAN:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1334
        case T_CHAR:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1335
        case T_BYTE:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1336
        case T_SHORT:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1337
        case T_INT: {
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1338
          const Register   reg = in_regs[i].first()->as_Register();
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1339
          Address   stackaddr(Z_SP, offset);
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1340
          if (map != NULL) {
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1341
            __ z_st(reg, stackaddr);
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1342
          } else {
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1343
            __ z_lgf(reg, stackaddr);
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1344
          }
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1345
          slot++;
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1346
          assert(slot <= stack_slots, "overflow (after INT or smaller stack slot)");
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1347
          break;
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1348
        }
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1349
        case T_ARRAY:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1350
        case T_LONG:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1351
          // handled above
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1352
          break;
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1353
        case T_OBJECT:
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1354
        default: ShouldNotReachHere();
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1355
      }
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1356
    } else if (in_regs[i].first()->is_FloatRegister()) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1357
      if (in_sig_bt[i] == T_FLOAT) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1358
        int offset = slot * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1359
        slot++;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1360
        assert(slot <= stack_slots, "overflow (after FLOAT stack slot)");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1361
        const FloatRegister   freg = in_regs[i].first()->as_FloatRegister();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1362
        Address   stackaddr(Z_SP, offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1363
        if (map != NULL) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1364
          __ freg2mem_opt(freg, stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1365
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1366
          __ mem2freg_opt(freg, stackaddr, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1367
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1368
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1369
    } else if (in_regs[i].first()->is_stack() &&
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1370
               in_sig_bt[i] == T_ARRAY && map != NULL) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1371
      int offset_in_older_frame = in_regs[i].first()->reg2stack() + SharedRuntime::out_preserve_stack_slots();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1372
      map->set_oop(VMRegImpl::stack2reg(offset_in_older_frame + stack_slots));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1373
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1374
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1375
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1376
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1377
// Check GCLocker::needs_gc and enter the runtime if it's true. This
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1378
// keeps a new JNI critical region from starting until a GC has been
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1379
// forced. Save down any oops in registers and describe them in an OopMap.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1380
static void check_needs_gc_for_critical_native(MacroAssembler   *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1381
                                                const int stack_slots,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1382
                                                const int total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1383
                                                const int arg_save_area,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1384
                                                OopMapSet *oop_maps,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1385
                                                VMRegPair *in_regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1386
                                                BasicType *in_sig_bt) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1387
  __ block_comment("check GCLocker::needs_gc");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1388
  Label cont;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1389
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1390
  // Check GCLocker::_needs_gc flag.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1391
  __ load_const_optimized(Z_R1_scratch, (long) GCLocker::needs_gc_address());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1392
  __ z_cli(0, Z_R1_scratch, 0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1393
  __ z_bre(cont);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1394
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1395
  // Save down any values that are live in registers and call into the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1396
  // runtime to halt for a GC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1397
  OopMap *map = new OopMap(stack_slots * 2, 0 /* arg_slots*/);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1398
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1399
  save_or_restore_arguments(masm, stack_slots, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1400
                            arg_save_area, map, in_regs, in_sig_bt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1401
  address the_pc = __ pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1402
  __ set_last_Java_frame(Z_SP, noreg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1403
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1404
  __ block_comment("block_for_jni_critical");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1405
  __ z_lgr(Z_ARG1, Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1406
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1407
  address entry_point = CAST_FROM_FN_PTR(address, SharedRuntime::block_for_jni_critical);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1408
  __ call_c(entry_point);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1409
  oop_maps->add_gc_map(__ offset(), map);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1410
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1411
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1412
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1413
  // Reload all the register arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1414
  save_or_restore_arguments(masm, stack_slots, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1415
                            arg_save_area, NULL, in_regs, in_sig_bt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1416
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1417
  __ bind(cont);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1418
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1419
  if (StressCriticalJNINatives) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1420
    // Stress register saving
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1421
    OopMap *map = new OopMap(stack_slots * 2, 0 /* arg_slots*/);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1422
    save_or_restore_arguments(masm, stack_slots, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1423
                              arg_save_area, map, in_regs, in_sig_bt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1424
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1425
    // Destroy argument registers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1426
    for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1427
      if (in_regs[i].first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1428
        // Don't set CC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1429
        __ clear_reg(in_regs[i].first()->as_Register(), true, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1430
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1431
        if (in_regs[i].first()->is_FloatRegister()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1432
          FloatRegister fr = in_regs[i].first()->as_FloatRegister();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1433
          __ z_lcdbr(fr, fr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1434
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1435
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1436
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1437
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1438
    save_or_restore_arguments(masm, stack_slots, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1439
                              arg_save_area, NULL, in_regs, in_sig_bt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1440
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1441
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1442
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1443
static void move_ptr(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1444
                     VMRegPair src,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1445
                     VMRegPair dst,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1446
                     int framesize_in_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1447
  int frame_offset = framesize_in_slots * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1448
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1449
  if (src.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1450
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1451
      // stack to stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1452
      __ mem2reg_opt(Z_R0_scratch, Address(Z_SP, reg2offset(src.first()) + frame_offset));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1453
      __ reg2mem_opt(Z_R0_scratch, Address(Z_SP, reg2offset(dst.first())));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1454
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1455
      // stack to reg
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1456
      __ mem2reg_opt(dst.first()->as_Register(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1457
                     Address(Z_SP, reg2offset(src.first()) + frame_offset));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1458
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1459
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1460
    if (dst.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1461
      // reg to stack
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1462
    __ reg2mem_opt(src.first()->as_Register(), Address(Z_SP, reg2offset(dst.first())));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1463
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1464
    __ lgr_if_needed(dst.first()->as_Register(), src.first()->as_Register());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1465
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1466
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1467
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1468
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1469
// Unpack an array argument into a pointer to the body and the length
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1470
// if the array is non-null, otherwise pass 0 for both.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1471
static void unpack_array_argument(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1472
                                   VMRegPair reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1473
                                   BasicType in_elem_type,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1474
                                   VMRegPair body_arg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1475
                                   VMRegPair length_arg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1476
                                   int framesize_in_slots) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1477
  Register tmp_reg = Z_tmp_2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1478
  Register tmp2_reg = Z_tmp_1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1479
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1480
  assert(!body_arg.first()->is_Register() || body_arg.first()->as_Register() != tmp_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1481
         "possible collision");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1482
  assert(!length_arg.first()->is_Register() || length_arg.first()->as_Register() != tmp_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1483
         "possible collision");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1484
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1485
  // Pass the length, ptr pair.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1486
  NearLabel set_out_args;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1487
  VMRegPair tmp, tmp2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1488
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1489
  tmp.set_ptr(tmp_reg->as_VMReg());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1490
  tmp2.set_ptr(tmp2_reg->as_VMReg());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1491
  if (reg.first()->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1492
    // Load the arg up from the stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1493
    move_ptr(masm, reg, tmp, framesize_in_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1494
    reg = tmp;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1495
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1496
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1497
  const Register first = reg.first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1498
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1499
  // Don't set CC, indicate unused result.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1500
  (void) __ clear_reg(tmp2_reg, true, false);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1501
  if (tmp_reg != first) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1502
    __ clear_reg(tmp_reg, true, false);  // Don't set CC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1503
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1504
  __ compare64_and_branch(first, (RegisterOrConstant)0L, Assembler::bcondEqual, set_out_args);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1505
  __ z_lgf(tmp2_reg, Address(first, arrayOopDesc::length_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1506
  __ add2reg(tmp_reg, arrayOopDesc::base_offset_in_bytes(in_elem_type), first);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1507
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1508
  __ bind(set_out_args);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1509
  move_ptr(masm, tmp, body_arg, framesize_in_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1510
  move32_64(masm, tmp2, length_arg, framesize_in_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1511
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1512
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1513
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1514
// Wrap a JNI call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1515
//----------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1516
#undef USE_RESIZE_FRAME
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1517
nmethod *SharedRuntime::generate_native_wrapper(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1518
                                                const methodHandle& method,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1519
                                                int compile_id,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1520
                                                BasicType *in_sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1521
                                                VMRegPair *in_regs,
57710
05ff6e27de45 8229236: CriticalJNINatives: dll handling should be done in native thread state
mdoerr
parents: 55749
diff changeset
  1522
                                                BasicType ret_type,
05ff6e27de45 8229236: CriticalJNINatives: dll handling should be done in native thread state
mdoerr
parents: 55749
diff changeset
  1523
                                                address critical_entry) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1524
#ifdef COMPILER2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1525
  int total_in_args = method->size_of_parameters();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1526
  if (method->is_method_handle_intrinsic()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1527
    vmIntrinsics::ID iid = method->intrinsic_id();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1528
    intptr_t start = (intptr_t) __ pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1529
    int vep_offset = ((intptr_t) __ pc()) - start;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1530
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1531
    gen_special_dispatch(masm, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1532
                         method->intrinsic_id(), in_sig_bt, in_regs);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1533
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1534
    int frame_complete = ((intptr_t)__ pc()) - start; // Not complete, period.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1535
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1536
    __ flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1537
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1538
    int stack_slots = SharedRuntime::out_preserve_stack_slots();  // No out slots at all, actually.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1539
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1540
    return nmethod::new_native_nmethod(method,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1541
                                       compile_id,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1542
                                       masm->code(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1543
                                       vep_offset,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1544
                                       frame_complete,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1545
                                       stack_slots / VMRegImpl::slots_per_word,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1546
                                       in_ByteSize(-1),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1547
                                       in_ByteSize(-1),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1548
                                       (OopMapSet *) NULL);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1549
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1550
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1551
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1552
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1553
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1554
  //  Precalculations before generating any code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1555
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1556
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1557
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1558
  bool is_critical_native = true;
57710
05ff6e27de45 8229236: CriticalJNINatives: dll handling should be done in native thread state
mdoerr
parents: 55749
diff changeset
  1559
  address native_func = critical_entry;
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1560
  if (native_func == NULL) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1561
    native_func = method->native_function();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1562
    is_critical_native = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1563
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1564
  assert(native_func != NULL, "must have function");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1565
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1566
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1567
  // We have received a description of where all the java args are located
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1568
  // on entry to the wrapper. We need to convert these args to where
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1569
  // the jni function will expect them. To figure out where they go
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1570
  // we convert the java signature to a C signature by inserting
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1571
  // the hidden arguments as arg[0] and possibly arg[1] (static method).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1572
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1573
  // The first hidden argument arg[0] is a pointer to the JNI environment.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1574
  // It is generated for every call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1575
  // The second argument arg[1] to the JNI call, which is hidden for static
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1576
  // methods, is the boxed lock object. For static calls, the lock object
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1577
  // is the static method itself. The oop is constructed here. for instance
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1578
  // calls, the lock is performed on the object itself, the pointer of
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1579
  // which is passed as the first visible argument.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1580
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1581
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1582
  // Additionally, on z/Architecture we must convert integers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1583
  // to longs in the C signature. We do this in advance in order to have
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1584
  // no trouble with indexes into the bt-arrays.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1585
  // So convert the signature and registers now, and adjust the total number
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1586
  // of in-arguments accordingly.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1587
  bool method_is_static = method->is_static();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1588
  int  total_c_args     = total_in_args;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1589
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1590
  if (!is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1591
    int n_hidden_args = method_is_static ? 2 : 1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1592
    total_c_args += n_hidden_args;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1593
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1594
    // No JNIEnv*, no this*, but unpacked arrays (base+length).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1595
    for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1596
      if (in_sig_bt[i] == T_ARRAY) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1597
        total_c_args ++;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1598
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1599
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1600
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1601
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1602
  BasicType *out_sig_bt = NEW_RESOURCE_ARRAY(BasicType, total_c_args);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1603
  VMRegPair *out_regs   = NEW_RESOURCE_ARRAY(VMRegPair, total_c_args);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1604
  BasicType* in_elem_bt = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1605
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1606
  // Create the signature for the C call:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1607
  //   1) add the JNIEnv*
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1608
  //   2) add the class if the method is static
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1609
  //   3) copy the rest of the incoming signature (shifted by the number of
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1610
  //      hidden arguments)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1611
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1612
  int argc = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1613
  if (!is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1614
    out_sig_bt[argc++] = T_ADDRESS;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1615
    if (method->is_static()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1616
      out_sig_bt[argc++] = T_OBJECT;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1617
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1618
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1619
    for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1620
      out_sig_bt[argc++] = in_sig_bt[i];
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1621
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1622
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1623
    in_elem_bt = NEW_RESOURCE_ARRAY(BasicType, total_in_args);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1624
    SignatureStream ss(method->signature());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1625
    int o = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1626
    for (int i = 0; i < total_in_args; i++, o++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1627
      if (in_sig_bt[i] == T_ARRAY) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1628
        // Arrays are passed as tuples (int, elem*).
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 52460
diff changeset
  1629
        Symbol* atype = ss.as_symbol();
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1630
        const char* at = atype->as_C_string();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1631
        if (strlen(at) == 2) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1632
          assert(at[0] == '[', "must be");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1633
          switch (at[1]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1634
            case 'B': in_elem_bt[o]  = T_BYTE; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1635
            case 'C': in_elem_bt[o]  = T_CHAR; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1636
            case 'D': in_elem_bt[o]  = T_DOUBLE; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1637
            case 'F': in_elem_bt[o]  = T_FLOAT; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1638
            case 'I': in_elem_bt[o]  = T_INT; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1639
            case 'J': in_elem_bt[o]  = T_LONG; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1640
            case 'S': in_elem_bt[o]  = T_SHORT; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1641
            case 'Z': in_elem_bt[o]  = T_BOOLEAN; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1642
            default: ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1643
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1644
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1645
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1646
        in_elem_bt[o] = T_VOID;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1647
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1648
      if (in_sig_bt[i] != T_VOID) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1649
        assert(in_sig_bt[i] == ss.type(), "must match");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1650
        ss.next();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1651
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1652
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1653
    assert(total_in_args == o, "must match");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1654
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1655
    for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1656
      if (in_sig_bt[i] == T_ARRAY) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1657
        // Arrays are passed as tuples (int, elem*).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1658
        out_sig_bt[argc++] = T_INT;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1659
        out_sig_bt[argc++] = T_ADDRESS;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1660
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1661
        out_sig_bt[argc++] = in_sig_bt[i];
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1662
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1663
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1664
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1665
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1666
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1667
  // Now figure out where the args must be stored and how much stack space
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1668
  // they require (neglecting out_preserve_stack_slots but providing space
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1669
  // for storing the first five register arguments).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1670
  // It's weird, see int_stk_helper.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1671
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1672
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1673
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1674
  // Compute framesize for the wrapper.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1675
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1676
  // - We need to handlize all oops passed in registers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1677
  // - We must create space for them here that is disjoint from the save area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1678
  // - We always just allocate 5 words for storing down these object.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1679
  //   This allows us to simply record the base and use the Ireg number to
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1680
  //   decide which slot to use.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1681
  // - Note that the reg number used to index the stack slot is the inbound
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1682
  //   number, not the outbound number.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1683
  // - We must shuffle args to match the native convention,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1684
  //   and to include var-args space.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1685
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1686
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1687
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1688
  // Calculate the total number of stack slots we will need:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1689
  // - 1) abi requirements
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1690
  // - 2) outgoing args
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1691
  // - 3) space for inbound oop handle area
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1692
  // - 4) space for handlizing a klass if static method
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1693
  // - 5) space for a lock if synchronized method
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1694
  // - 6) workspace (save rtn value, int<->float reg moves, ...)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1695
  // - 7) filler slots for alignment
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1696
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1697
  // Here is how the space we have allocated will look like.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1698
  // Since we use resize_frame, we do not create a new stack frame,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1699
  // but just extend the one we got with our own data area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1700
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1701
  // If an offset or pointer name points to a separator line, it is
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1702
  // assumed that addressing with offset 0 selects storage starting
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1703
  // at the first byte above the separator line.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1704
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1705
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1706
  //     ...                   ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1707
  //      | caller's frame      |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1708
  // FP-> |---------------------|
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1709
  //      | filler slots, if any|
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1710
  //     7| #slots == mult of 2 |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1711
  //      |---------------------|
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1712
  //      | work space          |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1713
  //     6| 2 slots = 8 bytes   |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1714
  //      |---------------------|
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1715
  //     5| lock box (if sync)  |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1716
  //      |---------------------| <- lock_slot_offset
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1717
  //     4| klass (if static)   |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1718
  //      |---------------------| <- klass_slot_offset
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1719
  //     3| oopHandle area      |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1720
  //      | (save area for      |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1721
  //      |  critical natives)  |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1722
  //      |                     |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1723
  //      |                     |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1724
  //      |---------------------| <- oop_handle_offset
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1725
  //     2| outbound memory     |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1726
  //     ...                   ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1727
  //      | based arguments     |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1728
  //      |---------------------|
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1729
  //      | vararg              |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1730
  //     ...                   ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1731
  //      | area                |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1732
  //      |---------------------| <- out_arg_slot_offset
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1733
  //     1| out_preserved_slots |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1734
  //     ...                   ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1735
  //      | (z_abi spec)        |
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1736
  // SP-> |---------------------| <- FP_slot_offset (back chain)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1737
  //     ...                   ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1738
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1739
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1740
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1741
  // *_slot_offset indicates offset from SP in #stack slots
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1742
  // *_offset      indicates offset from SP in #bytes
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1743
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1744
  int stack_slots = c_calling_convention(out_sig_bt, out_regs, /*regs2=*/NULL, total_c_args) + // 1+2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1745
                    SharedRuntime::out_preserve_stack_slots(); // see c_calling_convention
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1746
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1747
  // Now the space for the inbound oop handle area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1748
  int total_save_slots = RegisterImpl::number_of_arg_registers * VMRegImpl::slots_per_word;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1749
  if (is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1750
    // Critical natives may have to call out so they need a save area
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1751
    // for register arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1752
    int double_slots = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1753
    int single_slots = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1754
    for (int i = 0; i < total_in_args; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1755
      if (in_regs[i].first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1756
        const Register reg = in_regs[i].first()->as_Register();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1757
        switch (in_sig_bt[i]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1758
          case T_BOOLEAN:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1759
          case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1760
          case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1761
          case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1762
          case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1763
          // Fall through.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1764
          case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1765
          case T_LONG: double_slots++; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1766
          default:  ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1767
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1768
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1769
        if (in_regs[i].first()->is_FloatRegister()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1770
          switch (in_sig_bt[i]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1771
            case T_FLOAT:  single_slots++; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1772
            case T_DOUBLE: double_slots++; break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1773
            default:  ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1774
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1775
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1776
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1777
    }  // for
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
  1778
    total_save_slots = double_slots * 2 + align_up(single_slots, 2); // Round to even.
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1779
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1780
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1781
  int oop_handle_slot_offset = stack_slots;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1782
  stack_slots += total_save_slots;                                        // 3)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1783
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1784
  int klass_slot_offset = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1785
  int klass_offset      = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1786
  if (method_is_static && !is_critical_native) {                          // 4)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1787
    klass_slot_offset  = stack_slots;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1788
    klass_offset       = klass_slot_offset * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1789
    stack_slots       += VMRegImpl::slots_per_word;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1790
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1791
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1792
  int lock_slot_offset = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1793
  int lock_offset      = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1794
  if (method->is_synchronized()) {                                        // 5)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1795
    lock_slot_offset   = stack_slots;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1796
    lock_offset        = lock_slot_offset * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1797
    stack_slots       += VMRegImpl::slots_per_word;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1798
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1799
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1800
  int workspace_slot_offset= stack_slots;                                 // 6)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1801
  stack_slots         += 2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1802
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1803
  // Now compute actual number of stack words we need.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1804
  // Round to align stack properly.
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
  1805
  stack_slots = align_up(stack_slots,                                     // 7)
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1806
                         frame::alignment_in_bytes / VMRegImpl::stack_slot_size);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1807
  int frame_size_in_bytes = stack_slots * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1808
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1809
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1810
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1811
  // Now we can start generating code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1812
  ///////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1813
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1814
  unsigned int wrapper_CodeStart  = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1815
  unsigned int wrapper_UEPStart;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1816
  unsigned int wrapper_VEPStart;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1817
  unsigned int wrapper_FrameDone;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1818
  unsigned int wrapper_CRegsSet;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1819
  Label     handle_pending_exception;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1820
  Label     ic_miss;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1821
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1822
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1823
  // Unverified entry point (UEP)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1824
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1825
  wrapper_UEPStart = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1826
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1827
  // check ic: object class <-> cached class
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1828
  if (!method_is_static) __ nmethod_UEP(ic_miss);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1829
  // Fill with nops (alignment of verified entry point).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1830
  __ align(CodeEntryAlignment);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1831
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1832
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1833
  // Verified entry point (VEP)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1834
  //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1835
  wrapper_VEPStart = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1836
55343
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1837
  if (VM_Version::supports_fast_class_init_checks() && method->needs_clinit_barrier()) {
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1838
    Label L_skip_barrier;
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1839
    Register klass = Z_R1_scratch;
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1840
    // Notify OOP recorder (don't need the relocation)
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1841
    AddressLiteral md = __ constant_metadata_address(method->method_holder());
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1842
    __ load_const_optimized(klass, md.value());
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1843
    __ clinit_barrier(klass, Z_thread, &L_skip_barrier /*L_fast_path*/);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1844
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1845
    __ load_const_optimized(klass, SharedRuntime::get_handle_wrong_method_stub());
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1846
    __ z_br(klass);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1847
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1848
    __ bind(L_skip_barrier);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1849
  }
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  1850
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1851
  __ save_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1852
  __ generate_stack_overflow_check(frame_size_in_bytes);  // Check before creating frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1853
#ifndef USE_RESIZE_FRAME
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1854
  __ push_frame(frame_size_in_bytes);                     // Create a new frame for the wrapper.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1855
#else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1856
  __ resize_frame(-frame_size_in_bytes, Z_R0_scratch);    // No new frame for the wrapper.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1857
                                                          // Just resize the existing one.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1858
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1859
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1860
  wrapper_FrameDone = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1861
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1862
  __ verify_thread();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1863
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1864
  // Native nmethod wrappers never take possession of the oop arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1865
  // So the caller will gc the arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1866
  // The only thing we need an oopMap for is if the call is static.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1867
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1868
  // An OopMap for lock (and class if static), and one for the VM call itself
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1869
  OopMapSet  *oop_maps        = new OopMapSet();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1870
  OopMap     *map             = new OopMap(stack_slots * 2, 0 /* arg_slots*/);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1871
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1872
  if (is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1873
    check_needs_gc_for_critical_native(masm, stack_slots, total_in_args,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1874
                                       oop_handle_slot_offset, oop_maps, in_regs, in_sig_bt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1875
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1876
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1877
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1878
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1879
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1880
  // The Grand Shuffle
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1881
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1882
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1883
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1884
  // We immediately shuffle the arguments so that for any vm call we have
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1885
  // to make from here on out (sync slow path, jvmti, etc.) we will have
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1886
  // captured the oops from our caller and have a valid oopMap for them.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1887
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1888
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1889
  // Natives require 1 or 2 extra arguments over the normal ones: the JNIEnv*
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1890
  // (derived from JavaThread* which is in Z_thread) and, if static,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1891
  // the class mirror instead of a receiver. This pretty much guarantees that
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1892
  // register layout will not match. We ignore these extra arguments during
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1893
  // the shuffle. The shuffle is described by the two calling convention
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1894
  // vectors we have in our possession. We simply walk the java vector to
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1895
  // get the source locations and the c vector to get the destinations.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1896
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1897
  // This is a trick. We double the stack slots so we can claim
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1898
  // the oops in the caller's frame. Since we are sure to have
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1899
  // more args than the caller doubling is enough to make
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1900
  // sure we can capture all the incoming oop args from the caller.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1901
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1902
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1903
  // Record sp-based slot for receiver on stack for non-static methods.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1904
  int receiver_offset = -1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1905
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1906
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1907
  // We move the arguments backwards because the floating point registers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1908
  // destination will always be to a register with a greater or equal
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1909
  // register number or the stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1910
  //   jix is the index of the incoming Java arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1911
  //   cix is the index of the outgoing C arguments.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1912
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1913
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1914
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1915
  bool reg_destroyed[RegisterImpl::number_of_registers];
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1916
  bool freg_destroyed[FloatRegisterImpl::number_of_registers];
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1917
  for (int r = 0; r < RegisterImpl::number_of_registers; r++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1918
    reg_destroyed[r] = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1919
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1920
  for (int f = 0; f < FloatRegisterImpl::number_of_registers; f++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1921
    freg_destroyed[f] = false;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1922
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1923
#endif // ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1924
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1925
  for (int jix = total_in_args - 1, cix = total_c_args - 1; jix >= 0; jix--, cix--) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1926
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1927
    if (in_regs[jix].first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1928
      assert(!reg_destroyed[in_regs[jix].first()->as_Register()->encoding()], "ack!");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1929
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1930
      if (in_regs[jix].first()->is_FloatRegister()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1931
        assert(!freg_destroyed[in_regs[jix].first()->as_FloatRegister()->encoding()], "ack!");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1932
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1933
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1934
    if (out_regs[cix].first()->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1935
      reg_destroyed[out_regs[cix].first()->as_Register()->encoding()] = true;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1936
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1937
      if (out_regs[cix].first()->is_FloatRegister()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1938
        freg_destroyed[out_regs[cix].first()->as_FloatRegister()->encoding()] = true;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1939
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1940
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1941
#endif // ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1942
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1943
    switch (in_sig_bt[jix]) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1944
      // Due to casting, small integers should only occur in pairs with type T_LONG.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1945
      case T_BOOLEAN:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1946
      case T_CHAR:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1947
      case T_BYTE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1948
      case T_SHORT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1949
      case T_INT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1950
        // Move int and do sign extension.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1951
        move32_64(masm, in_regs[jix], out_regs[cix], stack_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1952
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1953
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1954
      case T_LONG :
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1955
        long_move(masm, in_regs[jix], out_regs[cix], stack_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1956
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1957
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1958
      case T_ARRAY:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1959
        if (is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1960
          int body_arg = cix;
47795
5a69ba3a4fd1 8190781: ppc64 + s390: Fix CriticalJNINatives
mdoerr
parents: 47216
diff changeset
  1961
          cix -= 1; // Point to length arg.
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1962
          unpack_array_argument(masm, in_regs[jix], in_elem_bt[jix], out_regs[body_arg], out_regs[cix], stack_slots);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1963
          break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1964
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1965
        // else fallthrough
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1966
      case T_OBJECT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1967
        assert(!is_critical_native, "no oop arguments");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1968
        object_move(masm, map, oop_handle_slot_offset, stack_slots, in_regs[jix], out_regs[cix],
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1969
                    ((jix == 0) && (!method_is_static)),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1970
                    &receiver_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1971
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1972
      case T_VOID:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1973
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1974
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1975
      case T_FLOAT:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1976
        float_move(masm, in_regs[jix], out_regs[cix], stack_slots, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1977
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1978
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1979
      case T_DOUBLE:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1980
        assert(jix+1 <  total_in_args && in_sig_bt[jix+1]  == T_VOID && out_sig_bt[cix+1] == T_VOID, "bad arg list");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1981
        double_move(masm, in_regs[jix], out_regs[cix], stack_slots, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1982
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1983
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1984
      case T_ADDRESS:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1985
        assert(false, "found T_ADDRESS in java args");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1986
        break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1987
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1988
      default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1989
        ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1990
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1991
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1992
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1993
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1994
  // Pre-load a static method's oop into ARG2.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1995
  // Used both by locking code and the normal JNI call code.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1996
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1997
  if (method_is_static && !is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1998
    __ set_oop_constant(JNIHandles::make_local(method->method_holder()->java_mirror()), Z_ARG2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  1999
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2000
    // Now handlize the static class mirror in ARG2. It's known not-null.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2001
    __ z_stg(Z_ARG2, klass_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2002
    map->set_oop(VMRegImpl::stack2reg(klass_slot_offset));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2003
    __ add2reg(Z_ARG2, klass_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2004
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2005
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2006
  // Get JNIEnv* which is first argument to native.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2007
  if (!is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2008
    __ add2reg(Z_ARG1, in_bytes(JavaThread::jni_environment_offset()), Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2009
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2010
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2011
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2012
  // We have all of the arguments setup at this point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2013
  // We MUST NOT touch any outgoing regs from this point on.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2014
  // So if we must call out we must push a new frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2015
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2016
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2017
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2018
  // Calc the current pc into Z_R10 and into wrapper_CRegsSet.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2019
  // Both values represent the same position.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2020
  __ get_PC(Z_R10);                // PC into register
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2021
  wrapper_CRegsSet = __ offset();  // and into into variable.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2022
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2023
  // Z_R10 now has the pc loaded that we will use when we finally call to native.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2024
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2025
  // We use the same pc/oopMap repeatedly when we call out.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2026
  oop_maps->add_gc_map((int)(wrapper_CRegsSet-wrapper_CodeStart), map);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2027
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2028
  // Lock a synchronized method.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2029
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2030
  if (method->is_synchronized()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2031
    assert(!is_critical_native, "unhandled");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2032
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2033
    // ATTENTION: args and Z_R10 must be preserved.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2034
    Register r_oop  = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2035
    Register r_box  = Z_R12;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2036
    Register r_tmp1 = Z_R13;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2037
    Register r_tmp2 = Z_R7;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2038
    Label done;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2039
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2040
    // Load the oop for the object or class. R_carg2_classorobject contains
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2041
    // either the handlized oop from the incoming arguments or the handlized
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2042
    // class mirror (if the method is static).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2043
    __ z_lg(r_oop, 0, Z_ARG2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2044
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2045
    lock_offset = (lock_slot_offset * VMRegImpl::stack_slot_size);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2046
    // Get the lock box slot's address.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2047
    __ add2reg(r_box, lock_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2048
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2049
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2050
    if (UseBiasedLocking)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2051
      // Making the box point to itself will make it clear it went unused
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2052
      // but also be obviously invalid.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2053
      __ z_stg(r_box, 0, r_box);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2054
#endif // ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2055
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2056
    // Try fastpath for locking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2057
    // Fast_lock kills r_temp_1, r_temp_2. (Don't use R1 as temp, won't work!)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2058
    __ compiler_fast_lock_object(r_oop, r_box, r_tmp1, r_tmp2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2059
    __ z_bre(done);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2060
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2061
    //-------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2062
    // None of the above fast optimizations worked so we have to get into the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2063
    // slow case of monitor enter. Inline a special case of call_VM that
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2064
    // disallows any pending_exception.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2065
    //-------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2066
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2067
    Register oldSP = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2068
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2069
    __ z_lgr(oldSP, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2070
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2071
    RegisterSaver::save_live_registers(masm, RegisterSaver::arg_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2072
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2073
    // Prepare arguments for call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2074
    __ z_lg(Z_ARG1, 0, Z_ARG2); // Ynboxed class mirror or unboxed object.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2075
    __ add2reg(Z_ARG2, lock_offset, oldSP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2076
    __ z_lgr(Z_ARG3, Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2077
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2078
    __ set_last_Java_frame(oldSP, Z_R10 /* gc map pc */);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2079
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2080
    // Do the call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2081
    __ load_const_optimized(Z_R1_scratch, CAST_FROM_FN_PTR(address, SharedRuntime::complete_monitor_locking_C));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2082
    __ call(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2083
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2084
    __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2085
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2086
    RegisterSaver::restore_live_registers(masm, RegisterSaver::arg_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2087
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2088
    { Label L;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2089
      __ load_and_test_long(Z_R0, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2090
      __ z_bre(L);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2091
      __ stop("no pending exception allowed on exit from IR::monitorenter");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2092
      __ bind(L);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2093
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2094
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2095
    __ bind(done);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2096
  } // lock for synchronized methods
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2097
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2098
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2099
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2100
  // Finally just about ready to make the JNI call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2101
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2102
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2103
  // Use that pc we placed in Z_R10 a while back as the current frame anchor.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2104
  __ set_last_Java_frame(Z_SP, Z_R10);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2105
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2106
  // Transition from _thread_in_Java to _thread_in_native.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2107
  __ set_thread_state(_thread_in_native);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2108
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2109
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2110
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2111
  // This is the JNI call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2112
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2113
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2114
  __ call_c(native_func);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2115
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2116
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2117
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2118
  // We have survived the call once we reach here.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2119
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2120
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2121
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2122
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2123
  // Unpack native results.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2124
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2125
  // For int-types, we do any needed sign-extension required.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2126
  // Care must be taken that the return value (in Z_ARG1 = Z_RET = Z_R2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2127
  // or in Z_FARG0 = Z_FRET = Z_F0) will survive any VM calls for
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2128
  // blocking or unlocking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2129
  // An OOP result (handle) is done specially in the slow-path code.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2130
  //--------------------------------------------------------------------
49368
2ed1c37df3a5 8197405: Improve messages of AbstractMethodErrors and IncompatibleClassChangeErrors.
goetz
parents: 48557
diff changeset
  2131
  switch (ret_type) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2132
    case T_VOID:    break;         // Nothing to do!
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2133
    case T_FLOAT:   break;         // Got it where we want it (unless slow-path)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2134
    case T_DOUBLE:  break;         // Got it where we want it (unless slow-path)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2135
    case T_LONG:    break;         // Got it where we want it (unless slow-path)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2136
    case T_OBJECT:  break;         // Really a handle.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2137
                                   // Cannot de-handlize until after reclaiming jvm_lock.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2138
    case T_ARRAY:   break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2139
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2140
    case T_BOOLEAN:                // 0 -> false(0); !0 -> true(1)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2141
      __ z_lngfr(Z_RET, Z_RET);    // Force sign bit on except for zero.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2142
      __ z_srlg(Z_RET, Z_RET, 63); // Shift sign bit into least significant pos.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2143
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2144
    case T_BYTE:    __ z_lgbr(Z_RET, Z_RET);  break; // sign extension
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2145
    case T_CHAR:    __ z_llghr(Z_RET, Z_RET); break; // unsigned result
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2146
    case T_SHORT:   __ z_lghr(Z_RET, Z_RET);  break; // sign extension
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2147
    case T_INT:     __ z_lgfr(Z_RET, Z_RET);  break; // sign-extend for beauty.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2148
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2149
    default:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2150
      ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2151
      break;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2152
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2153
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2154
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2155
  // Switch thread to "native transition" state before reading the synchronization state.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2156
  // This additional state is necessary because reading and testing the synchronization
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2157
  // state is not atomic w.r.t. GC, as this scenario demonstrates:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2158
  //   - Java thread A, in _thread_in_native state, loads _not_synchronized and is preempted.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2159
  //   - VM thread changes sync state to synchronizing and suspends threads for GC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2160
  //   - Thread A is resumed to finish this native method, but doesn't block here since it
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2161
  //     didn't see any synchronization in progress, and escapes.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2162
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2163
  // Transition from _thread_in_native to _thread_in_native_trans.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2164
  __ set_thread_state(_thread_in_native_trans);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2165
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2166
  // Safepoint synchronization
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2167
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2168
  // Must we block?
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2169
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2170
  // Block, if necessary, before resuming in _thread_in_Java state.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2171
  // In order for GC to work, don't clear the last_Java_sp until after blocking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2172
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2173
  Label after_transition;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2174
  {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2175
    Label no_block, sync;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2176
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2177
    save_native_result(masm, ret_type, workspace_slot_offset); // Make Z_R2 available as work reg.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2178
52460
f1bb77833b59 8213436: Obsolete UseMembar
mikael
parents: 51996
diff changeset
  2179
    // Force this write out before the read below.
f1bb77833b59 8213436: Obsolete UseMembar
mikael
parents: 51996
diff changeset
  2180
    __ z_fence();
51996
84743156e780 8188764: Obsolete AssumeMP and then remove all support for non-MP builds
dholmes
parents: 51966
diff changeset
  2181
48332
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  2182
    __ safepoint_poll(sync, Z_R1);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2183
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2184
    __ load_and_test_int(Z_R0, Address(Z_thread, JavaThread::suspend_flags_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2185
    __ z_bre(no_block);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2186
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2187
    // Block. Save any potential method result value before the operation and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2188
    // use a leaf call to leave the last_Java_frame setup undisturbed. Doing this
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2189
    // lets us share the oopMap we used when we went native rather than create
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2190
    // a distinct one for this pc.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2191
    //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2192
    __ bind(sync);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2193
    __ z_acquire();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2194
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2195
    address entry_point = is_critical_native ? CAST_FROM_FN_PTR(address, JavaThread::check_special_condition_for_native_trans_and_transition)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2196
                                             : CAST_FROM_FN_PTR(address, JavaThread::check_special_condition_for_native_trans);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2197
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2198
    __ call_VM_leaf(entry_point, Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2199
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2200
    if (is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2201
      restore_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2202
      __ z_bru(after_transition); // No thread state transition here.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2203
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2204
    __ bind(no_block);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2205
    restore_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2206
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2207
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2208
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2209
  // Thread state is thread_in_native_trans. Any safepoint blocking has
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2210
  // already happened so we can now change state to _thread_in_Java.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2211
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2212
  // Transition from _thread_in_native_trans to _thread_in_Java.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2213
  __ set_thread_state(_thread_in_Java);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2214
  __ bind(after_transition);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2215
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2216
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2217
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2218
  // Reguard any pages if necessary.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2219
  // Protect native result from being destroyed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2220
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2221
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2222
  Label no_reguard;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2223
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2224
  __ z_cli(Address(Z_thread, JavaThread::stack_guard_state_offset() + in_ByteSize(sizeof(JavaThread::StackGuardState) - 1)),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2225
           JavaThread::stack_guard_yellow_reserved_disabled);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2226
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2227
  __ z_bre(no_reguard);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2228
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2229
  save_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2230
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, SharedRuntime::reguard_yellow_pages), Z_method);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2231
  restore_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2232
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2233
  __ bind(no_reguard);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2234
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2235
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2236
  // Synchronized methods (slow path only)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2237
  // No pending exceptions for now.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2238
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2239
  // Handle possibly pending exception (will unlock if necessary).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2240
  // Native result is, if any is live, in Z_FRES or Z_RES.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2241
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2242
  // Unlock
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2243
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2244
  if (method->is_synchronized()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2245
    const Register r_oop        = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2246
    const Register r_box        = Z_R12;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2247
    const Register r_tmp1       = Z_R13;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2248
    const Register r_tmp2       = Z_R7;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2249
    Label done;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2250
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2251
    // Get unboxed oop of class mirror or object ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2252
    int   offset = method_is_static ? klass_offset : receiver_offset;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2253
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2254
    assert(offset != -1, "");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2255
    __ z_lg(r_oop, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2256
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2257
    // ... and address of lock object box.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2258
    __ add2reg(r_box, lock_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2259
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2260
    // Try fastpath for unlocking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2261
    __ compiler_fast_unlock_object(r_oop, r_box, r_tmp1, r_tmp2); // Don't use R1 as temp.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2262
    __ z_bre(done);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2263
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2264
    // Slow path for unlocking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2265
    // Save and restore any potential method result value around the unlocking operation.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2266
    const Register R_exc = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2267
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2268
    save_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2269
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2270
    // Must save pending exception around the slow-path VM call. Since it's a
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2271
    // leaf call, the pending exception (if any) can be kept in a register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2272
    __ z_lg(R_exc, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2273
    assert(R_exc->is_nonvolatile(), "exception register must be non-volatile");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2274
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2275
    // Must clear pending-exception before re-entering the VM. Since this is
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2276
    // a leaf call, pending-exception-oop can be safely kept in a register.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2277
    __ clear_mem(Address(Z_thread, Thread::pending_exception_offset()), sizeof(intptr_t));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2278
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2279
    // Inline a special case of call_VM that disallows any pending_exception.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2280
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2281
    // Get locked oop from the handle we passed to jni.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2282
    __ z_lg(Z_ARG1, offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2283
    __ add2reg(Z_ARG2, lock_offset, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2284
    __ z_lgr(Z_ARG3, Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2285
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2286
    __ load_const_optimized(Z_R1_scratch, CAST_FROM_FN_PTR(address, SharedRuntime::complete_monitor_unlocking_C));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2287
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2288
    __ call(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2289
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2290
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2291
    {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2292
      Label L;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2293
      __ load_and_test_long(Z_R0, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2294
      __ z_bre(L);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2295
      __ stop("no pending exception allowed on exit from IR::monitorexit");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2296
      __ bind(L);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2297
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2298
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2299
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2300
    // Check_forward_pending_exception jump to forward_exception if any pending
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2301
    // exception is set. The forward_exception routine expects to see the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2302
    // exception in pending_exception and not in a register. Kind of clumsy,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2303
    // since all folks who branch to forward_exception must have tested
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2304
    // pending_exception first and hence have it in a register already.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2305
    __ z_stg(R_exc, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2306
    restore_native_result(masm, ret_type, workspace_slot_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2307
    __ z_bru(done);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2308
    __ z_illtrap(0x66);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2309
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2310
    __ bind(done);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2311
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2312
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2313
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2314
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2315
  // Clear "last Java frame" SP and PC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2316
  //--------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2317
  __ verify_thread(); // Z_thread must be correct.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2318
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2319
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2320
44406
a46a6c4d1dd9 8176100: [REDO][REDO] G1 Needs pre barrier on dereference of weak JNI handles
mgerdin
parents: 44093
diff changeset
  2321
  // Unpack oop result, e.g. JNIHandles::resolve result.
58356
feff88c68082 8231448: s390 and ppc - replace JVM type comparisons to T_OBJECT and T_ARRAY with call to is_reference_type
mbaesken
parents: 57710
diff changeset
  2322
  if (is_reference_type(ret_type)) {
44406
a46a6c4d1dd9 8176100: [REDO][REDO] G1 Needs pre barrier on dereference of weak JNI handles
mgerdin
parents: 44093
diff changeset
  2323
    __ resolve_jobject(Z_RET, /* tmp1 */ Z_R13, /* tmp2 */ Z_R7);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2324
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2325
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2326
  if (CheckJNICalls) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2327
    // clear_pending_jni_exception_check
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2328
    __ clear_mem(Address(Z_thread, JavaThread::pending_jni_exception_check_fn_offset()), sizeof(oop));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2329
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2330
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2331
  // Reset handle block.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2332
  if (!is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2333
    __ z_lg(Z_R1_scratch, Address(Z_thread, JavaThread::active_handles_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2334
    __ clear_mem(Address(Z_R1_scratch, JNIHandleBlock::top_offset_in_bytes()), 4);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2335
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2336
    // Check for pending exceptions.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2337
    __ load_and_test_long(Z_R0, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2338
    __ z_brne(handle_pending_exception);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2339
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2340
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2341
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2342
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2343
  // Return
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2344
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2345
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2346
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2347
#ifndef USE_RESIZE_FRAME
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2348
  __ pop_frame();                     // Pop wrapper frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2349
#else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2350
  __ resize_frame(frame_size_in_bytes, Z_R0_scratch);  // Revert stack extension.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2351
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2352
  __ restore_return_pc();             // This is the way back to the caller.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2353
  __ z_br(Z_R14);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2354
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2355
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2356
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2357
  // Out-of-line calls to the runtime.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2358
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2359
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2360
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2361
  if (!is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2362
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2363
    //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2364
    // Handler for pending exceptions (out-of-line).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2365
    //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2366
    // Since this is a native call, we know the proper exception handler
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2367
    // is the empty function. We just pop this frame and then jump to
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2368
    // forward_exception_entry. Z_R14 will contain the native caller's
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2369
    // return PC.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2370
    __ bind(handle_pending_exception);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2371
    __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2372
    __ load_const_optimized(Z_R1_scratch, StubRoutines::forward_exception_entry());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2373
    __ restore_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2374
    __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2375
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2376
    //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2377
    // Handler for a cache miss (out-of-line)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2378
    //---------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2379
    __ call_ic_miss_handler(ic_miss, 0x77, 0, Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2380
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2381
  __ flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2382
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2383
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2384
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2385
  // end of code generation
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2386
  //////////////////////////////////////////////////////////////////////
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2387
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2388
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2389
  nmethod *nm = nmethod::new_native_nmethod(method,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2390
                                            compile_id,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2391
                                            masm->code(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2392
                                            (int)(wrapper_VEPStart-wrapper_CodeStart),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2393
                                            (int)(wrapper_FrameDone-wrapper_CodeStart),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2394
                                            stack_slots / VMRegImpl::slots_per_word,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2395
                                            (method_is_static ? in_ByteSize(klass_offset) : in_ByteSize(receiver_offset)),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2396
                                            in_ByteSize(lock_offset),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2397
                                            oop_maps);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2398
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2399
  if (is_critical_native) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2400
    nm->set_lazy_critical_native(true);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2401
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2402
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2403
  return nm;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2404
#else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2405
  ShouldNotReachHere();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2406
  return NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2407
#endif // COMPILER2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2408
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2409
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2410
static address gen_c2i_adapter(MacroAssembler  *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2411
                               int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2412
                               int comp_args_on_stack,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2413
                               const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2414
                               const VMRegPair *regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2415
                               Label &skip_fixup) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2416
  // Before we get into the guts of the C2I adapter, see if we should be here
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2417
  // at all. We've come from compiled code and are attempting to jump to the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2418
  // interpreter, which means the caller made a static call to get here
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2419
  // (vcalls always get a compiled target if there is one). Check for a
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2420
  // compiled target. If there is one, we need to patch the caller's call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2421
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2422
  // These two defs MUST MATCH code in gen_i2c2i_adapter!
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2423
  const Register ientry = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2424
  const Register code   = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2425
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2426
  address c2i_entrypoint;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2427
  Label   patch_callsite;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2428
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2429
  // Regular (verified) c2i entry point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2430
  c2i_entrypoint = __ pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2431
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2432
  // Call patching needed?
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2433
  __ load_and_test_long(Z_R0_scratch, method_(code));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2434
  __ z_lg(ientry, method_(interpreter_entry));  // Preload interpreter entry (also if patching).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2435
  __ z_brne(patch_callsite);                    // Patch required if code != NULL (compiled target exists).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2436
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2437
  __ bind(skip_fixup);  // Return point from patch_callsite.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2438
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2439
  // Since all args are passed on the stack, total_args_passed*wordSize is the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2440
  // space we need. We need ABI scratch area but we use the caller's since
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2441
  // it has already been allocated.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2442
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2443
  const int abi_scratch = frame::z_top_ijava_frame_abi_size;
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
  2444
  int       extraspace  = align_up(total_args_passed, 2)*wordSize + abi_scratch;
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2445
  Register  sender_SP   = Z_R10;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2446
  Register  value       = Z_R12;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2447
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2448
  // Remember the senderSP so we can pop the interpreter arguments off of the stack.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2449
  // In addition, frame manager expects initial_caller_sp in Z_R10.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2450
  __ z_lgr(sender_SP, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2451
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2452
  // This should always fit in 14 bit immediate.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2453
  __ resize_frame(-extraspace, Z_R0_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2454
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2455
  // We use the caller's ABI scratch area (out_preserved_stack_slots) for the initial
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2456
  // args. This essentially moves the callers ABI scratch area from the top to the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2457
  // bottom of the arg area.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2458
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2459
  int st_off =  extraspace - wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2460
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2461
  // Now write the args into the outgoing interpreter space.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2462
  for (int i = 0; i < total_args_passed; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2463
    VMReg r_1 = regs[i].first();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2464
    VMReg r_2 = regs[i].second();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2465
    if (!r_1->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2466
      assert(!r_2->is_valid(), "");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2467
      continue;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2468
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2469
    if (r_1->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2470
      // The calling convention produces OptoRegs that ignore the preserve area (abi scratch).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2471
      // We must account for it here.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2472
      int ld_off = (r_1->reg2stack() + SharedRuntime::out_preserve_stack_slots()) * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2473
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2474
      if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2475
        __ z_mvc(Address(Z_SP, st_off), Address(sender_SP, ld_off), sizeof(void*));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2476
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2477
        // longs are given 2 64-bit slots in the interpreter,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2478
        // but the data is passed in only 1 slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2479
        if (sig_bt[i] == T_LONG || sig_bt[i] == T_DOUBLE) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2480
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2481
          __ clear_mem(Address(Z_SP, st_off), sizeof(void *));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2482
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2483
          st_off -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2484
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2485
        __ z_mvc(Address(Z_SP, st_off), Address(sender_SP, ld_off), sizeof(void*));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2486
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2487
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2488
      if (r_1->is_Register()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2489
        if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2490
          __ z_st(r_1->as_Register(), st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2491
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2492
          // longs are given 2 64-bit slots in the interpreter, but the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2493
          // data is passed in only 1 slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2494
          if (sig_bt[i] == T_LONG || sig_bt[i] == T_DOUBLE) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2495
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2496
            __ clear_mem(Address(Z_SP, st_off), sizeof(void *));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2497
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2498
            st_off -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2499
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2500
          __ z_stg(r_1->as_Register(), st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2501
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2502
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2503
        assert(r_1->is_FloatRegister(), "");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2504
        if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2505
          __ z_ste(r_1->as_FloatRegister(), st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2506
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2507
          // In 64bit, doubles are given 2 64-bit slots in the interpreter, but the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2508
          // data is passed in only 1 slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2509
          // One of these should get known junk...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2510
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2511
          __ z_lzdr(Z_F1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2512
          __ z_std(Z_F1, st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2513
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2514
          st_off-=wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2515
          __ z_std(r_1->as_FloatRegister(), st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2516
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2517
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2518
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2519
    st_off -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2520
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2521
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2522
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2523
  // Jump to the interpreter just as if interpreter was doing it.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2524
  __ add2reg(Z_esp, st_off, Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2525
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2526
  // Frame_manager expects initial_caller_sp (= SP without resize by c2i) in Z_R10.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2527
  __ z_br(ientry);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2528
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2529
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2530
  // Prevent illegal entry to out-of-line code.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2531
  __ z_illtrap(0x22);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2532
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2533
  // Generate out-of-line runtime call to patch caller,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2534
  // then continue as interpreted.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2535
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2536
  // IF you lose the race you go interpreted.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2537
  // We don't see any possible endless c2i -> i2c -> c2i ...
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2538
  // transitions no matter how rare.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2539
  __ bind(patch_callsite);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2540
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2541
  RegisterSaver::save_live_registers(masm, RegisterSaver::arg_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2542
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, SharedRuntime::fixup_callers_callsite), Z_method, Z_R14);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2543
  RegisterSaver::restore_live_registers(masm, RegisterSaver::arg_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2544
  __ z_bru(skip_fixup);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2545
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2546
  // end of out-of-line code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2547
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2548
  return c2i_entrypoint;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2549
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2550
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2551
// On entry, the following registers are set
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2552
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2553
//    Z_thread  r8  - JavaThread*
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2554
//    Z_method  r9  - callee's method (method to be invoked)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2555
//    Z_esp     r7  - operand (or expression) stack pointer of caller. one slot above last arg.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2556
//    Z_SP      r15 - SP prepared by call stub such that caller's outgoing args are near top
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2557
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2558
void SharedRuntime::gen_i2c_adapter(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2559
                                    int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2560
                                    int comp_args_on_stack,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2561
                                    const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2562
                                    const VMRegPair *regs) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2563
  const Register value = Z_R12;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2564
  const Register ld_ptr= Z_esp;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2565
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2566
  int ld_offset = total_args_passed * wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2567
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2568
  // Cut-out for having no stack args.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2569
  if (comp_args_on_stack) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2570
    // Sig words on the stack are greater than VMRegImpl::stack0. Those in
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2571
    // registers are below. By subtracting stack0, we either get a negative
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2572
    // number (all values in registers) or the maximum stack slot accessed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2573
    // Convert VMRegImpl (4 byte) stack slots to words.
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
  2574
    int comp_words_on_stack = align_up(comp_args_on_stack*VMRegImpl::stack_slot_size, wordSize)>>LogBytesPerWord;
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2575
    // Round up to miminum stack alignment, in wordSize
46620
750c6edff33b 8178500: Replace usages of round_to and round_down with align_up and align_down
stefank
parents: 44406
diff changeset
  2576
    comp_words_on_stack = align_up(comp_words_on_stack, 2);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2577
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2578
    __ resize_frame(-comp_words_on_stack*wordSize, Z_R0_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2579
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2580
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2581
  // Now generate the shuffle code. Pick up all register args and move the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2582
  // rest through register value=Z_R12.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2583
  for (int i = 0; i < total_args_passed; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2584
    if (sig_bt[i] == T_VOID) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2585
      assert(i > 0 && (sig_bt[i-1] == T_LONG || sig_bt[i-1] == T_DOUBLE), "missing half");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2586
      continue;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2587
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2588
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2589
    // Pick up 0, 1 or 2 words from ld_ptr.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2590
    assert(!regs[i].second()->is_valid() || regs[i].first()->next() == regs[i].second(),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2591
           "scrambled load targets?");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2592
    VMReg r_1 = regs[i].first();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2593
    VMReg r_2 = regs[i].second();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2594
    if (!r_1->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2595
      assert(!r_2->is_valid(), "");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2596
      continue;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2597
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2598
    if (r_1->is_FloatRegister()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2599
      if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2600
        __ z_le(r_1->as_FloatRegister(), ld_offset, ld_ptr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2601
        ld_offset-=wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2602
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2603
        // Skip the unused interpreter slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2604
        __ z_ld(r_1->as_FloatRegister(), ld_offset - wordSize, ld_ptr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2605
        ld_offset -= 2 * wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2606
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2607
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2608
      if (r_1->is_stack()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2609
        // Must do a memory to memory move.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2610
        int st_off = (r_1->reg2stack() + SharedRuntime::out_preserve_stack_slots()) * VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2611
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2612
        if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2613
          __ z_mvc(Address(Z_SP, st_off), Address(ld_ptr, ld_offset), sizeof(void*));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2614
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2615
          // In 64bit, longs are given 2 64-bit slots in the interpreter, but the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2616
          // data is passed in only 1 slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2617
          if (sig_bt[i] == T_LONG || sig_bt[i] == T_DOUBLE) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2618
            ld_offset -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2619
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2620
          __ z_mvc(Address(Z_SP, st_off), Address(ld_ptr, ld_offset), sizeof(void*));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2621
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2622
      } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2623
        if (!r_2->is_valid()) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2624
          // Not sure we need to do this but it shouldn't hurt.
58356
feff88c68082 8231448: s390 and ppc - replace JVM type comparisons to T_OBJECT and T_ARRAY with call to is_reference_type
mbaesken
parents: 57710
diff changeset
  2625
          if (is_reference_type(sig_bt[i]) || sig_bt[i] == T_ADDRESS) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2626
            __ z_lg(r_1->as_Register(), ld_offset, ld_ptr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2627
          } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2628
            __ z_l(r_1->as_Register(), ld_offset, ld_ptr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2629
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2630
        } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2631
          // In 64bit, longs are given 2 64-bit slots in the interpreter, but the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2632
          // data is passed in only 1 slot.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2633
          if (sig_bt[i] == T_LONG || sig_bt[i] == T_DOUBLE) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2634
            ld_offset -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2635
          }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2636
          __ z_lg(r_1->as_Register(), ld_offset, ld_ptr);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2637
        }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2638
      }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2639
      ld_offset -= wordSize;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2640
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2641
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2642
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2643
  // Jump to the compiled code just as if compiled code was doing it.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2644
  // load target address from method oop:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2645
  __ z_lg(Z_R1_scratch, Address(Z_method, Method::from_compiled_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2646
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2647
  // Store method oop into thread->callee_target.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2648
  // 6243940: We might end up in handle_wrong_method if
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2649
  // the callee is deoptimized as we race thru here. If that
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2650
  // happens we don't want to take a safepoint because the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2651
  // caller frame will look interpreted and arguments are now
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2652
  // "compiled" so it is much better to make this transition
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2653
  // invisible to the stack walking code. Unfortunately, if
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2654
  // we try and find the callee by normal means a safepoint
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2655
  // is possible. So we stash the desired callee in the thread
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2656
  // and the vm will find it there should this case occur.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2657
  __ z_stg(Z_method, thread_(callee_target));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2658
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2659
  __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2660
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2661
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2662
AdapterHandlerEntry* SharedRuntime::generate_i2c2i_adapters(MacroAssembler *masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2663
                                                            int total_args_passed,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2664
                                                            int comp_args_on_stack,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2665
                                                            const BasicType *sig_bt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2666
                                                            const VMRegPair *regs,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2667
                                                            AdapterFingerPrint* fingerprint) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2668
  __ align(CodeEntryAlignment);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2669
  address i2c_entry = __ pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2670
  gen_i2c_adapter(masm, total_args_passed, comp_args_on_stack, sig_bt, regs);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2671
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2672
  address c2i_unverified_entry;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2673
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2674
  Label skip_fixup;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2675
  {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2676
    Label ic_miss;
48557
2e867226b914 8174962: Better interface invocations
vlivanov
parents: 48332
diff changeset
  2677
    const int klass_offset           = oopDesc::klass_offset_in_bytes();
2e867226b914 8174962: Better interface invocations
vlivanov
parents: 48332
diff changeset
  2678
    const int holder_klass_offset    = CompiledICHolder::holder_klass_offset();
2e867226b914 8174962: Better interface invocations
vlivanov
parents: 48332
diff changeset
  2679
    const int holder_metadata_offset = CompiledICHolder::holder_metadata_offset();
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2680
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2681
    // Out-of-line call to ic_miss handler.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2682
    __ call_ic_miss_handler(ic_miss, 0x11, 0, Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2683
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2684
    // Unverified Entry Point UEP
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2685
    __ align(CodeEntryAlignment);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2686
    c2i_unverified_entry = __ pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2687
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2688
    // Check the pointers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2689
    if (!ImplicitNullChecks || MacroAssembler::needs_explicit_null_check(klass_offset)) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2690
      __ z_ltgr(Z_ARG1, Z_ARG1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2691
      __ z_bre(ic_miss);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2692
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2693
    __ verify_oop(Z_ARG1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2694
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2695
    // Check ic: object class <-> cached class
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2696
    // Compress cached class for comparison. That's more efficient.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2697
    if (UseCompressedClassPointers) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2698
      __ z_lg(Z_R11, holder_klass_offset, Z_method);             // Z_R11 is overwritten a few instructions down anyway.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2699
      __ compare_klass_ptr(Z_R11, klass_offset, Z_ARG1, false); // Cached class can't be zero.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2700
    } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2701
      __ z_clc(klass_offset, sizeof(void *)-1, Z_ARG1, holder_klass_offset, Z_method);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2702
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2703
    __ z_brne(ic_miss);  // Cache miss: call runtime to handle this.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2704
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2705
    // This def MUST MATCH code in gen_c2i_adapter!
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2706
    const Register code = Z_R11;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2707
48557
2e867226b914 8174962: Better interface invocations
vlivanov
parents: 48332
diff changeset
  2708
    __ z_lg(Z_method, holder_metadata_offset, Z_method);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2709
    __ load_and_test_long(Z_R0, method_(code));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2710
    __ z_brne(ic_miss);  // Cache miss: call runtime to handle this.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2711
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2712
    // Fallthru to VEP. Duplicate LTG, but saved taken branch.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2713
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2714
55343
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2715
  address c2i_entry = __ pc();
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2716
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2717
  // Class initialization barrier for static methods
55749
cff8aad2593f 8227260: JNI upcalls should bypass class initialization barrier in c2i adapter
vlivanov
parents: 55343
diff changeset
  2718
  address c2i_no_clinit_check_entry = NULL;
55343
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2719
  if (VM_Version::supports_fast_class_init_checks()) {
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2720
    Label L_skip_barrier;
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2721
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2722
    { // Bypass the barrier for non-static methods
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2723
      __ testbit(Address(Z_method, Method::access_flags_offset()), JVM_ACC_STATIC_BIT);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2724
      __ z_bfalse(L_skip_barrier); // non-static
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2725
    }
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2726
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2727
    Register klass = Z_R11;
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2728
    __ load_method_holder(klass, Z_method);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2729
    __ clinit_barrier(klass, Z_thread, &L_skip_barrier /*L_fast_path*/);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2730
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2731
    __ load_const_optimized(klass, SharedRuntime::get_handle_wrong_method_stub());
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2732
    __ z_br(klass);
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2733
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2734
    __ bind(L_skip_barrier);
55749
cff8aad2593f 8227260: JNI upcalls should bypass class initialization barrier in c2i adapter
vlivanov
parents: 55343
diff changeset
  2735
    c2i_no_clinit_check_entry = __ pc();
55343
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2736
  }
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2737
03d417fd7d9a 8224827: Implement fast class initialization checks on s390
mdoerr
parents: 54847
diff changeset
  2738
  gen_c2i_adapter(masm, total_args_passed, comp_args_on_stack, sig_bt, regs, skip_fixup);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2739
55749
cff8aad2593f 8227260: JNI upcalls should bypass class initialization barrier in c2i adapter
vlivanov
parents: 55343
diff changeset
  2740
  return AdapterHandlerLibrary::new_entry(fingerprint, i2c_entry, c2i_entry, c2i_unverified_entry, c2i_no_clinit_check_entry);
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2741
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2742
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2743
// This function returns the adjust size (in number of words) to a c2i adapter
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2744
// activation for use during deoptimization.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2745
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2746
// Actually only compiled frames need to be adjusted, but it
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2747
// doesn't harm to adjust entry and interpreter frames, too.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2748
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2749
int Deoptimization::last_frame_adjust(int callee_parameters, int callee_locals) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2750
  assert(callee_locals >= callee_parameters,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2751
          "test and remove; got more parms than locals");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2752
  // Handle the abi adjustment here instead of doing it in push_skeleton_frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2753
  return (callee_locals - callee_parameters) * Interpreter::stackElementWords +
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2754
         frame::z_parent_ijava_frame_abi_size / BytesPerWord;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2755
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2756
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2757
uint SharedRuntime::out_preserve_stack_slots() {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2758
  return frame::z_jit_out_preserve_size/VMRegImpl::stack_slot_size;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2759
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2760
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2761
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2762
// Frame generation for deopt and uncommon trap blobs.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2763
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2764
static void push_skeleton_frame(MacroAssembler* masm,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2765
                          /* Unchanged */
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2766
                          Register frame_sizes_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2767
                          Register pcs_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2768
                          /* Invalidate */
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2769
                          Register frame_size_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2770
                          Register pc_reg) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2771
  BLOCK_COMMENT("  push_skeleton_frame {");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2772
   __ z_lg(pc_reg, 0, pcs_reg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2773
   __ z_lg(frame_size_reg, 0, frame_sizes_reg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2774
   __ z_stg(pc_reg, _z_abi(return_pc), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2775
   Register fp = pc_reg;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2776
   __ push_frame(frame_size_reg, fp);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2777
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2778
   // The magic is required for successful walking skeletal frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2779
   __ load_const_optimized(frame_size_reg/*tmp*/, frame::z_istate_magic_number);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2780
   __ z_stg(frame_size_reg, _z_ijava_state_neg(magic), fp);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2781
   // Fill other slots that are supposedly not necessary with eye catchers.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2782
   __ load_const_optimized(frame_size_reg/*use as tmp*/, 0xdeadbad1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2783
   __ z_stg(frame_size_reg, _z_ijava_state_neg(top_frame_sp), fp);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2784
   // The sender_sp of the bottom frame is set before pushing it.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2785
   // The sender_sp of non bottom frames is their caller's top_frame_sp, which
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2786
   // is unknown here. Luckily it is not needed before filling the frame in
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2787
   // layout_activation(), we assert this by setting an eye catcher (see
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2788
   // comments on sender_sp in frame_s390.hpp).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2789
   __ z_stg(frame_size_reg, _z_ijava_state_neg(sender_sp), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2790
#endif // ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2791
  BLOCK_COMMENT("  } push_skeleton_frame");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2792
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2793
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2794
// Loop through the UnrollBlock info and create new frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2795
static void push_skeleton_frames(MacroAssembler* masm, bool deopt,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2796
                            /* read */
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2797
                            Register unroll_block_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2798
                            /* invalidate */
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2799
                            Register frame_sizes_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2800
                            Register number_of_frames_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2801
                            Register pcs_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2802
                            Register tmp1,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2803
                            Register tmp2) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2804
  BLOCK_COMMENT("push_skeleton_frames {");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2805
  // _number_of_frames is of type int (deoptimization.hpp).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2806
  __ z_lgf(number_of_frames_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2807
           Address(unroll_block_reg, Deoptimization::UnrollBlock::number_of_frames_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2808
  __ z_lg(pcs_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2809
          Address(unroll_block_reg, Deoptimization::UnrollBlock::frame_pcs_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2810
  __ z_lg(frame_sizes_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2811
          Address(unroll_block_reg, Deoptimization::UnrollBlock::frame_sizes_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2812
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2813
  // stack: (caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2814
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2815
  // If caller_of_deoptee is a compiled frame, then we extend it to make
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2816
  // room for the callee's locals and the frame::z_parent_ijava_frame_abi.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2817
  // See also Deoptimization::last_frame_adjust() above.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2818
  // Note: entry and interpreted frames are adjusted, too. But this doesn't harm.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2819
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2820
  __ z_lgf(Z_R1_scratch,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2821
           Address(unroll_block_reg, Deoptimization::UnrollBlock::caller_adjustment_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2822
  __ z_lgr(tmp1, Z_SP);  // Save the sender sp before extending the frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2823
  __ resize_frame_sub(Z_R1_scratch, tmp2/*tmp*/);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2824
  // The oldest skeletal frame requires a valid sender_sp to make it walkable
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2825
  // (it is required to find the original pc of caller_of_deoptee if it is marked
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2826
  // for deoptimization - see nmethod::orig_pc_addr()).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2827
  __ z_stg(tmp1, _z_ijava_state_neg(sender_sp), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2828
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2829
  // Now push the new interpreter frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2830
  Label loop, loop_entry;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2831
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2832
  // Make sure that there is at least one entry in the array.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2833
  DEBUG_ONLY(__ z_ltgr(number_of_frames_reg, number_of_frames_reg));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2834
  __ asm_assert_ne("array_size must be > 0", 0x205);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2835
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2836
  __ z_bru(loop_entry);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2837
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2838
  __ bind(loop);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2839
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2840
  __ add2reg(frame_sizes_reg, wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2841
  __ add2reg(pcs_reg, wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2842
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2843
  __ bind(loop_entry);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2844
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2845
  // Allocate a new frame, fill in the pc.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2846
  push_skeleton_frame(masm, frame_sizes_reg, pcs_reg, tmp1, tmp2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2847
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2848
  __ z_aghi(number_of_frames_reg, -1);  // Emit AGHI, because it sets the condition code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2849
  __ z_brne(loop);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2850
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2851
  // Set the top frame's return pc.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2852
  __ add2reg(pcs_reg, wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2853
  __ z_lg(Z_R0_scratch, 0, pcs_reg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2854
  __ z_stg(Z_R0_scratch, _z_abi(return_pc), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2855
  BLOCK_COMMENT("} push_skeleton_frames");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2856
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2857
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2858
//------------------------------generate_deopt_blob----------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2859
void SharedRuntime::generate_deopt_blob() {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2860
  // Allocate space for the code.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2861
  ResourceMark rm;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2862
  // Setup code generation tools.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2863
  CodeBuffer buffer("deopt_blob", 2048, 1024);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2864
  InterpreterMacroAssembler* masm = new InterpreterMacroAssembler(&buffer);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2865
  Label exec_mode_initialized;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2866
  OopMap* map = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2867
  OopMapSet *oop_maps = new OopMapSet();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2868
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2869
  unsigned int start_off = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2870
  Label cont;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2871
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2872
  // --------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2873
  // Normal entry (non-exception case)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2874
  //
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2875
  // We have been called from the deopt handler of the deoptee.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2876
  // Z_R14 points behind the call in the deopt handler. We adjust
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2877
  // it such that it points to the start of the deopt handler.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2878
  // The return_pc has been stored in the frame of the deoptee and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2879
  // will replace the address of the deopt_handler in the call
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2880
  // to Deoptimization::fetch_unroll_info below.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2881
  // The (int) cast is necessary, because -((unsigned int)14)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2882
  // is an unsigned int.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2883
  __ add2reg(Z_R14, -(int)HandlerImpl::size_deopt_handler());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2884
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2885
  const Register   exec_mode_reg = Z_tmp_1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2886
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2887
  // stack: (deoptee, caller of deoptee, ...)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2888
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2889
  // pushes an "unpack" frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2890
  // R14 contains the return address pointing into the deoptimized
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2891
  // nmethod that was valid just before the nmethod was deoptimized.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2892
  // save R14 into the deoptee frame.  the `fetch_unroll_info'
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2893
  // procedure called below will read it from there.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2894
  map = RegisterSaver::save_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2895
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2896
  // note the entry point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2897
  __ load_const_optimized(exec_mode_reg, Deoptimization::Unpack_deopt);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2898
  __ z_bru(exec_mode_initialized);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2899
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2900
#ifndef COMPILER1
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2901
  int reexecute_offset = 1; // odd offset will produce odd pc, which triggers an hardware trap
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2902
#else
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2903
  // --------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2904
  // Reexecute entry
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2905
  // - Z_R14 = Deopt Handler in nmethod
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2906
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2907
  int reexecute_offset = __ offset() - start_off;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2908
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2909
  // No need to update map as each call to save_live_registers will produce identical oopmap
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2910
  (void) RegisterSaver::save_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2911
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2912
  __ load_const_optimized(exec_mode_reg, Deoptimization::Unpack_reexecute);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2913
  __ z_bru(exec_mode_initialized);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2914
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2915
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2916
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2917
  // --------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2918
  // Exception entry. We reached here via a branch. Registers on entry:
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2919
  // - Z_EXC_OOP (Z_ARG1) = exception oop
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2920
  // - Z_EXC_PC  (Z_ARG2) = the exception pc.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2921
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2922
  int exception_offset = __ offset() - start_off;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2923
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2924
  // all registers are dead at this entry point, except for Z_EXC_OOP, and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2925
  // Z_EXC_PC which contain the exception oop and exception pc
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2926
  // respectively.  Set them in TLS and fall thru to the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2927
  // unpack_with_exception_in_tls entry point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2928
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2929
  // Store exception oop and pc in thread (location known to GC).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2930
  // Need this since the call to "fetch_unroll_info()" may safepoint.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2931
  __ z_stg(Z_EXC_OOP, Address(Z_thread, JavaThread::exception_oop_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2932
  __ z_stg(Z_EXC_PC,  Address(Z_thread, JavaThread::exception_pc_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2933
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2934
  // fall through
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2935
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2936
  int exception_in_tls_offset = __ offset() - start_off;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2937
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2938
  // new implementation because exception oop is now passed in JavaThread
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2939
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2940
  // Prolog for exception case
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2941
  // All registers must be preserved because they might be used by LinearScan
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2942
  // Exceptiop oop and throwing PC are passed in JavaThread
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2943
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2944
  // load throwing pc from JavaThread and us it as the return address of the current frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2945
  __ z_lg(Z_R1_scratch, Address(Z_thread, JavaThread::exception_pc_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2946
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2947
  // Save everything in sight.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2948
  (void) RegisterSaver::save_live_registers(masm, RegisterSaver::all_registers, Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2949
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2950
  // Now it is safe to overwrite any register
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2951
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2952
  // Clear the exception pc field in JavaThread
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2953
  __ clear_mem(Address(Z_thread, JavaThread::exception_pc_offset()), 8);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2954
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2955
  // Deopt during an exception.  Save exec mode for unpack_frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2956
  __ load_const_optimized(exec_mode_reg, Deoptimization::Unpack_exception);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2957
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2958
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2959
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2960
  // verify that there is really an exception oop in JavaThread
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2961
  __ z_lg(Z_ARG1, Address(Z_thread, JavaThread::exception_oop_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2962
  __ verify_oop(Z_ARG1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2963
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2964
  // verify that there is no pending exception
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2965
  __ asm_assert_mem8_is_zero(in_bytes(Thread::pending_exception_offset()), Z_thread,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2966
                             "must not have pending exception here", __LINE__);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2967
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2968
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2969
  // --------------------------------------------------------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2970
  // At this point, the live registers are saved and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2971
  // the exec_mode_reg has been set up correctly.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2972
  __ bind(exec_mode_initialized);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2973
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2974
  // stack: ("unpack" frame, deoptee, caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2975
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2976
  {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2977
  const Register unroll_block_reg  = Z_tmp_2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2978
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2979
  // we need to set `last_Java_frame' because `fetch_unroll_info' will
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2980
  // call `last_Java_frame()'.  however we can't block and no gc will
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2981
  // occur so we don't need an oopmap. the value of the pc in the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2982
  // frame is not particularly important.  it just needs to identify the blob.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2983
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2984
  // Don't set last_Java_pc anymore here (is implicitly NULL then).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2985
  // the correct PC is retrieved in pd_last_frame() in that case.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2986
  __ set_last_Java_frame(/*sp*/Z_SP, noreg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2987
  // With EscapeAnalysis turned on, this call may safepoint
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2988
  // despite it's marked as "leaf call"!
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2989
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, Deoptimization::fetch_unroll_info), Z_thread, exec_mode_reg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2990
  // Set an oopmap for the call site this describes all our saved volatile registers
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2991
  int offs = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2992
  oop_maps->add_gc_map(offs, map);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2993
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2994
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2995
  // save the return value.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2996
  __ z_lgr(unroll_block_reg, Z_RET);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2997
  // restore the return registers that have been saved
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2998
  // (among other registers) by save_live_registers(...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  2999
  RegisterSaver::restore_result_registers(masm);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3000
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3001
  // reload the exec mode from the UnrollBlock (it might have changed)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3002
  __ z_llgf(exec_mode_reg, Address(unroll_block_reg, Deoptimization::UnrollBlock::unpack_kind_offset_in_bytes()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3003
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3004
  // In excp_deopt_mode, restore and clear exception oop which we
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3005
  // stored in the thread during exception entry above. The exception
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3006
  // oop will be the return value of this stub.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3007
  NearLabel skip_restore_excp;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3008
  __ compare64_and_branch(exec_mode_reg, Deoptimization::Unpack_exception, Assembler::bcondNotEqual, skip_restore_excp);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3009
  __ z_lg(Z_RET, thread_(exception_oop));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3010
  __ clear_mem(thread_(exception_oop), 8);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3011
  __ bind(skip_restore_excp);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3012
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3013
  // remove the "unpack" frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3014
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3015
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3016
  // stack: (deoptee, caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3017
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3018
  // pop the deoptee's frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3019
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3020
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3021
  // stack: (caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3022
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3023
  // loop through the `UnrollBlock' info and create interpreter frames.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3024
  push_skeleton_frames(masm, true/*deopt*/,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3025
                  unroll_block_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3026
                  Z_tmp_3,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3027
                  Z_tmp_4,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3028
                  Z_ARG5,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3029
                  Z_ARG4,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3030
                  Z_ARG3);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3031
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3032
  // stack: (skeletal interpreter frame, ..., optional skeletal
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3033
  // interpreter frame, caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3034
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3035
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3036
  // push an "unpack" frame taking care of float / int return values.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3037
  __ push_frame(RegisterSaver::live_reg_frame_size(RegisterSaver::all_registers));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3038
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3039
  // stack: (unpack frame, skeletal interpreter frame, ..., optional
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3040
  // skeletal interpreter frame, caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3041
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3042
  // spill live volatile registers since we'll do a call.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3043
  __ z_stg(Z_RET, offset_of(frame::z_abi_160_spill, spill[0]), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3044
  __ z_std(Z_FRET, offset_of(frame::z_abi_160_spill, spill[1]), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3045
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3046
  // let the unpacker layout information in the skeletal frames just allocated.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3047
  __ get_PC(Z_RET);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3048
  __ set_last_Java_frame(/*sp*/Z_SP, /*pc*/Z_RET);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3049
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, Deoptimization::unpack_frames),
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3050
                  Z_thread/*thread*/, exec_mode_reg/*exec_mode*/);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3051
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3052
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3053
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3054
  // restore the volatiles saved above.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3055
  __ z_lg(Z_RET, offset_of(frame::z_abi_160_spill, spill[0]), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3056
  __ z_ld(Z_FRET, offset_of(frame::z_abi_160_spill, spill[1]), Z_SP);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3057
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3058
  // pop the "unpack" frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3059
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3060
  __ restore_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3061
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3062
  // stack: (top interpreter frame, ..., optional interpreter frame,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3063
  // caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3064
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3065
  __ z_lg(Z_fp, _z_abi(callers_sp), Z_SP); // restore frame pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3066
  __ restore_bcp();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3067
  __ restore_locals();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3068
  __ restore_esp();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3069
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3070
  // return to the interpreter entry point.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3071
  __ z_br(Z_R14);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3072
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3073
  // Make sure all code is generated
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3074
  masm->flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3075
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3076
  _deopt_blob = DeoptimizationBlob::create(&buffer, oop_maps, 0, exception_offset, reexecute_offset, RegisterSaver::live_reg_frame_size(RegisterSaver::all_registers)/wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3077
  _deopt_blob->set_unpack_with_exception_in_tls_offset(exception_in_tls_offset);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3078
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3079
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3080
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3081
#ifdef COMPILER2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3082
//------------------------------generate_uncommon_trap_blob--------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3083
void SharedRuntime::generate_uncommon_trap_blob() {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3084
  // Allocate space for the code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3085
  ResourceMark rm;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3086
  // Setup code generation tools
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3087
  CodeBuffer buffer("uncommon_trap_blob", 2048, 1024);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3088
  InterpreterMacroAssembler* masm = new InterpreterMacroAssembler(&buffer);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3089
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3090
  Register unroll_block_reg = Z_tmp_1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3091
  Register klass_index_reg  = Z_ARG2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3092
  Register unc_trap_reg     = Z_ARG2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3093
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3094
  // stack: (deoptee, caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3095
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3096
  // push a dummy "unpack" frame and call
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3097
  // `Deoptimization::uncommon_trap' to pack the compiled frame into a
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3098
  // vframe array and return the `UnrollBlock' information.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3099
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3100
  // save R14 to compiled frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3101
  __ save_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3102
  // push the "unpack_frame".
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3103
  __ push_frame_abi160(0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3104
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3105
  // stack: (unpack frame, deoptee, caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3106
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3107
  // set the "unpack" frame as last_Java_frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3108
  // `Deoptimization::uncommon_trap' expects it and considers its
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3109
  // sender frame as the deoptee frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3110
  __ get_PC(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3111
  __ set_last_Java_frame(/*sp*/Z_SP, /*pc*/Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3112
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3113
  __ z_lgr(klass_index_reg, Z_ARG1);  // passed implicitly as ARG2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3114
  __ z_lghi(Z_ARG3, Deoptimization::Unpack_uncommon_trap);  // passed implicitly as ARG3
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3115
  BLOCK_COMMENT("call Deoptimization::uncommon_trap()");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3116
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, Deoptimization::uncommon_trap), Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3117
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3118
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3119
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3120
  // pop the "unpack" frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3121
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3122
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3123
  // stack: (deoptee, caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3124
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3125
  // save the return value.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3126
  __ z_lgr(unroll_block_reg, Z_RET);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3127
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3128
  // pop the deoptee frame.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3129
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3130
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3131
  // stack: (caller_of_deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3132
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3133
#ifdef ASSERT
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3134
  assert(Immediate::is_uimm8(Deoptimization::Unpack_LIMIT), "Code not fit for larger immediates");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3135
  assert(Immediate::is_uimm8(Deoptimization::Unpack_uncommon_trap), "Code not fit for larger immediates");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3136
  const int unpack_kind_byte_offset = Deoptimization::UnrollBlock::unpack_kind_offset_in_bytes()
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3137
#ifndef VM_LITTLE_ENDIAN
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3138
  + 3
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3139
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3140
  ;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3141
  if (Displacement::is_shortDisp(unpack_kind_byte_offset)) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3142
    __ z_cli(unpack_kind_byte_offset, unroll_block_reg, Deoptimization::Unpack_uncommon_trap);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3143
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3144
    __ z_cliy(unpack_kind_byte_offset, unroll_block_reg, Deoptimization::Unpack_uncommon_trap);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3145
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3146
  __ asm_assert_eq("SharedRuntime::generate_deopt_blob: expected Unpack_uncommon_trap", 0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3147
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3148
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3149
  __ zap_from_to(Z_SP, Z_SP, Z_R0_scratch, Z_R1, 500, -1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3150
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3151
  // allocate new interpreter frame(s) and possibly resize the caller's frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3152
  // (no more adapters !)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3153
  push_skeleton_frames(masm, false/*deopt*/,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3154
                  unroll_block_reg,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3155
                  Z_tmp_2,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3156
                  Z_tmp_3,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3157
                  Z_tmp_4,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3158
                  Z_ARG5,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3159
                  Z_ARG4);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3160
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3161
  // stack: (skeletal interpreter frame, ..., optional skeletal
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3162
  // interpreter frame, (resized) caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3163
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3164
  // push a dummy "unpack" frame taking care of float return values.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3165
  // call `Deoptimization::unpack_frames' to layout information in the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3166
  // interpreter frames just created
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3167
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3168
  // push the "unpack" frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3169
   const unsigned int framesize_in_bytes = __ push_frame_abi160(0);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3170
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3171
  // stack: (unpack frame, skeletal interpreter frame, ..., optional
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3172
  // skeletal interpreter frame, (resized) caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3173
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3174
  // set the "unpack" frame as last_Java_frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3175
  __ get_PC(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3176
  __ set_last_Java_frame(/*sp*/Z_SP, /*pc*/Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3177
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3178
  // indicate it is the uncommon trap case
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3179
  BLOCK_COMMENT("call Deoptimization::Unpack_uncommon_trap()");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3180
  __ load_const_optimized(unc_trap_reg, Deoptimization::Unpack_uncommon_trap);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3181
  // let the unpacker layout information in the skeletal frames just allocated.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3182
  __ call_VM_leaf(CAST_FROM_FN_PTR(address, Deoptimization::unpack_frames), Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3183
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3184
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3185
  // pop the "unpack" frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3186
  __ pop_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3187
  // restore LR from top interpreter frame
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3188
  __ restore_return_pc();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3189
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3190
  // stack: (top interpreter frame, ..., optional interpreter frame,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3191
  // (resized) caller of deoptee, ...).
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3192
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3193
  __ z_lg(Z_fp, _z_abi(callers_sp), Z_SP); // restore frame pointer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3194
  __ restore_bcp();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3195
  __ restore_locals();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3196
  __ restore_esp();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3197
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3198
  // return to the interpreter entry point
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3199
  __ z_br(Z_R14);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3200
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3201
  masm->flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3202
  _uncommon_trap_blob = UncommonTrapBlob::create(&buffer, NULL, framesize_in_bytes/wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3203
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3204
#endif // COMPILER2
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3205
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3206
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3207
//------------------------------generate_handler_blob------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3208
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3209
// Generate a special Compile2Runtime blob that saves all registers,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3210
// and setup oopmap.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3211
SafepointBlob* SharedRuntime::generate_handler_blob(address call_ptr, int poll_type) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3212
  assert(StubRoutines::forward_exception_entry() != NULL,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3213
         "must be generated before");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3214
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3215
  ResourceMark rm;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3216
  OopMapSet *oop_maps = new OopMapSet();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3217
  OopMap* map;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3218
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3219
  // Allocate space for the code. Setup code generation tools.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3220
  CodeBuffer buffer("handler_blob", 2048, 1024);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3221
  MacroAssembler* masm = new MacroAssembler(&buffer);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3222
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3223
  unsigned int start_off = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3224
  address call_pc = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3225
  int frame_size_in_bytes;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3226
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3227
  bool cause_return = (poll_type == POLL_AT_RETURN);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3228
  // Make room for return address (or push it again)
48332
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3229
  if (!cause_return) {
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3230
    __ z_lg(Z_R14, Address(Z_thread, JavaThread::saved_exception_pc_offset()));
48332
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3231
  }
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3232
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3233
  // Save registers, fpu state, and flags
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3234
  map = RegisterSaver::save_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3235
48332
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3236
  if (SafepointMechanism::uses_thread_local_poll() && !cause_return) {
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3237
    // Keep a copy of the return pc to detect if it gets modified.
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3238
    __ z_lgr(Z_R6, Z_R14);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3239
  }
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3240
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3241
  // The following is basically a call_VM. However, we need the precise
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3242
  // address of the call in order to generate an oopmap. Hence, we do all the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3243
  // work outselves.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3244
  __ set_last_Java_frame(Z_SP, noreg);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3245
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3246
  // call into the runtime to handle the safepoint poll
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3247
  __ call_VM_leaf(call_ptr, Z_thread);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3248
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3249
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3250
  // Set an oopmap for the call site. This oopmap will map all
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3251
  // oop-registers and debug-info registers as callee-saved. This
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3252
  // will allow deoptimization at this safepoint to find all possible
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3253
  // debug-info recordings, as well as let GC find all oops.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3254
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3255
  oop_maps->add_gc_map((int)(__ offset()-start_off), map);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3256
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3257
  Label noException;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3258
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3259
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3260
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3261
  __ load_and_test_long(Z_R1, thread_(pending_exception));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3262
  __ z_bre(noException);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3263
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3264
  // Pending exception case, used (sporadically) by
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3265
  // api/java_lang/Thread.State/index#ThreadState et al.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3266
  RegisterSaver::restore_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3267
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3268
  // Jump to forward_exception_entry, with the issuing PC in Z_R14
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3269
  // so it looks like the original nmethod called forward_exception_entry.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3270
  __ load_const_optimized(Z_R1_scratch, StubRoutines::forward_exception_entry());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3271
  __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3272
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3273
  // No exception case
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3274
  __ bind(noException);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3275
48332
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3276
  if (SafepointMechanism::uses_thread_local_poll() && !cause_return) {
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3277
    Label no_adjust;
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3278
     // If our stashed return pc was modified by the runtime we avoid touching it
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3279
    const int offset_of_return_pc = _z_abi16(return_pc) + RegisterSaver::live_reg_frame_size(RegisterSaver::all_registers);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3280
    __ z_cg(Z_R6, offset_of_return_pc, Z_SP);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3281
    __ z_brne(no_adjust);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3282
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3283
    // Adjust return pc forward to step over the safepoint poll instruction
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3284
    __ instr_size(Z_R1_scratch, Z_R6);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3285
    __ z_agr(Z_R6, Z_R1_scratch);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3286
    __ z_stg(Z_R6, offset_of_return_pc, Z_SP);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3287
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3288
    __ bind(no_adjust);
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3289
  }
651a95f30dfb 8193257: PPC64, s390 implementation for Thread-local handshakes
mdoerr
parents: 47795
diff changeset
  3290
42065
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3291
  // Normal exit, restore registers and exit.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3292
  RegisterSaver::restore_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3293
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3294
  __ z_br(Z_R14);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3295
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3296
  // Make sure all code is generated
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3297
  masm->flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3298
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3299
  // Fill-out other meta info
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3300
  return SafepointBlob::create(&buffer, oop_maps, RegisterSaver::live_reg_frame_size(RegisterSaver::all_registers)/wordSize);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3301
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3302
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3303
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3304
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3305
// generate_resolve_blob - call resolution (static/virtual/opt-virtual/ic-miss
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3306
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3307
// Generate a stub that calls into vm to find out the proper destination
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3308
// of a Java call. All the argument registers are live at this point
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3309
// but since this is generic code we don't know what they are and the caller
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3310
// must do any gc of the args.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3311
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3312
RuntimeStub* SharedRuntime::generate_resolve_blob(address destination, const char* name) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3313
  assert (StubRoutines::forward_exception_entry() != NULL, "must be generated before");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3314
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3315
  // allocate space for the code
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3316
  ResourceMark rm;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3317
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3318
  CodeBuffer buffer(name, 1000, 512);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3319
  MacroAssembler* masm                = new MacroAssembler(&buffer);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3320
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3321
  OopMapSet *oop_maps = new OopMapSet();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3322
  OopMap* map = NULL;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3323
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3324
  unsigned int start_off = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3325
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3326
  map = RegisterSaver::save_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3327
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3328
  // We must save a PC from within the stub as return PC
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3329
  // C code doesn't store the LR where we expect the PC,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3330
  // so we would run into trouble upon stack walking.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3331
  __ get_PC(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3332
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3333
  unsigned int frame_complete = __ offset();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3334
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3335
  __ set_last_Java_frame(/*sp*/Z_SP, Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3336
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3337
  __ call_VM_leaf(destination, Z_thread, Z_method);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3338
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3339
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3340
  // Set an oopmap for the call site.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3341
  // We need this not only for callee-saved registers, but also for volatile
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3342
  // registers that the compiler might be keeping live across a safepoint.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3343
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3344
  oop_maps->add_gc_map((int)(frame_complete-start_off), map);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3345
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3346
  // clear last_Java_sp
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3347
  __ reset_last_Java_frame();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3348
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3349
  // check for pending exceptions
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3350
  Label pending;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3351
  __ load_and_test_long(Z_R0, Address(Z_thread, Thread::pending_exception_offset()));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3352
  __ z_brne(pending);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3353
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3354
  __ z_lgr(Z_R1_scratch, Z_R2); // r1 is neither saved nor restored, r2 contains the continuation.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3355
  RegisterSaver::restore_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3356
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3357
  // get the returned method
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3358
  __ get_vm_result_2(Z_method);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3359
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3360
  // We are back the the original state on entry and ready to go.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3361
  __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3362
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3363
  // Pending exception after the safepoint
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3364
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3365
  __ bind(pending);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3366
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3367
  RegisterSaver::restore_live_registers(masm, RegisterSaver::all_registers);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3368
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3369
  // exception pending => remove activation and forward to exception handler
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3370
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3371
  __ z_lgr(Z_R2, Z_R0); // pending_exception
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3372
  __ clear_mem(Address(Z_thread, JavaThread::vm_result_offset()), sizeof(jlong));
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3373
  __ load_const_optimized(Z_R1_scratch, StubRoutines::forward_exception_entry());
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3374
  __ z_br(Z_R1_scratch);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3375
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3376
  // -------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3377
  // make sure all code is generated
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3378
  masm->flush();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3379
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3380
  // return the blob
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3381
  // frame_size_words or bytes??
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3382
  return RuntimeStub::new_runtime_stub(name, &buffer, frame_complete, RegisterSaver::live_reg_frame_size(RegisterSaver::all_registers)/wordSize,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3383
                                       oop_maps, true);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3384
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3385
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3386
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3387
//------------------------------Montgomery multiplication------------------------
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3388
//
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3389
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3390
// Subtract 0:b from carry:a. Return carry.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3391
static unsigned long
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3392
sub(unsigned long a[], unsigned long b[], unsigned long carry, long len) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3393
  unsigned long i, c = 8 * (unsigned long)(len - 1);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3394
  __asm__ __volatile__ (
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3395
    "SLGR   %[i], %[i]         \n" // initialize to 0 and pre-set carry
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3396
    "LGHI   0, 8               \n" // index increment (for BRXLG)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3397
    "LGR    1, %[c]            \n" // index limit (for BRXLG)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3398
    "0:                        \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3399
    "LG     %[c], 0(%[i],%[a]) \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3400
    "SLBG   %[c], 0(%[i],%[b]) \n" // subtract with borrow
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3401
    "STG    %[c], 0(%[i],%[a]) \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3402
    "BRXLG  %[i], 0, 0b        \n" // while ((i+=8)<limit);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3403
    "SLBGR  %[c], %[c]         \n" // save carry - 1
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3404
    : [i]"=&a"(i), [c]"+r"(c)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3405
    : [a]"a"(a), [b]"a"(b)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3406
    : "cc", "memory", "r0", "r1"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3407
 );
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3408
  return carry + c;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3409
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3410
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3411
// Multiply (unsigned) Long A by Long B, accumulating the double-
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3412
// length result into the accumulator formed of T0, T1, and T2.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3413
inline void MACC(unsigned long A[], long A_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3414
                 unsigned long B[], long B_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3415
                 unsigned long &T0, unsigned long &T1, unsigned long &T2) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3416
  long A_si = 8 * A_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3417
       B_si = 8 * B_ind;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3418
  __asm__ __volatile__ (
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3419
    "LG     1, 0(%[A_si],%[A]) \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3420
    "MLG    0, 0(%[B_si],%[B]) \n" // r0r1 = A * B
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3421
    "ALGR   %[T0], 1           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3422
    "LGHI   1, 0               \n" // r1 = 0
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3423
    "ALCGR  %[T1], 0           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3424
    "ALCGR  %[T2], 1           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3425
    : [T0]"+r"(T0), [T1]"+r"(T1), [T2]"+r"(T2)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3426
    : [A]"r"(A), [A_si]"r"(A_si), [B]"r"(B), [B_si]"r"(B_si)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3427
    : "cc", "r0", "r1"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3428
 );
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3429
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3430
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3431
// As above, but add twice the double-length result into the
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3432
// accumulator.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3433
inline void MACC2(unsigned long A[], long A_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3434
                  unsigned long B[], long B_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3435
                  unsigned long &T0, unsigned long &T1, unsigned long &T2) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3436
  const unsigned long zero = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3437
  long A_si = 8 * A_ind,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3438
       B_si = 8 * B_ind;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3439
  __asm__ __volatile__ (
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3440
    "LG     1, 0(%[A_si],%[A]) \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3441
    "MLG    0, 0(%[B_si],%[B]) \n" // r0r1 = A * B
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3442
    "ALGR   %[T0], 1           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3443
    "ALCGR  %[T1], 0           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3444
    "ALCGR  %[T2], %[zero]     \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3445
    "ALGR   %[T0], 1           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3446
    "ALCGR  %[T1], 0           \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3447
    "ALCGR  %[T2], %[zero]     \n"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3448
    : [T0]"+r"(T0), [T1]"+r"(T1), [T2]"+r"(T2)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3449
    : [A]"r"(A), [A_si]"r"(A_si), [B]"r"(B), [B_si]"r"(B_si), [zero]"r"(zero)
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3450
    : "cc", "r0", "r1"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3451
 );
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3452
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3453
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3454
// Fast Montgomery multiplication. The derivation of the algorithm is
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3455
// in "A Cryptographic Library for the Motorola DSP56000,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3456
// Dusse and Kaliski, Proc. EUROCRYPT 90, pp. 230-237".
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3457
static void
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3458
montgomery_multiply(unsigned long a[], unsigned long b[], unsigned long n[],
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3459
                    unsigned long m[], unsigned long inv, int len) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3460
  unsigned long t0 = 0, t1 = 0, t2 = 0; // Triple-precision accumulator
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3461
  int i;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3462
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3463
  assert(inv * n[0] == -1UL, "broken inverse in Montgomery multiply");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3464
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3465
  for (i = 0; i < len; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3466
    int j;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3467
    for (j = 0; j < i; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3468
      MACC(a, j, b, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3469
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3470
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3471
    MACC(a, i, b, 0, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3472
    m[i] = t0 * inv;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3473
    MACC(m, i, n, 0, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3474
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3475
    assert(t0 == 0, "broken Montgomery multiply");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3476
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3477
    t0 = t1; t1 = t2; t2 = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3478
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3479
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3480
  for (i = len; i < 2 * len; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3481
    int j;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3482
    for (j = i - len + 1; j < len; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3483
      MACC(a, j, b, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3484
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3485
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3486
    m[i-len] = t0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3487
    t0 = t1; t1 = t2; t2 = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3488
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3489
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3490
  while (t0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3491
    t0 = sub(m, n, t0, len);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3492
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3493
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3494
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3495
// Fast Montgomery squaring. This uses asymptotically 25% fewer
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3496
// multiplies so it should be up to 25% faster than Montgomery
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3497
// multiplication. However, its loop control is more complex and it
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3498
// may actually run slower on some machines.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3499
static void
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3500
montgomery_square(unsigned long a[], unsigned long n[],
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3501
                  unsigned long m[], unsigned long inv, int len) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3502
  unsigned long t0 = 0, t1 = 0, t2 = 0; // Triple-precision accumulator
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3503
  int i;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3504
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3505
  assert(inv * n[0] == -1UL, "broken inverse in Montgomery multiply");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3506
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3507
  for (i = 0; i < len; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3508
    int j;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3509
    int end = (i+1)/2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3510
    for (j = 0; j < end; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3511
      MACC2(a, j, a, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3512
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3513
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3514
    if ((i & 1) == 0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3515
      MACC(a, j, a, j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3516
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3517
    for (; j < i; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3518
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3519
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3520
    m[i] = t0 * inv;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3521
    MACC(m, i, n, 0, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3522
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3523
    assert(t0 == 0, "broken Montgomery square");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3524
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3525
    t0 = t1; t1 = t2; t2 = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3526
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3527
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3528
  for (i = len; i < 2*len; i++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3529
    int start = i-len+1;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3530
    int end = start + (len - start)/2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3531
    int j;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3532
    for (j = start; j < end; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3533
      MACC2(a, j, a, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3534
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3535
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3536
    if ((i & 1) == 0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3537
      MACC(a, j, a, j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3538
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3539
    for (; j < len; j++) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3540
      MACC(m, j, n, i-j, t0, t1, t2);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3541
    }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3542
    m[i-len] = t0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3543
    t0 = t1; t1 = t2; t2 = 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3544
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3545
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3546
  while (t0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3547
    t0 = sub(m, n, t0, len);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3548
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3549
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3550
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3551
// The threshold at which squaring is advantageous was determined
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3552
// experimentally on an i7-3930K (Ivy Bridge) CPU @ 3.5GHz.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3553
// Value seems to be ok for other platforms, too.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3554
#define MONTGOMERY_SQUARING_THRESHOLD 64
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3555
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3556
// Copy len longwords from s to d, word-swapping as we go. The
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3557
// destination array is reversed.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3558
static void reverse_words(unsigned long *s, unsigned long *d, int len) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3559
  d += len;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3560
  while(len-- > 0) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3561
    d--;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3562
    unsigned long s_val = *s;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3563
    // Swap words in a longword on little endian machines.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3564
#ifdef VM_LITTLE_ENDIAN
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3565
     Unimplemented();
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3566
#endif
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3567
    *d = s_val;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3568
    s++;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3569
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3570
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3571
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3572
void SharedRuntime::montgomery_multiply(jint *a_ints, jint *b_ints, jint *n_ints,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3573
                                        jint len, jlong inv,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3574
                                        jint *m_ints) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3575
  len = len & 0x7fffFFFF; // C2 does not respect int to long conversion for stub calls.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3576
  assert(len % 2 == 0, "array length in montgomery_multiply must be even");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3577
  int longwords = len/2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3578
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3579
  // Make very sure we don't use so much space that the stack might
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3580
  // overflow. 512 jints corresponds to an 16384-bit integer and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3581
  // will use here a total of 8k bytes of stack space.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3582
  int total_allocation = longwords * sizeof (unsigned long) * 4;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3583
  guarantee(total_allocation <= 8192, "must be");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3584
  unsigned long *scratch = (unsigned long *)alloca(total_allocation);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3585
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3586
  // Local scratch arrays
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3587
  unsigned long
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3588
    *a = scratch + 0 * longwords,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3589
    *b = scratch + 1 * longwords,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3590
    *n = scratch + 2 * longwords,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3591
    *m = scratch + 3 * longwords;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3592
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3593
  reverse_words((unsigned long *)a_ints, a, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3594
  reverse_words((unsigned long *)b_ints, b, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3595
  reverse_words((unsigned long *)n_ints, n, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3596
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3597
  ::montgomery_multiply(a, b, n, m, (unsigned long)inv, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3598
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3599
  reverse_words(m, (unsigned long *)m_ints, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3600
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3601
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3602
void SharedRuntime::montgomery_square(jint *a_ints, jint *n_ints,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3603
                                      jint len, jlong inv,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3604
                                      jint *m_ints) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3605
  len = len & 0x7fffFFFF; // C2 does not respect int to long conversion for stub calls.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3606
  assert(len % 2 == 0, "array length in montgomery_square must be even");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3607
  int longwords = len/2;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3608
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3609
  // Make very sure we don't use so much space that the stack might
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3610
  // overflow. 512 jints corresponds to an 16384-bit integer and
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3611
  // will use here a total of 6k bytes of stack space.
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3612
  int total_allocation = longwords * sizeof (unsigned long) * 3;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3613
  guarantee(total_allocation <= 8192, "must be");
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3614
  unsigned long *scratch = (unsigned long *)alloca(total_allocation);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3615
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3616
  // Local scratch arrays
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3617
  unsigned long
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3618
    *a = scratch + 0 * longwords,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3619
    *n = scratch + 1 * longwords,
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3620
    *m = scratch + 2 * longwords;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3621
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3622
  reverse_words((unsigned long *)a_ints, a, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3623
  reverse_words((unsigned long *)n_ints, n, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3624
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3625
  if (len >= MONTGOMERY_SQUARING_THRESHOLD) {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3626
    ::montgomery_square(a, n, m, (unsigned long)inv, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3627
  } else {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3628
    ::montgomery_multiply(a, a, n, m, (unsigned long)inv, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3629
  }
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3630
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3631
  reverse_words(m, (unsigned long *)m_ints, longwords);
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3632
}
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3633
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3634
extern "C"
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3635
int SpinPause() {
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3636
  return 0;
6032b31e3719 8167673: [s390] The s390 port.
goetz
parents:
diff changeset
  3637
}