SPARCTestAssembler.java revision 10973:b7e54a66a397
1/* 2 * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. 8 * 9 * This code is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 12 * version 2 for more details (a copy is included in the LICENSE file that 13 * accompanied this code). 14 * 15 * You should have received a copy of the GNU General Public License version 16 * 2 along with this work; if not, write to the Free Software Foundation, 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 18 * 19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 20 * or visit www.oracle.com if you need additional information or have any 21 * questions. 22 */ 23 24package jdk.vm.ci.code.test.sparc; 25 26import jdk.vm.ci.code.CodeCacheProvider; 27import jdk.vm.ci.code.DebugInfo; 28import jdk.vm.ci.code.Register; 29import jdk.vm.ci.code.StackSlot; 30import jdk.vm.ci.code.site.ConstantReference; 31import jdk.vm.ci.code.site.DataSectionReference; 32import jdk.vm.ci.code.test.TestAssembler; 33import jdk.vm.ci.hotspot.HotSpotCallingConventionType; 34import jdk.vm.ci.hotspot.HotSpotCompiledCode; 35import jdk.vm.ci.hotspot.HotSpotConstant; 36import jdk.vm.ci.hotspot.HotSpotForeignCallTarget; 37import jdk.vm.ci.hotspot.HotSpotResolvedJavaMethod; 38import jdk.vm.ci.hotspot.HotSpotVMConfig; 39import jdk.vm.ci.meta.JavaKind; 40import jdk.vm.ci.meta.LIRKind; 41import jdk.vm.ci.meta.VMConstant; 42import jdk.vm.ci.sparc.SPARC; 43import jdk.vm.ci.sparc.SPARCKind; 44 45public class SPARCTestAssembler extends TestAssembler { 46 47 private static final int MASK13 = (1 << 13) - 1; 48 49 public SPARCTestAssembler(CodeCacheProvider codeCache) { 50 super(codeCache, 0, 16, SPARCKind.WORD, SPARC.l0, SPARC.l1, SPARC.l2, SPARC.l3, SPARC.l4, SPARC.l5, SPARC.l6, SPARC.l7); 51 } 52 53 private void emitOp2(Register rd, int op2, int imm22) { 54 code.emitInt((0b00 << 30) | (rd.encoding << 25) | (op2 << 22) | imm22); 55 } 56 57 private void emitOp3(int op, Register rd, int op3, Register rs1, Register rs2) { 58 code.emitInt((op << 30) | (rd.encoding << 25) | (op3 << 19) | (rs1.encoding << 14) | rs2.encoding); 59 } 60 61 private void emitOp3(int op, Register rd, int op3, Register rs1, int simm13) { 62 code.emitInt((op << 30) | (rd.encoding << 25) | (op3 << 19) | (rs1.encoding << 14) | (1 << 13) | (simm13 & MASK13)); 63 } 64 65 private void emitNop() { 66 code.emitInt(1 << 24); 67 } 68 69 @Override 70 public void emitPrologue() { 71 // SAVE sp, -128, sp 72 emitOp3(0b10, SPARC.sp, 0b111100, SPARC.sp, -SPARC.REGISTER_SAFE_AREA_SIZE); 73 setDeoptRescueSlot(newStackSlot(LIRKind.value(SPARCKind.XWORD))); 74 } 75 76 @Override 77 public void emitEpilogue() { 78 HotSpotVMConfig config = HotSpotVMConfig.config(); 79 recordMark(config.MARKID_DEOPT_HANDLER_ENTRY); 80 recordCall(new HotSpotForeignCallTarget(config.handleDeoptStub), 4, true, null); 81 code.emitInt(1 << 30); // CALL 82 } 83 84 @Override 85 public HotSpotCompiledCode finish(HotSpotResolvedJavaMethod method) { 86 frameSize += SPARC.REGISTER_SAFE_AREA_SIZE; 87 return super.finish(method); 88 } 89 90 @Override 91 public void emitGrowStack(int size) { 92 emitOp3(0b10, SPARC.sp, 0b000100, SPARC.sp, size); // SUB sp, size, sp 93 } 94 95 @Override 96 public Register emitIntArg0() { 97 return codeCache.getRegisterConfig().getCallingConventionRegisters(HotSpotCallingConventionType.JavaCallee, JavaKind.Int)[0]; 98 } 99 100 @Override 101 public Register emitIntArg1() { 102 return codeCache.getRegisterConfig().getCallingConventionRegisters(HotSpotCallingConventionType.JavaCallee, JavaKind.Int)[1]; 103 } 104 105 @Override 106 public Register emitLoadInt(int c) { 107 Register ret = newRegister(); 108 int hi = c >>> 10; 109 int lo = c & ((1 << 10) - 1); 110 if (hi == 0) { 111 emitOp3(0b10, ret, 0b000010, SPARC.g0, lo); // OR g0, lo, ret 112 } else { 113 emitOp2(ret, 0b100, hi); // SETHI hi, ret 114 if (lo != 0) { 115 emitOp3(0b10, ret, 0b000010, ret, lo); // OR ret, lo, ret 116 } 117 } 118 return ret; 119 } 120 121 @Override 122 public Register emitLoadLong(long c) { 123 if ((c & 0xFFFF_FFFFL) == c) { 124 return emitLoadInt((int) c); 125 } else { 126 DataSectionReference ref = new DataSectionReference(); 127 data.align(8); 128 ref.setOffset(data.position()); 129 data.emitLong(c); 130 return emitLoadPointer(ref); 131 } 132 } 133 134 private void emitPatchableSethi(Register ret, boolean wide) { 135 int startPos = code.position(); 136 emitOp2(ret, 0b100, 0); // SETHI 0, ret 137 if (wide) { 138 // pad for later patching 139 while (code.position() < (startPos + 28)) { 140 emitNop(); 141 } 142 } 143 } 144 145 @Override 146 public Register emitLoadFloat(float c) { 147 DataSectionReference ref = new DataSectionReference(); 148 data.align(4); 149 ref.setOffset(data.position()); 150 data.emitFloat(c); 151 152 Register ptr = newRegister(); 153 recordDataPatchInCode(ref); 154 emitPatchableSethi(ptr, true); 155 emitOp3(0b11, SPARC.f0, 0b100000, ptr, 0); // LDF [ptr+0], f0 156 return SPARC.f0; 157 } 158 159 @Override 160 public Register emitLoadPointer(HotSpotConstant c) { 161 Register ret = newRegister(); 162 recordDataPatchInCode(new ConstantReference((VMConstant) c)); 163 164 emitPatchableSethi(ret, !c.isCompressed()); 165 emitOp3(0b10, ret, 0b000010, ret, 0); // OR ret, 0, ret 166 167 return ret; 168 } 169 170 @Override 171 public Register emitLoadPointer(DataSectionReference ref) { 172 Register ret = newRegister(); 173 recordDataPatchInCode(ref); 174 emitPatchableSethi(ret, true); 175 emitOp3(0b11, ret, 0b001011, ret, 0); // LDX [ret+0], ret 176 return ret; 177 } 178 179 @Override 180 public Register emitLoadNarrowPointer(DataSectionReference ref) { 181 Register ret = newRegister(); 182 recordDataPatchInCode(ref); 183 emitPatchableSethi(ret, true); 184 emitOp3(0b11, ret, 0b000000, ret, 0); // LDUW [ret+0], ret 185 return ret; 186 } 187 188 @Override 189 public Register emitLoadPointer(Register b, int offset) { 190 Register ret = newRegister(); 191 emitOp3(0b11, ret, 0b001011, b, offset); // LDX [b+offset], ret 192 return ret; 193 } 194 195 @Override 196 public StackSlot emitIntToStack(Register a) { 197 StackSlot ret = newStackSlot(LIRKind.value(SPARCKind.WORD)); 198 // STW a, [fp+offset] 199 emitOp3(0b11, a, 0b000100, SPARC.fp, ret.getRawOffset() + SPARC.STACK_BIAS); 200 return ret; 201 } 202 203 @Override 204 public StackSlot emitLongToStack(Register a) { 205 StackSlot ret = newStackSlot(LIRKind.value(SPARCKind.XWORD)); 206 // STX a, [fp+offset] 207 emitOp3(0b11, a, 0b001110, SPARC.fp, ret.getRawOffset() + SPARC.STACK_BIAS); 208 return ret; 209 } 210 211 @Override 212 public StackSlot emitFloatToStack(Register a) { 213 StackSlot ret = newStackSlot(LIRKind.value(SPARCKind.SINGLE)); 214 // STF a, [fp+offset] 215 emitOp3(0b11, a, 0b100100, SPARC.fp, ret.getRawOffset() + SPARC.STACK_BIAS); 216 return ret; 217 } 218 219 @Override 220 public StackSlot emitPointerToStack(Register a) { 221 StackSlot ret = newStackSlot(LIRKind.reference(SPARCKind.XWORD)); 222 // STX a, [fp+offset] 223 emitOp3(0b11, a, 0b001110, SPARC.fp, ret.getRawOffset() + SPARC.STACK_BIAS); 224 return ret; 225 } 226 227 @Override 228 public StackSlot emitNarrowPointerToStack(Register a) { 229 StackSlot ret = newStackSlot(LIRKind.reference(SPARCKind.WORD)); 230 // STW a, [fp+offset] 231 emitOp3(0b11, a, 0b000100, SPARC.fp, ret.getRawOffset() + SPARC.STACK_BIAS); 232 return ret; 233 } 234 235 @Override 236 public Register emitUncompressPointer(Register compressed, long base, int shift) { 237 Register ret; 238 if (shift > 0) { 239 ret = newRegister(); 240 emitOp3(0b10, ret, 0b100101, compressed, shift); // SLL compressed, shift, ret 241 } else { 242 ret = compressed; 243 } 244 if (base == 0) { 245 return ret; 246 } else { 247 Register b = emitLoadLong(base); 248 emitOp3(0b10, b, 0b00000, ret, b); // ADD b, ret, b 249 return b; 250 } 251 } 252 253 @Override 254 public Register emitIntAdd(Register a, Register b) { 255 Register ret = newRegister(); 256 emitOp3(0b10, ret, 0b00000, a, b); // ADD a, b, ret 257 return ret; 258 } 259 260 private void emitMove(Register to, Register from) { 261 if (to != from) { 262 emitOp3(0b10, to, 0b000010, from, SPARC.g0); // OR from, g0, to 263 } 264 } 265 266 @Override 267 public void emitIntRet(Register a) { 268 emitPointerRet(a); 269 } 270 271 @Override 272 public void emitPointerRet(Register a) { 273 emitMove(SPARC.i0, a); 274 emitOp3(0b10, SPARC.g0, 0b111000, SPARC.i7, 8); // JMPL [i7+8], g0 275 emitOp3(0b10, SPARC.g0, 0b111101, SPARC.g0, SPARC.g0); // RESTORE g0, g0, g0 276 } 277 278 @Override 279 public void emitTrap(DebugInfo info) { 280 recordImplicitException(info); 281 emitOp3(0b11, SPARC.g0, 0b001011, SPARC.g0, 0); // LDX [g0+0], g0 282 } 283 284 @Override 285 public DataSectionReference emitDataItem(HotSpotConstant c) { 286 if (c.isCompressed()) { 287 data.align(4); 288 } else { 289 data.align(8); 290 } 291 return super.emitDataItem(c); 292 } 293} 294