1 /*
   2  * Copyright (c) 2020, 2023, Oracle and/or its affiliates. All rights reserved.
   3  * Copyright (c) 2020, 2022, Huawei Technologies Co., Ltd. All rights reserved.
   4  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   5  *
   6  * This code is free software; you can redistribute it and/or modify it
   7  * under the terms of the GNU General Public License version 2 only, as
   8  * published by the Free Software Foundation.
   9  *
  10  * This code is distributed in the hope that it will be useful, but WITHOUT
  11  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  12  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  13  * version 2 for more details (a copy is included in the LICENSE file that
  14  * accompanied this code).
  15  *
  16  * You should have received a copy of the GNU General Public License version
  17  * 2 along with this work; if not, write to the Free Software Foundation,
  18  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  19  *
  20  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  21  * or visit www.oracle.com if you need additional information or have any
  22  * questions.
  23  *
  24  */
  25 
  26 #include "precompiled.hpp"
  27 #include "asm/assembler.hpp"
  28 #include "asm/assembler.inline.hpp"
  29 #include "opto/c2_MacroAssembler.hpp"
  30 #include "opto/compile.hpp"
  31 #include "opto/intrinsicnode.hpp"
  32 #include "opto/output.hpp"
  33 #include "opto/subnode.hpp"
  34 #include "runtime/stubRoutines.hpp"

  35 
  36 #ifdef PRODUCT
  37 #define BLOCK_COMMENT(str) /* nothing */
  38 #define STOP(error) stop(error)
  39 #else
  40 #define BLOCK_COMMENT(str) block_comment(str)
  41 #define STOP(error) block_comment(error); stop(error)
  42 #endif
  43 
  44 #define BIND(label) bind(label); BLOCK_COMMENT(#label ":")
  45 
  46 void C2_MacroAssembler::fast_lock(Register objectReg, Register boxReg,
  47                                   Register tmp1Reg, Register tmp2Reg, Register tmp3Reg) {
  48   // Use cr register to indicate the fast_lock result: zero for success; non-zero for failure.
  49   Register flag = t1;
  50   Register oop = objectReg;
  51   Register box = boxReg;
  52   Register disp_hdr = tmp1Reg;
  53   Register tmp = tmp2Reg;
  54   Label cont;
  55   Label object_has_monitor;
  56   Label count, no_count;



  57 

  58   assert_different_registers(oop, box, tmp, disp_hdr, flag, tmp3Reg, t0);
  59 


  60   // Load markWord from object into displaced_header.
  61   ld(disp_hdr, Address(oop, oopDesc::mark_offset_in_bytes()));
  62 
  63   if (DiagnoseSyncOnValueBasedClasses != 0) {
  64     load_klass(flag, oop);
  65     lwu(flag, Address(flag, Klass::access_flags_offset()));
  66     test_bit(flag, flag, exact_log2(JVM_ACC_IS_VALUE_BASED_CLASS));
  67     bnez(flag, cont, true /* is_far */);
  68   }
  69 
  70   // Check for existing monitor
  71   test_bit(t0, disp_hdr, exact_log2(markWord::monitor_value));
  72   bnez(t0, object_has_monitor);
  73 
  74   if (LockingMode == LM_MONITOR) {
  75     mv(flag, 1); // Set non-zero flag to indicate 'failure' -> take slow-path
  76     j(cont);
  77   } else if (LockingMode == LM_LEGACY) {
  78     // Set tmp to be (markWord of object | UNLOCK_VALUE).
  79     ori(tmp, disp_hdr, markWord::unlocked_value);
  80 
  81     // Initialize the box. (Must happen before we update the object mark!)
  82     sd(tmp, Address(box, BasicLock::displaced_header_offset_in_bytes()));
  83 
  84     // Compare object markWord with an unlocked value (tmp) and if
  85     // equal exchange the stack address of our box with object markWord.
  86     // On failure disp_hdr contains the possibly locked markWord.
  87     cmpxchg(/*memory address*/oop, /*expected value*/tmp, /*new value*/box, Assembler::int64, Assembler::aq,
  88             Assembler::rl, /*result*/disp_hdr);
  89     mv(flag, zr);
  90     beq(disp_hdr, tmp, cont); // prepare zero flag and goto cont if we won the cas
  91 
  92     assert(oopDesc::mark_offset_in_bytes() == 0, "offset of _mark is not 0");
  93 
  94     // If the compare-and-exchange succeeded, then we found an unlocked
  95     // object, will have now locked it will continue at label cont
  96     // We did not see an unlocked object so try the fast recursive case.
  97 
  98     // Check if the owner is self by comparing the value in the
  99     // markWord of object (disp_hdr) with the stack pointer.
 100     sub(disp_hdr, disp_hdr, sp);
 101     mv(tmp, (intptr_t) (~(os::vm_page_size()-1) | (uintptr_t)markWord::lock_mask_in_place));
 102     // If (mark & lock_mask) == 0 and mark - sp < page_size, we are stack-locking and goto cont,
 103     // hence we can store 0 as the displaced header in the box, which indicates that it is a
 104     // recursive lock.
 105     andr(tmp/*==0?*/, disp_hdr, tmp);
 106     sd(tmp/*==0, perhaps*/, Address(box, BasicLock::displaced_header_offset_in_bytes()));
 107     mv(flag, tmp); // we can use the value of tmp as the result here
 108     j(cont);
 109   } else {
 110     assert(LockingMode == LM_LIGHTWEIGHT, "");
 111     Label slow;
 112     lightweight_lock(oop, disp_hdr, tmp, tmp3Reg, slow);
 113 
 114     // Indicate success on completion.
 115     mv(flag, zr);
 116     j(count);
 117     bind(slow);
 118     mv(flag, 1); // Set non-zero flag to indicate 'failure' -> take slow-path
 119     j(no_count);
 120   }
 121 
 122   // Handle existing monitor.
 123   bind(object_has_monitor);
 124   // The object's monitor m is unlocked iff m->owner == NULL,
 125   // otherwise m->owner may contain a thread or a stack address.
 126   //
 127   // Try to CAS m->owner from NULL to current thread.
 128   add(tmp, disp_hdr, (in_bytes(ObjectMonitor::owner_offset()) - markWord::monitor_value));
 129   cmpxchg(/*memory address*/tmp, /*expected value*/zr, /*new value*/xthread, Assembler::int64, Assembler::aq,
 130           Assembler::rl, /*result*/flag); // cas succeeds if flag == zr(expected)
 131 
 132   if (LockingMode != LM_LIGHTWEIGHT) {
 133     // Store a non-null value into the box to avoid looking like a re-entrant
 134     // lock. The fast-path monitor unlock code checks for
 135     // markWord::monitor_value so use markWord::unused_mark which has the
 136     // relevant bit set, and also matches ObjectSynchronizer::slow_enter.
 137     mv(tmp, (address)markWord::unused_mark().value());
 138     sd(tmp, Address(box, BasicLock::displaced_header_offset_in_bytes()));
 139   }
 140 
 141   beqz(flag, cont); // CAS success means locking succeeded





 142 
 143   bne(flag, xthread, cont); // Check for recursive locking


 144 
 145   // Recursive lock case
 146   mv(flag, zr);
 147   increment(Address(disp_hdr, in_bytes(ObjectMonitor::recursions_offset()) - markWord::monitor_value), 1, t0, tmp);
 148 
 149   bind(cont);
 150   // zero flag indicates success
 151   // non-zero flag indicates failure
 152   bnez(flag, no_count);
 153 
 154   bind(count);
 155   increment(Address(xthread, JavaThread::held_monitor_count_offset()), 1, t0, tmp);




 156 
 157   bind(no_count);







 158 }
 159 
 160 void C2_MacroAssembler::fast_unlock(Register objectReg, Register boxReg,
 161                                     Register tmp1Reg, Register tmp2Reg) {
 162   // Use cr register to indicate the fast_unlock result: zero for success; non-zero for failure.
 163   Register flag = t1;
 164   Register oop = objectReg;
 165   Register box = boxReg;
 166   Register disp_hdr = tmp1Reg;
 167   Register tmp = tmp2Reg;
 168   Label cont;
 169   Label object_has_monitor;
 170   Label count, no_count;



 171 

 172   assert_different_registers(oop, box, tmp, disp_hdr, flag, t0);
 173 


 174   if (LockingMode == LM_LEGACY) {
 175     // Find the lock address and load the displaced header from the stack.
 176     ld(disp_hdr, Address(box, BasicLock::displaced_header_offset_in_bytes()));
 177 
 178     // If the displaced header is 0, we have a recursive unlock.
 179     mv(flag, disp_hdr);
 180     beqz(disp_hdr, cont);
 181   }
 182 
 183   // Handle existing monitor.
 184   ld(tmp, Address(oop, oopDesc::mark_offset_in_bytes()));
 185   test_bit(t0, tmp, exact_log2(markWord::monitor_value));
 186   bnez(t0, object_has_monitor);
 187 
 188   if (LockingMode == LM_MONITOR) {
 189     mv(flag, 1); // Set non-zero flag to indicate 'failure' -> take slow path
 190     j(cont);
 191   } else if (LockingMode == LM_LEGACY) {
 192     // Check if it is still a light weight lock, this is true if we
 193     // see the stack address of the basicLock in the markWord of the
 194     // object.
 195 
 196     cmpxchg(/*memory address*/oop, /*expected value*/box, /*new value*/disp_hdr, Assembler::int64, Assembler::relaxed,
 197             Assembler::rl, /*result*/tmp);
 198     xorr(flag, box, tmp); // box == tmp if cas succeeds
 199     j(cont);
 200   } else {
 201     assert(LockingMode == LM_LIGHTWEIGHT, "");
 202     Label slow;
 203     lightweight_unlock(oop, tmp, box, disp_hdr, slow);
 204 
 205     // Indicate success on completion.
 206     mv(flag, zr);
 207     j(count);
 208     bind(slow);
 209     mv(flag, 1); // Set non-zero flag to indicate 'failure' -> take slow path
 210     j(no_count);
 211   }
 212 
 213   assert(oopDesc::mark_offset_in_bytes() == 0, "offset of _mark is not 0");
 214 
 215   // Handle existing monitor.
 216   bind(object_has_monitor);
 217   STATIC_ASSERT(markWord::monitor_value <= INT_MAX);
 218   add(tmp, tmp, -(int)markWord::monitor_value); // monitor
 219 
 220   if (LockingMode == LM_LIGHTWEIGHT) {
 221     // If the owner is anonymous, we need to fix it -- in an outline stub.
 222     Register tmp2 = disp_hdr;
 223     ld(tmp2, Address(tmp, ObjectMonitor::owner_offset()));
 224     test_bit(t0, tmp2, exact_log2(ObjectMonitor::ANONYMOUS_OWNER));
 225     C2HandleAnonOMOwnerStub* stub = new (Compile::current()->comp_arena()) C2HandleAnonOMOwnerStub(tmp, tmp2);
 226     Compile::current()->output()->add_stub(stub);
 227     bnez(t0, stub->entry(), /* is_far */ true);
 228     bind(stub->continuation());
 229   }
 230 
 231   ld(disp_hdr, Address(tmp, ObjectMonitor::recursions_offset()));
 232 
 233   Label notRecursive;
 234   beqz(disp_hdr, notRecursive); // Will be 0 if not recursive.
 235 
 236   // Recursive lock
 237   addi(disp_hdr, disp_hdr, -1);
 238   sd(disp_hdr, Address(tmp, ObjectMonitor::recursions_offset()));
 239   mv(flag, zr);
 240   j(cont);
 241 
 242   bind(notRecursive);
 243   ld(flag, Address(tmp, ObjectMonitor::EntryList_offset()));
 244   ld(disp_hdr, Address(tmp, ObjectMonitor::cxq_offset()));
 245   orr(flag, flag, disp_hdr); // Will be 0 if both are 0.
 246   bnez(flag, cont);

 247   // need a release store here
 248   la(tmp, Address(tmp, ObjectMonitor::owner_offset()));
 249   membar(MacroAssembler::LoadStore | MacroAssembler::StoreStore);
 250   sd(zr, Address(tmp)); // set unowned
 251 
 252   bind(cont);
 253   // zero flag indicates success
 254   // non-zero flag indicates failure
 255   bnez(flag, no_count);
 256 
 257   bind(count);
 258   decrement(Address(xthread, JavaThread::held_monitor_count_offset()), 1, t0, tmp);




 259 
 260   bind(no_count);

















































































































































































































































































 261 }
 262 
 263 // short string
 264 // StringUTF16.indexOfChar
 265 // StringLatin1.indexOfChar
 266 void C2_MacroAssembler::string_indexof_char_short(Register str1, Register cnt1,
 267                                                   Register ch, Register result,
 268                                                   bool isL)
 269 {
 270   Register ch1 = t0;
 271   Register index = t1;
 272 
 273   BLOCK_COMMENT("string_indexof_char_short {");
 274 
 275   Label LOOP, LOOP1, LOOP4, LOOP8;
 276   Label MATCH,  MATCH1, MATCH2, MATCH3,
 277         MATCH4, MATCH5, MATCH6, MATCH7, NOMATCH;
 278 
 279   mv(result, -1);
 280   mv(index, zr);
 281 
 282   bind(LOOP);
 283   addi(t0, index, 8);
 284   ble(t0, cnt1, LOOP8);
 285   addi(t0, index, 4);
 286   ble(t0, cnt1, LOOP4);
 287   j(LOOP1);
 288 
 289   bind(LOOP8);
 290   isL ? lbu(ch1, Address(str1, 0)) : lhu(ch1, Address(str1, 0));
 291   beq(ch, ch1, MATCH);
 292   isL ? lbu(ch1, Address(str1, 1)) : lhu(ch1, Address(str1, 2));
 293   beq(ch, ch1, MATCH1);
 294   isL ? lbu(ch1, Address(str1, 2)) : lhu(ch1, Address(str1, 4));
 295   beq(ch, ch1, MATCH2);
 296   isL ? lbu(ch1, Address(str1, 3)) : lhu(ch1, Address(str1, 6));
 297   beq(ch, ch1, MATCH3);
 298   isL ? lbu(ch1, Address(str1, 4)) : lhu(ch1, Address(str1, 8));
 299   beq(ch, ch1, MATCH4);
 300   isL ? lbu(ch1, Address(str1, 5)) : lhu(ch1, Address(str1, 10));
 301   beq(ch, ch1, MATCH5);
 302   isL ? lbu(ch1, Address(str1, 6)) : lhu(ch1, Address(str1, 12));
 303   beq(ch, ch1, MATCH6);
 304   isL ? lbu(ch1, Address(str1, 7)) : lhu(ch1, Address(str1, 14));
 305   beq(ch, ch1, MATCH7);
 306   addi(index, index, 8);
 307   addi(str1, str1, isL ? 8 : 16);
 308   blt(index, cnt1, LOOP);
 309   j(NOMATCH);
 310 
 311   bind(LOOP4);
 312   isL ? lbu(ch1, Address(str1, 0)) : lhu(ch1, Address(str1, 0));
 313   beq(ch, ch1, MATCH);
 314   isL ? lbu(ch1, Address(str1, 1)) : lhu(ch1, Address(str1, 2));
 315   beq(ch, ch1, MATCH1);
 316   isL ? lbu(ch1, Address(str1, 2)) : lhu(ch1, Address(str1, 4));
 317   beq(ch, ch1, MATCH2);
 318   isL ? lbu(ch1, Address(str1, 3)) : lhu(ch1, Address(str1, 6));
 319   beq(ch, ch1, MATCH3);
 320   addi(index, index, 4);
 321   addi(str1, str1, isL ? 4 : 8);
 322   bge(index, cnt1, NOMATCH);
 323 
 324   bind(LOOP1);
 325   isL ? lbu(ch1, Address(str1)) : lhu(ch1, Address(str1));
 326   beq(ch, ch1, MATCH);
 327   addi(index, index, 1);
 328   addi(str1, str1, isL ? 1 : 2);
 329   blt(index, cnt1, LOOP1);
 330   j(NOMATCH);
 331 
 332   bind(MATCH1);
 333   addi(index, index, 1);
 334   j(MATCH);
 335 
 336   bind(MATCH2);
 337   addi(index, index, 2);
 338   j(MATCH);
 339 
 340   bind(MATCH3);
 341   addi(index, index, 3);
 342   j(MATCH);
 343 
 344   bind(MATCH4);
 345   addi(index, index, 4);
 346   j(MATCH);
 347 
 348   bind(MATCH5);
 349   addi(index, index, 5);
 350   j(MATCH);
 351 
 352   bind(MATCH6);
 353   addi(index, index, 6);
 354   j(MATCH);
 355 
 356   bind(MATCH7);
 357   addi(index, index, 7);
 358 
 359   bind(MATCH);
 360   mv(result, index);
 361   bind(NOMATCH);
 362   BLOCK_COMMENT("} string_indexof_char_short");
 363 }
 364 
 365 // StringUTF16.indexOfChar
 366 // StringLatin1.indexOfChar
 367 void C2_MacroAssembler::string_indexof_char(Register str1, Register cnt1,
 368                                             Register ch, Register result,
 369                                             Register tmp1, Register tmp2,
 370                                             Register tmp3, Register tmp4,
 371                                             bool isL)
 372 {
 373   Label CH1_LOOP, HIT, NOMATCH, DONE, DO_LONG;
 374   Register ch1 = t0;
 375   Register orig_cnt = t1;
 376   Register mask1 = tmp3;
 377   Register mask2 = tmp2;
 378   Register match_mask = tmp1;
 379   Register trailing_char = tmp4;
 380   Register unaligned_elems = tmp4;
 381 
 382   BLOCK_COMMENT("string_indexof_char {");
 383   beqz(cnt1, NOMATCH);
 384 
 385   addi(t0, cnt1, isL ? -32 : -16);
 386   bgtz(t0, DO_LONG);
 387   string_indexof_char_short(str1, cnt1, ch, result, isL);
 388   j(DONE);
 389 
 390   bind(DO_LONG);
 391   mv(orig_cnt, cnt1);
 392   if (AvoidUnalignedAccesses) {
 393     Label ALIGNED;
 394     andi(unaligned_elems, str1, 0x7);
 395     beqz(unaligned_elems, ALIGNED);
 396     sub(unaligned_elems, unaligned_elems, 8);
 397     neg(unaligned_elems, unaligned_elems);
 398     if (!isL) {
 399       srli(unaligned_elems, unaligned_elems, 1);
 400     }
 401     // do unaligned part per element
 402     string_indexof_char_short(str1, unaligned_elems, ch, result, isL);
 403     bgez(result, DONE);
 404     mv(orig_cnt, cnt1);
 405     sub(cnt1, cnt1, unaligned_elems);
 406     bind(ALIGNED);
 407   }
 408 
 409   // duplicate ch
 410   if (isL) {
 411     slli(ch1, ch, 8);
 412     orr(ch, ch1, ch);
 413   }
 414   slli(ch1, ch, 16);
 415   orr(ch, ch1, ch);
 416   slli(ch1, ch, 32);
 417   orr(ch, ch1, ch);
 418 
 419   if (!isL) {
 420     slli(cnt1, cnt1, 1);
 421   }
 422 
 423   uint64_t mask0101 = UCONST64(0x0101010101010101);
 424   uint64_t mask0001 = UCONST64(0x0001000100010001);
 425   mv(mask1, isL ? mask0101 : mask0001);
 426   uint64_t mask7f7f = UCONST64(0x7f7f7f7f7f7f7f7f);
 427   uint64_t mask7fff = UCONST64(0x7fff7fff7fff7fff);
 428   mv(mask2, isL ? mask7f7f : mask7fff);
 429 
 430   bind(CH1_LOOP);
 431   ld(ch1, Address(str1));
 432   addi(str1, str1, 8);
 433   addi(cnt1, cnt1, -8);
 434   compute_match_mask(ch1, ch, match_mask, mask1, mask2);
 435   bnez(match_mask, HIT);
 436   bgtz(cnt1, CH1_LOOP);
 437   j(NOMATCH);
 438 
 439   bind(HIT);
 440   ctzc_bit(trailing_char, match_mask, isL, ch1, result);
 441   srli(trailing_char, trailing_char, 3);
 442   addi(cnt1, cnt1, 8);
 443   ble(cnt1, trailing_char, NOMATCH);
 444   // match case
 445   if (!isL) {
 446     srli(cnt1, cnt1, 1);
 447     srli(trailing_char, trailing_char, 1);
 448   }
 449 
 450   sub(result, orig_cnt, cnt1);
 451   add(result, result, trailing_char);
 452   j(DONE);
 453 
 454   bind(NOMATCH);
 455   mv(result, -1);
 456 
 457   bind(DONE);
 458   BLOCK_COMMENT("} string_indexof_char");
 459 }
 460 
 461 typedef void (MacroAssembler::* load_chr_insn)(Register rd, const Address &adr, Register temp);
 462 
 463 // Search for needle in haystack and return index or -1
 464 // x10: result
 465 // x11: haystack
 466 // x12: haystack_len
 467 // x13: needle
 468 // x14: needle_len
 469 void C2_MacroAssembler::string_indexof(Register haystack, Register needle,
 470                                        Register haystack_len, Register needle_len,
 471                                        Register tmp1, Register tmp2,
 472                                        Register tmp3, Register tmp4,
 473                                        Register tmp5, Register tmp6,
 474                                        Register result, int ae)
 475 {
 476   assert(ae != StrIntrinsicNode::LU, "Invalid encoding");
 477 
 478   Label LINEARSEARCH, LINEARSTUB, DONE, NOMATCH;
 479 
 480   Register ch1 = t0;
 481   Register ch2 = t1;
 482   Register nlen_tmp = tmp1; // needle len tmp
 483   Register hlen_tmp = tmp2; // haystack len tmp
 484   Register result_tmp = tmp4;
 485 
 486   bool isLL = ae == StrIntrinsicNode::LL;
 487 
 488   bool needle_isL = ae == StrIntrinsicNode::LL || ae == StrIntrinsicNode::UL;
 489   bool haystack_isL = ae == StrIntrinsicNode::LL || ae == StrIntrinsicNode::LU;
 490   int needle_chr_shift = needle_isL ? 0 : 1;
 491   int haystack_chr_shift = haystack_isL ? 0 : 1;
 492   int needle_chr_size = needle_isL ? 1 : 2;
 493   int haystack_chr_size = haystack_isL ? 1 : 2;
 494   load_chr_insn needle_load_1chr = needle_isL ? (load_chr_insn)&MacroAssembler::lbu :
 495                               (load_chr_insn)&MacroAssembler::lhu;
 496   load_chr_insn haystack_load_1chr = haystack_isL ? (load_chr_insn)&MacroAssembler::lbu :
 497                                 (load_chr_insn)&MacroAssembler::lhu;
 498 
 499   BLOCK_COMMENT("string_indexof {");
 500 
 501   // Note, inline_string_indexOf() generates checks:
 502   // if (pattern.count > src.count) return -1;
 503   // if (pattern.count == 0) return 0;
 504 
 505   // We have two strings, a source string in haystack, haystack_len and a pattern string
 506   // in needle, needle_len. Find the first occurrence of pattern in source or return -1.
 507 
 508   // For larger pattern and source we use a simplified Boyer Moore algorithm.
 509   // With a small pattern and source we use linear scan.
 510 
 511   // needle_len >=8 && needle_len < 256 && needle_len < haystack_len/4, use bmh algorithm.
 512   sub(result_tmp, haystack_len, needle_len);
 513   // needle_len < 8, use linear scan
 514   sub(t0, needle_len, 8);
 515   bltz(t0, LINEARSEARCH);
 516   // needle_len >= 256, use linear scan
 517   sub(t0, needle_len, 256);
 518   bgez(t0, LINEARSTUB);
 519   // needle_len >= haystack_len/4, use linear scan
 520   srli(t0, haystack_len, 2);
 521   bge(needle_len, t0, LINEARSTUB);
 522 
 523   // Boyer-Moore-Horspool introduction:
 524   // The Boyer Moore alogorithm is based on the description here:-
 525   //
 526   // http://en.wikipedia.org/wiki/Boyer%E2%80%93Moore_string_search_algorithm
 527   //
 528   // This describes and algorithm with 2 shift rules. The 'Bad Character' rule
 529   // and the 'Good Suffix' rule.
 530   //
 531   // These rules are essentially heuristics for how far we can shift the
 532   // pattern along the search string.
 533   //
 534   // The implementation here uses the 'Bad Character' rule only because of the
 535   // complexity of initialisation for the 'Good Suffix' rule.
 536   //
 537   // This is also known as the Boyer-Moore-Horspool algorithm:
 538   //
 539   // http://en.wikipedia.org/wiki/Boyer-Moore-Horspool_algorithm
 540   //
 541   // #define ASIZE 256
 542   //
 543   //    int bm(unsigned char *pattern, int m, unsigned char *src, int n) {
 544   //      int i, j;
 545   //      unsigned c;
 546   //      unsigned char bc[ASIZE];
 547   //
 548   //      /* Preprocessing */
 549   //      for (i = 0; i < ASIZE; ++i)
 550   //        bc[i] = m;
 551   //      for (i = 0; i < m - 1; ) {
 552   //        c = pattern[i];
 553   //        ++i;
 554   //        // c < 256 for Latin1 string, so, no need for branch
 555   //        #ifdef PATTERN_STRING_IS_LATIN1
 556   //        bc[c] = m - i;
 557   //        #else
 558   //        if (c < ASIZE) bc[c] = m - i;
 559   //        #endif
 560   //      }
 561   //
 562   //      /* Searching */
 563   //      j = 0;
 564   //      while (j <= n - m) {
 565   //        c = src[i+j];
 566   //        if (pattern[m-1] == c)
 567   //          int k;
 568   //          for (k = m - 2; k >= 0 && pattern[k] == src[k + j]; --k);
 569   //          if (k < 0) return j;
 570   //          // c < 256 for Latin1 string, so, no need for branch
 571   //          #ifdef SOURCE_STRING_IS_LATIN1_AND_PATTERN_STRING_IS_LATIN1
 572   //          // LL case: (c< 256) always true. Remove branch
 573   //          j += bc[pattern[j+m-1]];
 574   //          #endif
 575   //          #ifdef SOURCE_STRING_IS_UTF_AND_PATTERN_STRING_IS_UTF
 576   //          // UU case: need if (c<ASIZE) check. Skip 1 character if not.
 577   //          if (c < ASIZE)
 578   //            j += bc[pattern[j+m-1]];
 579   //          else
 580   //            j += 1
 581   //          #endif
 582   //          #ifdef SOURCE_IS_UTF_AND_PATTERN_IS_LATIN1
 583   //          // UL case: need if (c<ASIZE) check. Skip <pattern length> if not.
 584   //          if (c < ASIZE)
 585   //            j += bc[pattern[j+m-1]];
 586   //          else
 587   //            j += m
 588   //          #endif
 589   //      }
 590   //      return -1;
 591   //    }
 592 
 593   // temp register:t0, tmp1, tmp2, tmp3, tmp4, tmp5, tmp6, result
 594   Label BCLOOP, BCSKIP, BMLOOPSTR2, BMLOOPSTR1, BMSKIP, BMADV, BMMATCH,
 595         BMLOOPSTR1_LASTCMP, BMLOOPSTR1_CMP, BMLOOPSTR1_AFTER_LOAD, BM_INIT_LOOP;
 596 
 597   Register haystack_end = haystack_len;
 598   Register skipch = tmp2;
 599 
 600   // pattern length is >=8, so, we can read at least 1 register for cases when
 601   // UTF->Latin1 conversion is not needed(8 LL or 4UU) and half register for
 602   // UL case. We'll re-read last character in inner pre-loop code to have
 603   // single outer pre-loop load
 604   const int firstStep = isLL ? 7 : 3;
 605 
 606   const int ASIZE = 256;
 607   const int STORE_BYTES = 8; // 8 bytes stored per instruction(sd)
 608 
 609   sub(sp, sp, ASIZE);
 610 
 611   // init BC offset table with default value: needle_len
 612   slli(t0, needle_len, 8);
 613   orr(t0, t0, needle_len); // [63...16][needle_len][needle_len]
 614   slli(tmp1, t0, 16);
 615   orr(t0, tmp1, t0); // [63...32][needle_len][needle_len][needle_len][needle_len]
 616   slli(tmp1, t0, 32);
 617   orr(tmp5, tmp1, t0); // tmp5: 8 elements [needle_len]
 618 
 619   mv(ch1, sp);  // ch1 is t0
 620   mv(tmp6, ASIZE / STORE_BYTES); // loop iterations
 621 
 622   bind(BM_INIT_LOOP);
 623   // for (i = 0; i < ASIZE; ++i)
 624   //   bc[i] = m;
 625   for (int i = 0; i < 4; i++) {
 626     sd(tmp5, Address(ch1, i * wordSize));
 627   }
 628   add(ch1, ch1, 32);
 629   sub(tmp6, tmp6, 4);
 630   bgtz(tmp6, BM_INIT_LOOP);
 631 
 632   sub(nlen_tmp, needle_len, 1); // m - 1, index of the last element in pattern
 633   Register orig_haystack = tmp5;
 634   mv(orig_haystack, haystack);
 635   // result_tmp = tmp4
 636   shadd(haystack_end, result_tmp, haystack, haystack_end, haystack_chr_shift);
 637   sub(ch2, needle_len, 1); // bc offset init value, ch2 is t1
 638   mv(tmp3, needle);
 639 
 640   //  for (i = 0; i < m - 1; ) {
 641   //    c = pattern[i];
 642   //    ++i;
 643   //    // c < 256 for Latin1 string, so, no need for branch
 644   //    #ifdef PATTERN_STRING_IS_LATIN1
 645   //    bc[c] = m - i;
 646   //    #else
 647   //    if (c < ASIZE) bc[c] = m - i;
 648   //    #endif
 649   //  }
 650   bind(BCLOOP);
 651   (this->*needle_load_1chr)(ch1, Address(tmp3), noreg);
 652   add(tmp3, tmp3, needle_chr_size);
 653   if (!needle_isL) {
 654     // ae == StrIntrinsicNode::UU
 655     mv(tmp6, ASIZE);
 656     bgeu(ch1, tmp6, BCSKIP);
 657   }
 658   add(tmp4, sp, ch1);
 659   sb(ch2, Address(tmp4)); // store skip offset to BC offset table
 660 
 661   bind(BCSKIP);
 662   sub(ch2, ch2, 1); // for next pattern element, skip distance -1
 663   bgtz(ch2, BCLOOP);
 664 
 665   // tmp6: pattern end, address after needle
 666   shadd(tmp6, needle_len, needle, tmp6, needle_chr_shift);
 667   if (needle_isL == haystack_isL) {
 668     // load last 8 bytes (8LL/4UU symbols)
 669     ld(tmp6, Address(tmp6, -wordSize));
 670   } else {
 671     // UL: from UTF-16(source) search Latin1(pattern)
 672     lwu(tmp6, Address(tmp6, -wordSize / 2)); // load last 4 bytes(4 symbols)
 673     // convert Latin1 to UTF. eg: 0x0000abcd -> 0x0a0b0c0d
 674     // We'll have to wait until load completed, but it's still faster than per-character loads+checks
 675     srli(tmp3, tmp6, BitsPerByte * (wordSize / 2 - needle_chr_size)); // pattern[m-1], eg:0x0000000a
 676     slli(ch2, tmp6, XLEN - 24);
 677     srli(ch2, ch2, XLEN - 8); // pattern[m-2], 0x0000000b
 678     slli(ch1, tmp6, XLEN - 16);
 679     srli(ch1, ch1, XLEN - 8); // pattern[m-3], 0x0000000c
 680     andi(tmp6, tmp6, 0xff); // pattern[m-4], 0x0000000d
 681     slli(ch2, ch2, 16);
 682     orr(ch2, ch2, ch1); // 0x00000b0c
 683     slli(result, tmp3, 48); // use result as temp register
 684     orr(tmp6, tmp6, result); // 0x0a00000d
 685     slli(result, ch2, 16);
 686     orr(tmp6, tmp6, result); // UTF-16:0x0a0b0c0d
 687   }
 688 
 689   // i = m - 1;
 690   // skipch = j + i;
 691   // if (skipch == pattern[m - 1]
 692   //   for (k = m - 2; k >= 0 && pattern[k] == src[k + j]; --k);
 693   // else
 694   //   move j with bad char offset table
 695   bind(BMLOOPSTR2);
 696   // compare pattern to source string backward
 697   shadd(result, nlen_tmp, haystack, result, haystack_chr_shift);
 698   (this->*haystack_load_1chr)(skipch, Address(result), noreg);
 699   sub(nlen_tmp, nlen_tmp, firstStep); // nlen_tmp is positive here, because needle_len >= 8
 700   if (needle_isL == haystack_isL) {
 701     // re-init tmp3. It's for free because it's executed in parallel with
 702     // load above. Alternative is to initialize it before loop, but it'll
 703     // affect performance on in-order systems with 2 or more ld/st pipelines
 704     srli(tmp3, tmp6, BitsPerByte * (wordSize - needle_chr_size)); // UU/LL: pattern[m-1]
 705   }
 706   if (!isLL) { // UU/UL case
 707     slli(ch2, nlen_tmp, 1); // offsets in bytes
 708   }
 709   bne(tmp3, skipch, BMSKIP); // if not equal, skipch is bad char
 710   add(result, haystack, isLL ? nlen_tmp : ch2);
 711   // load 8 bytes from source string
 712   // if isLL is false then read granularity can be 2
 713   load_long_misaligned(ch2, Address(result), ch1, isLL ? 1 : 2); // can use ch1 as temp register here as it will be trashed by next mv anyway
 714   mv(ch1, tmp6);
 715   if (isLL) {
 716     j(BMLOOPSTR1_AFTER_LOAD);
 717   } else {
 718     sub(nlen_tmp, nlen_tmp, 1); // no need to branch for UU/UL case. cnt1 >= 8
 719     j(BMLOOPSTR1_CMP);
 720   }
 721 
 722   bind(BMLOOPSTR1);
 723   shadd(ch1, nlen_tmp, needle, ch1, needle_chr_shift);
 724   (this->*needle_load_1chr)(ch1, Address(ch1), noreg);
 725   shadd(ch2, nlen_tmp, haystack, ch2, haystack_chr_shift);
 726   (this->*haystack_load_1chr)(ch2, Address(ch2), noreg);
 727 
 728   bind(BMLOOPSTR1_AFTER_LOAD);
 729   sub(nlen_tmp, nlen_tmp, 1);
 730   bltz(nlen_tmp, BMLOOPSTR1_LASTCMP);
 731 
 732   bind(BMLOOPSTR1_CMP);
 733   beq(ch1, ch2, BMLOOPSTR1);
 734 
 735   bind(BMSKIP);
 736   if (!isLL) {
 737     // if we've met UTF symbol while searching Latin1 pattern, then we can
 738     // skip needle_len symbols
 739     if (needle_isL != haystack_isL) {
 740       mv(result_tmp, needle_len);
 741     } else {
 742       mv(result_tmp, 1);
 743     }
 744     mv(t0, ASIZE);
 745     bgeu(skipch, t0, BMADV);
 746   }
 747   add(result_tmp, sp, skipch);
 748   lbu(result_tmp, Address(result_tmp)); // load skip offset
 749 
 750   bind(BMADV);
 751   sub(nlen_tmp, needle_len, 1);
 752   // move haystack after bad char skip offset
 753   shadd(haystack, result_tmp, haystack, result, haystack_chr_shift);
 754   ble(haystack, haystack_end, BMLOOPSTR2);
 755   add(sp, sp, ASIZE);
 756   j(NOMATCH);
 757 
 758   bind(BMLOOPSTR1_LASTCMP);
 759   bne(ch1, ch2, BMSKIP);
 760 
 761   bind(BMMATCH);
 762   sub(result, haystack, orig_haystack);
 763   if (!haystack_isL) {
 764     srli(result, result, 1);
 765   }
 766   add(sp, sp, ASIZE);
 767   j(DONE);
 768 
 769   bind(LINEARSTUB);
 770   sub(t0, needle_len, 16); // small patterns still should be handled by simple algorithm
 771   bltz(t0, LINEARSEARCH);
 772   mv(result, zr);
 773   RuntimeAddress stub = nullptr;
 774   if (isLL) {
 775     stub = RuntimeAddress(StubRoutines::riscv::string_indexof_linear_ll());
 776     assert(stub.target() != nullptr, "string_indexof_linear_ll stub has not been generated");
 777   } else if (needle_isL) {
 778     stub = RuntimeAddress(StubRoutines::riscv::string_indexof_linear_ul());
 779     assert(stub.target() != nullptr, "string_indexof_linear_ul stub has not been generated");
 780   } else {
 781     stub = RuntimeAddress(StubRoutines::riscv::string_indexof_linear_uu());
 782     assert(stub.target() != nullptr, "string_indexof_linear_uu stub has not been generated");
 783   }
 784   address call = trampoline_call(stub);
 785   if (call == nullptr) {
 786     DEBUG_ONLY(reset_labels(LINEARSEARCH, DONE, NOMATCH));
 787     ciEnv::current()->record_failure("CodeCache is full");
 788     return;
 789   }
 790   j(DONE);
 791 
 792   bind(NOMATCH);
 793   mv(result, -1);
 794   j(DONE);
 795 
 796   bind(LINEARSEARCH);
 797   string_indexof_linearscan(haystack, needle, haystack_len, needle_len, tmp1, tmp2, tmp3, tmp4, -1, result, ae);
 798 
 799   bind(DONE);
 800   BLOCK_COMMENT("} string_indexof");
 801 }
 802 
 803 // string_indexof
 804 // result: x10
 805 // src: x11
 806 // src_count: x12
 807 // pattern: x13
 808 // pattern_count: x14 or 1/2/3/4
 809 void C2_MacroAssembler::string_indexof_linearscan(Register haystack, Register needle,
 810                                                Register haystack_len, Register needle_len,
 811                                                Register tmp1, Register tmp2,
 812                                                Register tmp3, Register tmp4,
 813                                                int needle_con_cnt, Register result, int ae)
 814 {
 815   // Note:
 816   // needle_con_cnt > 0 means needle_len register is invalid, needle length is constant
 817   // for UU/LL: needle_con_cnt[1, 4], UL: needle_con_cnt = 1
 818   assert(needle_con_cnt <= 4, "Invalid needle constant count");
 819   assert(ae != StrIntrinsicNode::LU, "Invalid encoding");
 820 
 821   Register ch1 = t0;
 822   Register ch2 = t1;
 823   Register hlen_neg = haystack_len, nlen_neg = needle_len;
 824   Register nlen_tmp = tmp1, hlen_tmp = tmp2, result_tmp = tmp4;
 825 
 826   bool isLL = ae == StrIntrinsicNode::LL;
 827 
 828   bool needle_isL = ae == StrIntrinsicNode::LL || ae == StrIntrinsicNode::UL;
 829   bool haystack_isL = ae == StrIntrinsicNode::LL || ae == StrIntrinsicNode::LU;
 830   int needle_chr_shift = needle_isL ? 0 : 1;
 831   int haystack_chr_shift = haystack_isL ? 0 : 1;
 832   int needle_chr_size = needle_isL ? 1 : 2;
 833   int haystack_chr_size = haystack_isL ? 1 : 2;
 834 
 835   load_chr_insn needle_load_1chr = needle_isL ? (load_chr_insn)&MacroAssembler::lbu :
 836                               (load_chr_insn)&MacroAssembler::lhu;
 837   load_chr_insn haystack_load_1chr = haystack_isL ? (load_chr_insn)&MacroAssembler::lbu :
 838                                 (load_chr_insn)&MacroAssembler::lhu;
 839   load_chr_insn load_2chr = isLL ? (load_chr_insn)&MacroAssembler::lhu : (load_chr_insn)&MacroAssembler::lwu;
 840   load_chr_insn load_4chr = isLL ? (load_chr_insn)&MacroAssembler::lwu : (load_chr_insn)&MacroAssembler::ld;
 841 
 842   Label DO1, DO2, DO3, MATCH, NOMATCH, DONE;
 843 
 844   Register first = tmp3;
 845 
 846   if (needle_con_cnt == -1) {
 847     Label DOSHORT, FIRST_LOOP, STR2_NEXT, STR1_LOOP, STR1_NEXT;
 848 
 849     sub(t0, needle_len, needle_isL == haystack_isL ? 4 : 2);
 850     bltz(t0, DOSHORT);
 851 
 852     (this->*needle_load_1chr)(first, Address(needle), noreg);
 853     slli(t0, needle_len, needle_chr_shift);
 854     add(needle, needle, t0);
 855     neg(nlen_neg, t0);
 856     slli(t0, result_tmp, haystack_chr_shift);
 857     add(haystack, haystack, t0);
 858     neg(hlen_neg, t0);
 859 
 860     bind(FIRST_LOOP);
 861     add(t0, haystack, hlen_neg);
 862     (this->*haystack_load_1chr)(ch2, Address(t0), noreg);
 863     beq(first, ch2, STR1_LOOP);
 864 
 865     bind(STR2_NEXT);
 866     add(hlen_neg, hlen_neg, haystack_chr_size);
 867     blez(hlen_neg, FIRST_LOOP);
 868     j(NOMATCH);
 869 
 870     bind(STR1_LOOP);
 871     add(nlen_tmp, nlen_neg, needle_chr_size);
 872     add(hlen_tmp, hlen_neg, haystack_chr_size);
 873     bgez(nlen_tmp, MATCH);
 874 
 875     bind(STR1_NEXT);
 876     add(ch1, needle, nlen_tmp);
 877     (this->*needle_load_1chr)(ch1, Address(ch1), noreg);
 878     add(ch2, haystack, hlen_tmp);
 879     (this->*haystack_load_1chr)(ch2, Address(ch2), noreg);
 880     bne(ch1, ch2, STR2_NEXT);
 881     add(nlen_tmp, nlen_tmp, needle_chr_size);
 882     add(hlen_tmp, hlen_tmp, haystack_chr_size);
 883     bltz(nlen_tmp, STR1_NEXT);
 884     j(MATCH);
 885 
 886     bind(DOSHORT);
 887     if (needle_isL == haystack_isL) {
 888       sub(t0, needle_len, 2);
 889       bltz(t0, DO1);
 890       bgtz(t0, DO3);
 891     }
 892   }
 893 
 894   if (needle_con_cnt == 4) {
 895     Label CH1_LOOP;
 896     (this->*load_4chr)(ch1, Address(needle), noreg);
 897     sub(result_tmp, haystack_len, 4);
 898     slli(tmp3, result_tmp, haystack_chr_shift); // result as tmp
 899     add(haystack, haystack, tmp3);
 900     neg(hlen_neg, tmp3);
 901     if (AvoidUnalignedAccesses) {
 902       // preload first value, then we will read by 1 character per loop, instead of four
 903       // just shifting previous ch2 right by size of character in bits
 904       add(tmp3, haystack, hlen_neg);
 905       (this->*load_4chr)(ch2, Address(tmp3), noreg);
 906       if (isLL) {
 907         // need to erase 1 most significant byte in 32-bit value of ch2
 908         slli(ch2, ch2, 40);
 909         srli(ch2, ch2, 32);
 910       } else {
 911         slli(ch2, ch2, 16); // 2 most significant bytes will be erased by this operation
 912       }
 913     }
 914 
 915     bind(CH1_LOOP);
 916     add(tmp3, haystack, hlen_neg);
 917     if (AvoidUnalignedAccesses) {
 918       srli(ch2, ch2, isLL ? 8 : 16);
 919       (this->*haystack_load_1chr)(tmp3, Address(tmp3, isLL ? 3 : 6), noreg);
 920       slli(tmp3, tmp3, isLL ? 24 : 48);
 921       add(ch2, ch2, tmp3);
 922     } else {
 923       (this->*load_4chr)(ch2, Address(tmp3), noreg);
 924     }
 925     beq(ch1, ch2, MATCH);
 926     add(hlen_neg, hlen_neg, haystack_chr_size);
 927     blez(hlen_neg, CH1_LOOP);
 928     j(NOMATCH);
 929   }
 930 
 931   if ((needle_con_cnt == -1 && needle_isL == haystack_isL) || needle_con_cnt == 2) {
 932     Label CH1_LOOP;
 933     BLOCK_COMMENT("string_indexof DO2 {");
 934     bind(DO2);
 935     (this->*load_2chr)(ch1, Address(needle), noreg);
 936     if (needle_con_cnt == 2) {
 937       sub(result_tmp, haystack_len, 2);
 938     }
 939     slli(tmp3, result_tmp, haystack_chr_shift);
 940     add(haystack, haystack, tmp3);
 941     neg(hlen_neg, tmp3);
 942     if (AvoidUnalignedAccesses) {
 943       // preload first value, then we will read by 1 character per loop, instead of two
 944       // just shifting previous ch2 right by size of character in bits
 945       add(tmp3, haystack, hlen_neg);
 946       (this->*haystack_load_1chr)(ch2, Address(tmp3), noreg);
 947       slli(ch2, ch2, isLL ? 8 : 16);
 948     }
 949     bind(CH1_LOOP);
 950     add(tmp3, haystack, hlen_neg);
 951     if (AvoidUnalignedAccesses) {
 952       srli(ch2, ch2, isLL ? 8 : 16);
 953       (this->*haystack_load_1chr)(tmp3, Address(tmp3, isLL ? 1 : 2), noreg);
 954       slli(tmp3, tmp3, isLL ? 8 : 16);
 955       add(ch2, ch2, tmp3);
 956     } else {
 957       (this->*load_2chr)(ch2, Address(tmp3), noreg);
 958     }
 959     beq(ch1, ch2, MATCH);
 960     add(hlen_neg, hlen_neg, haystack_chr_size);
 961     blez(hlen_neg, CH1_LOOP);
 962     j(NOMATCH);
 963     BLOCK_COMMENT("} string_indexof DO2");
 964   }
 965 
 966   if ((needle_con_cnt == -1 && needle_isL == haystack_isL) || needle_con_cnt == 3) {
 967     Label FIRST_LOOP, STR2_NEXT, STR1_LOOP;
 968     BLOCK_COMMENT("string_indexof DO3 {");
 969 
 970     bind(DO3);
 971     (this->*load_2chr)(first, Address(needle), noreg);
 972     (this->*needle_load_1chr)(ch1, Address(needle, 2 * needle_chr_size), noreg);
 973     if (needle_con_cnt == 3) {
 974       sub(result_tmp, haystack_len, 3);
 975     }
 976     slli(hlen_tmp, result_tmp, haystack_chr_shift);
 977     add(haystack, haystack, hlen_tmp);
 978     neg(hlen_neg, hlen_tmp);
 979 
 980     bind(FIRST_LOOP);
 981     add(ch2, haystack, hlen_neg);
 982     if (AvoidUnalignedAccesses) {
 983       (this->*haystack_load_1chr)(tmp2, Address(ch2, isLL ? 1 : 2), noreg); // we need a temp register, we can safely use hlen_tmp here, which is a synonym for tmp2
 984       (this->*haystack_load_1chr)(ch2, Address(ch2), noreg);
 985       slli(tmp2, tmp2, isLL ? 8 : 16);
 986       add(ch2, ch2, tmp2);
 987     } else {
 988       (this->*load_2chr)(ch2, Address(ch2), noreg);
 989     }
 990     beq(first, ch2, STR1_LOOP);
 991 
 992     bind(STR2_NEXT);
 993     add(hlen_neg, hlen_neg, haystack_chr_size);
 994     blez(hlen_neg, FIRST_LOOP);
 995     j(NOMATCH);
 996 
 997     bind(STR1_LOOP);
 998     add(hlen_tmp, hlen_neg, 2 * haystack_chr_size);
 999     add(ch2, haystack, hlen_tmp);
1000     (this->*haystack_load_1chr)(ch2, Address(ch2), noreg);
1001     bne(ch1, ch2, STR2_NEXT);
1002     j(MATCH);
1003     BLOCK_COMMENT("} string_indexof DO3");
1004   }
1005 
1006   if (needle_con_cnt == -1 || needle_con_cnt == 1) {
1007     Label DO1_LOOP;
1008 
1009     BLOCK_COMMENT("string_indexof DO1 {");
1010     bind(DO1);
1011     (this->*needle_load_1chr)(ch1, Address(needle), noreg);
1012     sub(result_tmp, haystack_len, 1);
1013     slli(tmp3, result_tmp, haystack_chr_shift);
1014     add(haystack, haystack, tmp3);
1015     neg(hlen_neg, tmp3);
1016 
1017     bind(DO1_LOOP);
1018     add(tmp3, haystack, hlen_neg);
1019     (this->*haystack_load_1chr)(ch2, Address(tmp3), noreg);
1020     beq(ch1, ch2, MATCH);
1021     add(hlen_neg, hlen_neg, haystack_chr_size);
1022     blez(hlen_neg, DO1_LOOP);
1023     BLOCK_COMMENT("} string_indexof DO1");
1024   }
1025 
1026   bind(NOMATCH);
1027   mv(result, -1);
1028   j(DONE);
1029 
1030   bind(MATCH);
1031   srai(t0, hlen_neg, haystack_chr_shift);
1032   add(result, result_tmp, t0);
1033 
1034   bind(DONE);
1035 }
1036 
1037 // Compare strings.
1038 void C2_MacroAssembler::string_compare(Register str1, Register str2,
1039                                     Register cnt1, Register cnt2, Register result, Register tmp1, Register tmp2,
1040                                     Register tmp3, int ae)
1041 {
1042   Label DONE, SHORT_LOOP, SHORT_STRING, SHORT_LAST, TAIL, STUB,
1043       DIFFERENCE, NEXT_WORD, SHORT_LOOP_TAIL, SHORT_LAST2, SHORT_LAST_INIT,
1044       SHORT_LOOP_START, TAIL_CHECK, L;
1045 
1046   const int STUB_THRESHOLD = 64 + 8;
1047   bool isLL = ae == StrIntrinsicNode::LL;
1048   bool isLU = ae == StrIntrinsicNode::LU;
1049   bool isUL = ae == StrIntrinsicNode::UL;
1050 
1051   bool str1_isL = isLL || isLU;
1052   bool str2_isL = isLL || isUL;
1053 
1054   // for L strings, 1 byte for 1 character
1055   // for U strings, 2 bytes for 1 character
1056   int str1_chr_size = str1_isL ? 1 : 2;
1057   int str2_chr_size = str2_isL ? 1 : 2;
1058   int minCharsInWord = isLL ? wordSize : wordSize / 2;
1059 
1060   load_chr_insn str1_load_chr = str1_isL ? (load_chr_insn)&MacroAssembler::lbu : (load_chr_insn)&MacroAssembler::lhu;
1061   load_chr_insn str2_load_chr = str2_isL ? (load_chr_insn)&MacroAssembler::lbu : (load_chr_insn)&MacroAssembler::lhu;
1062 
1063   BLOCK_COMMENT("string_compare {");
1064 
1065   // Bizzarely, the counts are passed in bytes, regardless of whether they
1066   // are L or U strings, however the result is always in characters.
1067   if (!str1_isL) {
1068     sraiw(cnt1, cnt1, 1);
1069   }
1070   if (!str2_isL) {
1071     sraiw(cnt2, cnt2, 1);
1072   }
1073 
1074   // Compute the minimum of the string lengths and save the difference in result.
1075   sub(result, cnt1, cnt2);
1076   bgt(cnt1, cnt2, L);
1077   mv(cnt2, cnt1);
1078   bind(L);
1079 
1080   // A very short string
1081   mv(t0, minCharsInWord);
1082   ble(cnt2, t0, SHORT_STRING);
1083 
1084   // Compare longwords
1085   // load first parts of strings and finish initialization while loading
1086   {
1087     if (str1_isL == str2_isL) { // LL or UU
1088       // check if str1 and str2 is same pointer
1089       beq(str1, str2, DONE);
1090       // load 8 bytes once to compare
1091       ld(tmp1, Address(str1));
1092       ld(tmp2, Address(str2));
1093       mv(t0, STUB_THRESHOLD);
1094       bge(cnt2, t0, STUB);
1095       sub(cnt2, cnt2, minCharsInWord);
1096       beqz(cnt2, TAIL_CHECK);
1097       // convert cnt2 from characters to bytes
1098       if (!str1_isL) {
1099         slli(cnt2, cnt2, 1);
1100       }
1101       add(str2, str2, cnt2);
1102       add(str1, str1, cnt2);
1103       sub(cnt2, zr, cnt2);
1104     } else if (isLU) { // LU case
1105       lwu(tmp1, Address(str1));
1106       ld(tmp2, Address(str2));
1107       mv(t0, STUB_THRESHOLD);
1108       bge(cnt2, t0, STUB);
1109       addi(cnt2, cnt2, -4);
1110       add(str1, str1, cnt2);
1111       sub(cnt1, zr, cnt2);
1112       slli(cnt2, cnt2, 1);
1113       add(str2, str2, cnt2);
1114       inflate_lo32(tmp3, tmp1);
1115       mv(tmp1, tmp3);
1116       sub(cnt2, zr, cnt2);
1117       addi(cnt1, cnt1, 4);
1118     } else { // UL case
1119       ld(tmp1, Address(str1));
1120       lwu(tmp2, Address(str2));
1121       mv(t0, STUB_THRESHOLD);
1122       bge(cnt2, t0, STUB);
1123       addi(cnt2, cnt2, -4);
1124       slli(t0, cnt2, 1);
1125       sub(cnt1, zr, t0);
1126       add(str1, str1, t0);
1127       add(str2, str2, cnt2);
1128       inflate_lo32(tmp3, tmp2);
1129       mv(tmp2, tmp3);
1130       sub(cnt2, zr, cnt2);
1131       addi(cnt1, cnt1, 8);
1132     }
1133     addi(cnt2, cnt2, isUL ? 4 : 8);
1134     bne(tmp1, tmp2, DIFFERENCE);
1135     bgez(cnt2, TAIL);
1136 
1137     // main loop
1138     bind(NEXT_WORD);
1139     if (str1_isL == str2_isL) { // LL or UU
1140       add(t0, str1, cnt2);
1141       ld(tmp1, Address(t0));
1142       add(t0, str2, cnt2);
1143       ld(tmp2, Address(t0));
1144       addi(cnt2, cnt2, 8);
1145     } else if (isLU) { // LU case
1146       add(t0, str1, cnt1);
1147       lwu(tmp1, Address(t0));
1148       add(t0, str2, cnt2);
1149       ld(tmp2, Address(t0));
1150       addi(cnt1, cnt1, 4);
1151       inflate_lo32(tmp3, tmp1);
1152       mv(tmp1, tmp3);
1153       addi(cnt2, cnt2, 8);
1154     } else { // UL case
1155       add(t0, str2, cnt2);
1156       lwu(tmp2, Address(t0));
1157       add(t0, str1, cnt1);
1158       ld(tmp1, Address(t0));
1159       inflate_lo32(tmp3, tmp2);
1160       mv(tmp2, tmp3);
1161       addi(cnt1, cnt1, 8);
1162       addi(cnt2, cnt2, 4);
1163     }
1164     bne(tmp1, tmp2, DIFFERENCE);
1165     bltz(cnt2, NEXT_WORD);
1166     bind(TAIL);
1167     if (str1_isL == str2_isL) { // LL or UU
1168       load_long_misaligned(tmp1, Address(str1), tmp3, isLL ? 1 : 2);
1169       load_long_misaligned(tmp2, Address(str2), tmp3, isLL ? 1 : 2);
1170     } else if (isLU) { // LU case
1171       load_int_misaligned(tmp1, Address(str1), tmp3, false);
1172       load_long_misaligned(tmp2, Address(str2), tmp3, 2);
1173       inflate_lo32(tmp3, tmp1);
1174       mv(tmp1, tmp3);
1175     } else { // UL case
1176       load_int_misaligned(tmp2, Address(str2), tmp3, false);
1177       load_long_misaligned(tmp1, Address(str1), tmp3, 2);
1178       inflate_lo32(tmp3, tmp2);
1179       mv(tmp2, tmp3);
1180     }
1181     bind(TAIL_CHECK);
1182     beq(tmp1, tmp2, DONE);
1183 
1184     // Find the first different characters in the longwords and
1185     // compute their difference.
1186     bind(DIFFERENCE);
1187     xorr(tmp3, tmp1, tmp2);
1188     ctzc_bit(result, tmp3, isLL); // count zero from lsb to msb
1189     srl(tmp1, tmp1, result);
1190     srl(tmp2, tmp2, result);
1191     if (isLL) {
1192       andi(tmp1, tmp1, 0xFF);
1193       andi(tmp2, tmp2, 0xFF);
1194     } else {
1195       andi(tmp1, tmp1, 0xFFFF);
1196       andi(tmp2, tmp2, 0xFFFF);
1197     }
1198     sub(result, tmp1, tmp2);
1199     j(DONE);
1200   }
1201 
1202   bind(STUB);
1203   RuntimeAddress stub = nullptr;
1204   switch (ae) {
1205     case StrIntrinsicNode::LL:
1206       stub = RuntimeAddress(StubRoutines::riscv::compare_long_string_LL());
1207       break;
1208     case StrIntrinsicNode::UU:
1209       stub = RuntimeAddress(StubRoutines::riscv::compare_long_string_UU());
1210       break;
1211     case StrIntrinsicNode::LU:
1212       stub = RuntimeAddress(StubRoutines::riscv::compare_long_string_LU());
1213       break;
1214     case StrIntrinsicNode::UL:
1215       stub = RuntimeAddress(StubRoutines::riscv::compare_long_string_UL());
1216       break;
1217     default:
1218       ShouldNotReachHere();
1219   }
1220   assert(stub.target() != nullptr, "compare_long_string stub has not been generated");
1221   address call = trampoline_call(stub);
1222   if (call == nullptr) {
1223     DEBUG_ONLY(reset_labels(DONE, SHORT_LOOP, SHORT_STRING, SHORT_LAST, SHORT_LOOP_TAIL, SHORT_LAST2, SHORT_LAST_INIT, SHORT_LOOP_START));
1224     ciEnv::current()->record_failure("CodeCache is full");
1225     return;
1226   }
1227   j(DONE);
1228 
1229   bind(SHORT_STRING);
1230   // Is the minimum length zero?
1231   beqz(cnt2, DONE);
1232   // arrange code to do most branches while loading and loading next characters
1233   // while comparing previous
1234   (this->*str1_load_chr)(tmp1, Address(str1), t0);
1235   addi(str1, str1, str1_chr_size);
1236   addi(cnt2, cnt2, -1);
1237   beqz(cnt2, SHORT_LAST_INIT);
1238   (this->*str2_load_chr)(cnt1, Address(str2), t0);
1239   addi(str2, str2, str2_chr_size);
1240   j(SHORT_LOOP_START);
1241   bind(SHORT_LOOP);
1242   addi(cnt2, cnt2, -1);
1243   beqz(cnt2, SHORT_LAST);
1244   bind(SHORT_LOOP_START);
1245   (this->*str1_load_chr)(tmp2, Address(str1), t0);
1246   addi(str1, str1, str1_chr_size);
1247   (this->*str2_load_chr)(t0, Address(str2), t0);
1248   addi(str2, str2, str2_chr_size);
1249   bne(tmp1, cnt1, SHORT_LOOP_TAIL);
1250   addi(cnt2, cnt2, -1);
1251   beqz(cnt2, SHORT_LAST2);
1252   (this->*str1_load_chr)(tmp1, Address(str1), t0);
1253   addi(str1, str1, str1_chr_size);
1254   (this->*str2_load_chr)(cnt1, Address(str2), t0);
1255   addi(str2, str2, str2_chr_size);
1256   beq(tmp2, t0, SHORT_LOOP);
1257   sub(result, tmp2, t0);
1258   j(DONE);
1259   bind(SHORT_LOOP_TAIL);
1260   sub(result, tmp1, cnt1);
1261   j(DONE);
1262   bind(SHORT_LAST2);
1263   beq(tmp2, t0, DONE);
1264   sub(result, tmp2, t0);
1265 
1266   j(DONE);
1267   bind(SHORT_LAST_INIT);
1268   (this->*str2_load_chr)(cnt1, Address(str2), t0);
1269   addi(str2, str2, str2_chr_size);
1270   bind(SHORT_LAST);
1271   beq(tmp1, cnt1, DONE);
1272   sub(result, tmp1, cnt1);
1273 
1274   bind(DONE);
1275 
1276   BLOCK_COMMENT("} string_compare");
1277 }
1278 
1279 void C2_MacroAssembler::arrays_equals(Register a1, Register a2, Register tmp3,
1280                                       Register tmp4, Register tmp5, Register tmp6, Register result,
1281                                       Register cnt1, int elem_size) {
1282   Label DONE, SAME, NEXT_DWORD, SHORT, TAIL, TAIL2, IS_TMP5_ZR;
1283   Register tmp1 = t0;
1284   Register tmp2 = t1;
1285   Register cnt2 = tmp2;  // cnt2 only used in array length compare
1286   Register elem_per_word = tmp6;
1287   int log_elem_size = exact_log2(elem_size);
1288   int length_offset = arrayOopDesc::length_offset_in_bytes();
1289   int base_offset   = arrayOopDesc::base_offset_in_bytes(elem_size == 2 ? T_CHAR : T_BYTE);
1290 
1291   assert(elem_size == 1 || elem_size == 2, "must be char or byte");
1292   assert_different_registers(a1, a2, result, cnt1, t0, t1, tmp3, tmp4, tmp5, tmp6);
1293   mv(elem_per_word, wordSize / elem_size);
1294 
1295   BLOCK_COMMENT("arrays_equals {");
1296 
1297   // if (a1 == a2), return true
1298   beq(a1, a2, SAME);
1299 
1300   mv(result, false);
1301   beqz(a1, DONE);
1302   beqz(a2, DONE);
1303   lwu(cnt1, Address(a1, length_offset));
1304   lwu(cnt2, Address(a2, length_offset));
1305   bne(cnt2, cnt1, DONE);
1306   beqz(cnt1, SAME);
1307 
1308   slli(tmp5, cnt1, 3 + log_elem_size);
1309   sub(tmp5, zr, tmp5);
1310   add(a1, a1, base_offset);
1311   add(a2, a2, base_offset);
1312   ld(tmp3, Address(a1, 0));
1313   ld(tmp4, Address(a2, 0));
1314   ble(cnt1, elem_per_word, SHORT); // short or same
1315 
1316   // Main 16 byte comparison loop with 2 exits
1317   bind(NEXT_DWORD); {
1318     ld(tmp1, Address(a1, wordSize));
1319     ld(tmp2, Address(a2, wordSize));
1320     sub(cnt1, cnt1, 2 * wordSize / elem_size);
1321     blez(cnt1, TAIL);
1322     bne(tmp3, tmp4, DONE);
1323     ld(tmp3, Address(a1, 2 * wordSize));
1324     ld(tmp4, Address(a2, 2 * wordSize));
1325     add(a1, a1, 2 * wordSize);
1326     add(a2, a2, 2 * wordSize);
1327     ble(cnt1, elem_per_word, TAIL2);
1328   } beq(tmp1, tmp2, NEXT_DWORD);
1329   j(DONE);
1330 
1331   bind(TAIL);
1332   xorr(tmp4, tmp3, tmp4);
1333   xorr(tmp2, tmp1, tmp2);
1334   sll(tmp2, tmp2, tmp5);
1335   orr(tmp5, tmp4, tmp2);
1336   j(IS_TMP5_ZR);
1337 
1338   bind(TAIL2);
1339   bne(tmp1, tmp2, DONE);
1340 
1341   bind(SHORT);
1342   xorr(tmp4, tmp3, tmp4);
1343   sll(tmp5, tmp4, tmp5);
1344 
1345   bind(IS_TMP5_ZR);
1346   bnez(tmp5, DONE);
1347 
1348   bind(SAME);
1349   mv(result, true);
1350   // That's it.
1351   bind(DONE);
1352 
1353   BLOCK_COMMENT("} array_equals");
1354 }
1355 
1356 // Compare Strings
1357 
1358 // For Strings we're passed the address of the first characters in a1
1359 // and a2 and the length in cnt1.
1360 // elem_size is the element size in bytes: either 1 or 2.
1361 // There are two implementations.  For arrays >= 8 bytes, all
1362 // comparisons (for hw supporting unaligned access: including the final one,
1363 // which may overlap) are performed 8 bytes at a time.
1364 // For strings < 8 bytes (and for tails of long strings when
1365 // AvoidUnalignedAccesses is true), we compare a
1366 // halfword, then a short, and then a byte.
1367 
1368 void C2_MacroAssembler::string_equals(Register a1, Register a2,
1369                                       Register result, Register cnt1, int elem_size)
1370 {
1371   Label SAME, DONE, SHORT, NEXT_WORD;
1372   Register tmp1 = t0;
1373   Register tmp2 = t1;
1374 
1375   assert(elem_size == 1 || elem_size == 2, "must be 2 or 1 byte");
1376   assert_different_registers(a1, a2, result, cnt1, tmp1, tmp2);
1377 
1378   BLOCK_COMMENT("string_equals {");
1379 
1380   beqz(cnt1, SAME);
1381   mv(result, false);
1382 
1383   // Check for short strings, i.e. smaller than wordSize.
1384   sub(cnt1, cnt1, wordSize);
1385   bltz(cnt1, SHORT);
1386 
1387   // Main 8 byte comparison loop.
1388   bind(NEXT_WORD); {
1389     ld(tmp1, Address(a1, 0));
1390     add(a1, a1, wordSize);
1391     ld(tmp2, Address(a2, 0));
1392     add(a2, a2, wordSize);
1393     sub(cnt1, cnt1, wordSize);
1394     bne(tmp1, tmp2, DONE);
1395   } bgez(cnt1, NEXT_WORD);
1396 
1397   if (!AvoidUnalignedAccesses) {
1398     // Last longword.  In the case where length == 4 we compare the
1399     // same longword twice, but that's still faster than another
1400     // conditional branch.
1401     // cnt1 could be 0, -1, -2, -3, -4 for chars; -4 only happens when
1402     // length == 4.
1403     add(tmp1, a1, cnt1);
1404     ld(tmp1, Address(tmp1, 0));
1405     add(tmp2, a2, cnt1);
1406     ld(tmp2, Address(tmp2, 0));
1407     bne(tmp1, tmp2, DONE);
1408     j(SAME);
1409   } else {
1410     add(tmp1, cnt1, wordSize);
1411     beqz(tmp1, SAME);
1412   }
1413 
1414   bind(SHORT);
1415   Label TAIL03, TAIL01;
1416 
1417   // 0-7 bytes left.
1418   test_bit(tmp1, cnt1, 2);
1419   beqz(tmp1, TAIL03);
1420   {
1421     lwu(tmp1, Address(a1, 0));
1422     add(a1, a1, 4);
1423     lwu(tmp2, Address(a2, 0));
1424     add(a2, a2, 4);
1425     bne(tmp1, tmp2, DONE);
1426   }
1427 
1428   bind(TAIL03);
1429   // 0-3 bytes left.
1430   test_bit(tmp1, cnt1, 1);
1431   beqz(tmp1, TAIL01);
1432   {
1433     lhu(tmp1, Address(a1, 0));
1434     add(a1, a1, 2);
1435     lhu(tmp2, Address(a2, 0));
1436     add(a2, a2, 2);
1437     bne(tmp1, tmp2, DONE);
1438   }
1439 
1440   bind(TAIL01);
1441   if (elem_size == 1) { // Only needed when comparing 1-byte elements
1442     // 0-1 bytes left.
1443     test_bit(tmp1, cnt1, 0);
1444     beqz(tmp1, SAME);
1445     {
1446       lbu(tmp1, Address(a1, 0));
1447       lbu(tmp2, Address(a2, 0));
1448       bne(tmp1, tmp2, DONE);
1449     }
1450   }
1451 
1452   // Arrays are equal.
1453   bind(SAME);
1454   mv(result, true);
1455 
1456   // That's it.
1457   bind(DONE);
1458   BLOCK_COMMENT("} string_equals");
1459 }
1460 
1461 typedef void (Assembler::*conditional_branch_insn)(Register op1, Register op2, Label& label, bool is_far);
1462 typedef void (MacroAssembler::*float_conditional_branch_insn)(FloatRegister op1, FloatRegister op2, Label& label,
1463                                                               bool is_far, bool is_unordered);
1464 
1465 static conditional_branch_insn conditional_branches[] =
1466 {
1467   /* SHORT branches */
1468   (conditional_branch_insn)&MacroAssembler::beq,
1469   (conditional_branch_insn)&MacroAssembler::bgt,
1470   nullptr, // BoolTest::overflow
1471   (conditional_branch_insn)&MacroAssembler::blt,
1472   (conditional_branch_insn)&MacroAssembler::bne,
1473   (conditional_branch_insn)&MacroAssembler::ble,
1474   nullptr, // BoolTest::no_overflow
1475   (conditional_branch_insn)&MacroAssembler::bge,
1476 
1477   /* UNSIGNED branches */
1478   (conditional_branch_insn)&MacroAssembler::beq,
1479   (conditional_branch_insn)&MacroAssembler::bgtu,
1480   nullptr,
1481   (conditional_branch_insn)&MacroAssembler::bltu,
1482   (conditional_branch_insn)&MacroAssembler::bne,
1483   (conditional_branch_insn)&MacroAssembler::bleu,
1484   nullptr,
1485   (conditional_branch_insn)&MacroAssembler::bgeu
1486 };
1487 
1488 static float_conditional_branch_insn float_conditional_branches[] =
1489 {
1490   /* FLOAT SHORT branches */
1491   (float_conditional_branch_insn)&MacroAssembler::float_beq,
1492   (float_conditional_branch_insn)&MacroAssembler::float_bgt,
1493   nullptr,  // BoolTest::overflow
1494   (float_conditional_branch_insn)&MacroAssembler::float_blt,
1495   (float_conditional_branch_insn)&MacroAssembler::float_bne,
1496   (float_conditional_branch_insn)&MacroAssembler::float_ble,
1497   nullptr, // BoolTest::no_overflow
1498   (float_conditional_branch_insn)&MacroAssembler::float_bge,
1499 
1500   /* DOUBLE SHORT branches */
1501   (float_conditional_branch_insn)&MacroAssembler::double_beq,
1502   (float_conditional_branch_insn)&MacroAssembler::double_bgt,
1503   nullptr,
1504   (float_conditional_branch_insn)&MacroAssembler::double_blt,
1505   (float_conditional_branch_insn)&MacroAssembler::double_bne,
1506   (float_conditional_branch_insn)&MacroAssembler::double_ble,
1507   nullptr,
1508   (float_conditional_branch_insn)&MacroAssembler::double_bge
1509 };
1510 
1511 void C2_MacroAssembler::cmp_branch(int cmpFlag, Register op1, Register op2, Label& label, bool is_far) {
1512   assert(cmpFlag >= 0 && cmpFlag < (int)(sizeof(conditional_branches) / sizeof(conditional_branches[0])),
1513          "invalid conditional branch index");
1514   (this->*conditional_branches[cmpFlag])(op1, op2, label, is_far);
1515 }
1516 
1517 // This is a function should only be used by C2. Flip the unordered when unordered-greater, C2 would use
1518 // unordered-lesser instead of unordered-greater. Finally, commute the result bits at function do_one_bytecode().
1519 void C2_MacroAssembler::float_cmp_branch(int cmpFlag, FloatRegister op1, FloatRegister op2, Label& label, bool is_far) {
1520   assert(cmpFlag >= 0 && cmpFlag < (int)(sizeof(float_conditional_branches) / sizeof(float_conditional_branches[0])),
1521          "invalid float conditional branch index");
1522   int booltest_flag = cmpFlag & ~(C2_MacroAssembler::double_branch_mask);
1523   (this->*float_conditional_branches[cmpFlag])(op1, op2, label, is_far,
1524     (booltest_flag == (BoolTest::ge) || booltest_flag == (BoolTest::gt)) ? false : true);
1525 }
1526 
1527 void C2_MacroAssembler::enc_cmpUEqNeLeGt_imm0_branch(int cmpFlag, Register op1, Label& L, bool is_far) {
1528   switch (cmpFlag) {
1529     case BoolTest::eq:
1530     case BoolTest::le:
1531       beqz(op1, L, is_far);
1532       break;
1533     case BoolTest::ne:
1534     case BoolTest::gt:
1535       bnez(op1, L, is_far);
1536       break;
1537     default:
1538       ShouldNotReachHere();
1539   }
1540 }
1541 
1542 void C2_MacroAssembler::enc_cmpEqNe_imm0_branch(int cmpFlag, Register op1, Label& L, bool is_far) {
1543   switch (cmpFlag) {
1544     case BoolTest::eq:
1545       beqz(op1, L, is_far);
1546       break;
1547     case BoolTest::ne:
1548       bnez(op1, L, is_far);
1549       break;
1550     default:
1551       ShouldNotReachHere();
1552   }
1553 }
1554 
1555 void C2_MacroAssembler::enc_cmove(int cmpFlag, Register op1, Register op2, Register dst, Register src) {
1556   Label L;
1557   cmp_branch(cmpFlag ^ (1 << neg_cond_bits), op1, op2, L);
1558   mv(dst, src);
1559   bind(L);
1560 }
1561 
1562 // Set dst to NaN if any NaN input.
1563 void C2_MacroAssembler::minmax_fp(FloatRegister dst, FloatRegister src1, FloatRegister src2,
1564                                   bool is_double, bool is_min) {
1565   assert_different_registers(dst, src1, src2);
1566 
1567   Label Done, Compare;
1568 
1569   is_double ? fclass_d(t0, src1)
1570             : fclass_s(t0, src1);
1571   is_double ? fclass_d(t1, src2)
1572             : fclass_s(t1, src2);
1573   orr(t0, t0, t1);
1574   andi(t0, t0, 0b1100000000); //if src1 or src2 is quiet or signaling NaN then return NaN
1575   beqz(t0, Compare);
1576   is_double ? fadd_d(dst, src1, src2)
1577             : fadd_s(dst, src1, src2);
1578   j(Done);
1579 
1580   bind(Compare);
1581   if (is_double) {
1582     is_min ? fmin_d(dst, src1, src2)
1583            : fmax_d(dst, src1, src2);
1584   } else {
1585     is_min ? fmin_s(dst, src1, src2)
1586            : fmax_s(dst, src1, src2);
1587   }
1588 
1589   bind(Done);
1590 }
1591 
1592 void C2_MacroAssembler::element_compare(Register a1, Register a2, Register result, Register cnt, Register tmp1, Register tmp2,
1593                                         VectorRegister vr1, VectorRegister vr2, VectorRegister vrs, bool islatin, Label &DONE) {
1594   Label loop;
1595   Assembler::SEW sew = islatin ? Assembler::e8 : Assembler::e16;
1596 
1597   bind(loop);
1598   vsetvli(tmp1, cnt, sew, Assembler::m2);
1599   vlex_v(vr1, a1, sew);
1600   vlex_v(vr2, a2, sew);
1601   vmsne_vv(vrs, vr1, vr2);
1602   vfirst_m(tmp2, vrs);
1603   bgez(tmp2, DONE);
1604   sub(cnt, cnt, tmp1);
1605   if (!islatin) {
1606     slli(tmp1, tmp1, 1); // get byte counts
1607   }
1608   add(a1, a1, tmp1);
1609   add(a2, a2, tmp1);
1610   bnez(cnt, loop);
1611 
1612   mv(result, true);
1613 }
1614 
1615 void C2_MacroAssembler::string_equals_v(Register a1, Register a2, Register result, Register cnt, int elem_size) {
1616   Label DONE;
1617   Register tmp1 = t0;
1618   Register tmp2 = t1;
1619 
1620   BLOCK_COMMENT("string_equals_v {");
1621 
1622   mv(result, false);
1623 
1624   if (elem_size == 2) {
1625     srli(cnt, cnt, 1);
1626   }
1627 
1628   element_compare(a1, a2, result, cnt, tmp1, tmp2, v2, v4, v2, elem_size == 1, DONE);
1629 
1630   bind(DONE);
1631   BLOCK_COMMENT("} string_equals_v");
1632 }
1633 
1634 // used by C2 ClearArray patterns.
1635 // base: Address of a buffer to be zeroed
1636 // cnt: Count in HeapWords
1637 //
1638 // base, cnt, v4, v5, v6, v7 and t0 are clobbered.
1639 void C2_MacroAssembler::clear_array_v(Register base, Register cnt) {
1640   Label loop;
1641 
1642   // making zero words
1643   vsetvli(t0, cnt, Assembler::e64, Assembler::m4);
1644   vxor_vv(v4, v4, v4);
1645 
1646   bind(loop);
1647   vsetvli(t0, cnt, Assembler::e64, Assembler::m4);
1648   vse64_v(v4, base);
1649   sub(cnt, cnt, t0);
1650   shadd(base, t0, base, t0, 3);
1651   bnez(cnt, loop);
1652 }
1653 
1654 void C2_MacroAssembler::arrays_equals_v(Register a1, Register a2, Register result,
1655                                         Register cnt1, int elem_size) {
1656   Label DONE;
1657   Register tmp1 = t0;
1658   Register tmp2 = t1;
1659   Register cnt2 = tmp2;
1660   int length_offset = arrayOopDesc::length_offset_in_bytes();
1661   int base_offset = arrayOopDesc::base_offset_in_bytes(elem_size == 2 ? T_CHAR : T_BYTE);
1662 
1663   BLOCK_COMMENT("arrays_equals_v {");
1664 
1665   // if (a1 == a2), return true
1666   mv(result, true);
1667   beq(a1, a2, DONE);
1668 
1669   mv(result, false);
1670   // if a1 == null or a2 == null, return false
1671   beqz(a1, DONE);
1672   beqz(a2, DONE);
1673   // if (a1.length != a2.length), return false
1674   lwu(cnt1, Address(a1, length_offset));
1675   lwu(cnt2, Address(a2, length_offset));
1676   bne(cnt1, cnt2, DONE);
1677 
1678   la(a1, Address(a1, base_offset));
1679   la(a2, Address(a2, base_offset));
1680 
1681   element_compare(a1, a2, result, cnt1, tmp1, tmp2, v2, v4, v2, elem_size == 1, DONE);
1682 
1683   bind(DONE);
1684 
1685   BLOCK_COMMENT("} arrays_equals_v");
1686 }
1687 
1688 void C2_MacroAssembler::string_compare_v(Register str1, Register str2, Register cnt1, Register cnt2,
1689                                          Register result, Register tmp1, Register tmp2, int encForm) {
1690   Label DIFFERENCE, DONE, L, loop;
1691   bool encLL = encForm == StrIntrinsicNode::LL;
1692   bool encLU = encForm == StrIntrinsicNode::LU;
1693   bool encUL = encForm == StrIntrinsicNode::UL;
1694 
1695   bool str1_isL = encLL || encLU;
1696   bool str2_isL = encLL || encUL;
1697 
1698   int minCharsInWord = encLL ? wordSize : wordSize / 2;
1699 
1700   BLOCK_COMMENT("string_compare {");
1701 
1702   // for Latin strings, 1 byte for 1 character
1703   // for UTF16 strings, 2 bytes for 1 character
1704   if (!str1_isL)
1705     sraiw(cnt1, cnt1, 1);
1706   if (!str2_isL)
1707     sraiw(cnt2, cnt2, 1);
1708 
1709   // if str1 == str2, return the difference
1710   // save the minimum of the string lengths in cnt2.
1711   sub(result, cnt1, cnt2);
1712   bgt(cnt1, cnt2, L);
1713   mv(cnt2, cnt1);
1714   bind(L);
1715 
1716   if (str1_isL == str2_isL) { // LL or UU
1717     element_compare(str1, str2, zr, cnt2, tmp1, tmp2, v2, v4, v2, encLL, DIFFERENCE);
1718     j(DONE);
1719   } else { // LU or UL
1720     Register strL = encLU ? str1 : str2;
1721     Register strU = encLU ? str2 : str1;
1722     VectorRegister vstr1 = encLU ? v8 : v4;
1723     VectorRegister vstr2 = encLU ? v4 : v8;
1724 
1725     bind(loop);
1726     vsetvli(tmp1, cnt2, Assembler::e8, Assembler::m2);
1727     vle8_v(vstr1, strL);
1728     vsetvli(tmp1, cnt2, Assembler::e16, Assembler::m4);
1729     vzext_vf2(vstr2, vstr1);
1730     vle16_v(vstr1, strU);
1731     vmsne_vv(v4, vstr2, vstr1);
1732     vfirst_m(tmp2, v4);
1733     bgez(tmp2, DIFFERENCE);
1734     sub(cnt2, cnt2, tmp1);
1735     add(strL, strL, tmp1);
1736     shadd(strU, tmp1, strU, tmp1, 1);
1737     bnez(cnt2, loop);
1738     j(DONE);
1739   }
1740 
1741   bind(DIFFERENCE);
1742   slli(tmp1, tmp2, 1);
1743   add(str1, str1, str1_isL ? tmp2 : tmp1);
1744   add(str2, str2, str2_isL ? tmp2 : tmp1);
1745   str1_isL ? lbu(tmp1, Address(str1, 0)) : lhu(tmp1, Address(str1, 0));
1746   str2_isL ? lbu(tmp2, Address(str2, 0)) : lhu(tmp2, Address(str2, 0));
1747   sub(result, tmp1, tmp2);
1748 
1749   bind(DONE);
1750 }
1751 
1752 void C2_MacroAssembler::byte_array_inflate_v(Register src, Register dst, Register len, Register tmp) {
1753   Label loop;
1754   assert_different_registers(src, dst, len, tmp, t0);
1755 
1756   BLOCK_COMMENT("byte_array_inflate_v {");
1757   bind(loop);
1758   vsetvli(tmp, len, Assembler::e8, Assembler::m2);
1759   vle8_v(v6, src);
1760   vsetvli(t0, len, Assembler::e16, Assembler::m4);
1761   vzext_vf2(v4, v6);
1762   vse16_v(v4, dst);
1763   sub(len, len, tmp);
1764   add(src, src, tmp);
1765   shadd(dst, tmp, dst, tmp, 1);
1766   bnez(len, loop);
1767   BLOCK_COMMENT("} byte_array_inflate_v");
1768 }
1769 
1770 // Compress char[] array to byte[].
1771 // result: the array length if every element in array can be encoded; 0, otherwise.
1772 void C2_MacroAssembler::char_array_compress_v(Register src, Register dst, Register len,
1773                                               Register result, Register tmp) {
1774   Label done;
1775   encode_iso_array_v(src, dst, len, result, tmp, false);
1776   beqz(len, done);
1777   mv(result, zr);
1778   bind(done);
1779 }
1780 
1781 // Intrinsic for
1782 //
1783 // - sun/nio/cs/ISO_8859_1$Encoder.implEncodeISOArray
1784 //     return the number of characters copied.
1785 // - java/lang/StringUTF16.compress
1786 //     return zero (0) if copy fails, otherwise 'len'.
1787 //
1788 // This version always returns the number of characters copied. A successful
1789 // copy will complete with the post-condition: 'res' == 'len', while an
1790 // unsuccessful copy will exit with the post-condition: 0 <= 'res' < 'len'.
1791 //
1792 // Clobbers: src, dst, len, result, t0
1793 void C2_MacroAssembler::encode_iso_array_v(Register src, Register dst, Register len,
1794                                            Register result, Register tmp, bool ascii) {
1795   Label loop, fail, done;
1796 
1797   BLOCK_COMMENT("encode_iso_array_v {");
1798   mv(result, 0);
1799 
1800   bind(loop);
1801   mv(tmp, ascii ? 0x7f : 0xff);
1802   vsetvli(t0, len, Assembler::e16, Assembler::m2);
1803   vle16_v(v2, src);
1804 
1805   vmsgtu_vx(v1, v2, tmp);
1806   vfirst_m(tmp, v1);
1807   vmsbf_m(v0, v1);
1808   // compress char to byte
1809   vsetvli(t0, len, Assembler::e8);
1810   vncvt_x_x_w(v1, v2, Assembler::v0_t);
1811   vse8_v(v1, dst, Assembler::v0_t);
1812 
1813   // fail if char > 0x7f/0xff
1814   bgez(tmp, fail);
1815   add(result, result, t0);
1816   add(dst, dst, t0);
1817   sub(len, len, t0);
1818   shadd(src, t0, src, t0, 1);
1819   bnez(len, loop);
1820   j(done);
1821 
1822   bind(fail);
1823   add(result, result, tmp);
1824 
1825   bind(done);
1826   BLOCK_COMMENT("} encode_iso_array_v");
1827 }
1828 
1829 void C2_MacroAssembler::count_positives_v(Register ary, Register len, Register result, Register tmp) {
1830   Label LOOP, SET_RESULT, DONE;
1831 
1832   BLOCK_COMMENT("count_positives_v {");
1833   assert_different_registers(ary, len, result, tmp);
1834 
1835   mv(result, zr);
1836 
1837   bind(LOOP);
1838   vsetvli(t0, len, Assembler::e8, Assembler::m4);
1839   vle8_v(v4, ary);
1840   vmslt_vx(v4, v4, zr);
1841   vfirst_m(tmp, v4);
1842   bgez(tmp, SET_RESULT);
1843   // if tmp == -1, all bytes are positive
1844   add(result, result, t0);
1845 
1846   sub(len, len, t0);
1847   add(ary, ary, t0);
1848   bnez(len, LOOP);
1849   j(DONE);
1850 
1851   // add remaining positive bytes count
1852   bind(SET_RESULT);
1853   add(result, result, tmp);
1854 
1855   bind(DONE);
1856   BLOCK_COMMENT("} count_positives_v");
1857 }
1858 
1859 void C2_MacroAssembler::string_indexof_char_v(Register str1, Register cnt1,
1860                                               Register ch, Register result,
1861                                               Register tmp1, Register tmp2,
1862                                               bool isL) {
1863   mv(result, zr);
1864 
1865   Label loop, MATCH, DONE;
1866   Assembler::SEW sew = isL ? Assembler::e8 : Assembler::e16;
1867   bind(loop);
1868   vsetvli(tmp1, cnt1, sew, Assembler::m4);
1869   vlex_v(v4, str1, sew);
1870   vmseq_vx(v4, v4, ch);
1871   vfirst_m(tmp2, v4);
1872   bgez(tmp2, MATCH); // if equal, return index
1873 
1874   add(result, result, tmp1);
1875   sub(cnt1, cnt1, tmp1);
1876   if (!isL) slli(tmp1, tmp1, 1);
1877   add(str1, str1, tmp1);
1878   bnez(cnt1, loop);
1879 
1880   mv(result, -1);
1881   j(DONE);
1882 
1883   bind(MATCH);
1884   add(result, result, tmp2);
1885 
1886   bind(DONE);
1887 }
1888 
1889 // Set dst to NaN if any NaN input.
1890 void C2_MacroAssembler::minmax_fp_v(VectorRegister dst, VectorRegister src1, VectorRegister src2,
1891                                     bool is_double, bool is_min, int vector_length) {
1892   assert_different_registers(dst, src1, src2);
1893 
1894   vsetvli_helper(is_double ? T_DOUBLE : T_FLOAT, vector_length);
1895 
1896   is_min ? vfmin_vv(dst, src1, src2)
1897          : vfmax_vv(dst, src1, src2);
1898 
1899   vmfne_vv(v0,  src1, src1);
1900   vfadd_vv(dst, src1, src1, Assembler::v0_t);
1901   vmfne_vv(v0,  src2, src2);
1902   vfadd_vv(dst, src2, src2, Assembler::v0_t);
1903 }
1904 
1905 // Set dst to NaN if any NaN input.
1906 // The destination vector register elements corresponding to masked-off elements
1907 // are handled with a mask-undisturbed policy.
1908 void C2_MacroAssembler::minmax_fp_masked_v(VectorRegister dst, VectorRegister src1, VectorRegister src2,
1909                                            VectorRegister vmask, VectorRegister tmp1, VectorRegister tmp2,
1910                                            bool is_double, bool is_min, int vector_length) {
1911   assert_different_registers(src1, src2, tmp1, tmp2);
1912   vsetvli_helper(is_double ? T_DOUBLE : T_FLOAT, vector_length);
1913 
1914   // Check vector elements of src1 and src2 for NaN.
1915   vmfeq_vv(tmp1, src1, src1);
1916   vmfeq_vv(tmp2, src2, src2);
1917 
1918   vmandn_mm(v0, vmask, tmp1);
1919   vfadd_vv(dst, src1, src1, Assembler::v0_t);
1920   vmandn_mm(v0, vmask, tmp2);
1921   vfadd_vv(dst, src2, src2, Assembler::v0_t);
1922 
1923   vmand_mm(tmp2, tmp1, tmp2);
1924   vmand_mm(v0, vmask, tmp2);
1925   is_min ? vfmin_vv(dst, src1, src2, Assembler::v0_t)
1926          : vfmax_vv(dst, src1, src2, Assembler::v0_t);
1927 }
1928 
1929 // Set dst to NaN if any NaN input.
1930 void C2_MacroAssembler::reduce_minmax_fp_v(FloatRegister dst,
1931                                            FloatRegister src1, VectorRegister src2,
1932                                            VectorRegister tmp1, VectorRegister tmp2,
1933                                            bool is_double, bool is_min, int vector_length, VectorMask vm) {
1934   assert_different_registers(dst, src1);
1935   assert_different_registers(src2, tmp1, tmp2);
1936 
1937   Label L_done, L_NaN_1, L_NaN_2;
1938   // Set dst to src1 if src1 is NaN
1939   is_double ? feq_d(t0, src1, src1)
1940             : feq_s(t0, src1, src1);
1941   beqz(t0, L_NaN_2);
1942 
1943   vsetvli_helper(is_double ? T_DOUBLE : T_FLOAT, vector_length);
1944   vfmv_s_f(tmp2, src1);
1945 
1946   is_min ? vfredmin_vs(tmp1, src2, tmp2, vm)
1947          : vfredmax_vs(tmp1, src2, tmp2, vm);
1948   vfmv_f_s(dst, tmp1);
1949 
1950   // Checking NaNs in src2
1951   vmfne_vv(tmp1, src2, src2, vm);
1952   vcpop_m(t0, tmp1, vm);
1953   beqz(t0, L_done);
1954 
1955   bind(L_NaN_1);
1956   vfredusum_vs(tmp1, src2, tmp2, vm);
1957   vfmv_f_s(dst, tmp1);
1958   j(L_done);
1959 
1960   bind(L_NaN_2);
1961   is_double ? fmv_d(dst, src1)
1962             : fmv_s(dst, src1);
1963   bind(L_done);
1964 }
1965 
1966 bool C2_MacroAssembler::in_scratch_emit_size() {
1967   if (ciEnv::current()->task() != nullptr) {
1968     PhaseOutput* phase_output = Compile::current()->output();
1969     if (phase_output != nullptr && phase_output->in_scratch_emit_size()) {
1970       return true;
1971     }
1972   }
1973   return MacroAssembler::in_scratch_emit_size();
1974 }
1975 
1976 void C2_MacroAssembler::reduce_integral_v(Register dst, Register src1,
1977                                           VectorRegister src2, VectorRegister tmp,
1978                                           int opc, BasicType bt, int vector_length, VectorMask vm) {
1979   assert(bt == T_BYTE || bt == T_SHORT || bt == T_INT || bt == T_LONG, "unsupported element type");
1980   vsetvli_helper(bt, vector_length);
1981   vmv_s_x(tmp, src1);
1982   switch (opc) {
1983     case Op_AddReductionVI:
1984     case Op_AddReductionVL:
1985       vredsum_vs(tmp, src2, tmp, vm);
1986       break;
1987     case Op_AndReductionV:
1988       vredand_vs(tmp, src2, tmp, vm);
1989       break;
1990     case Op_OrReductionV:
1991       vredor_vs(tmp, src2, tmp, vm);
1992       break;
1993     case Op_XorReductionV:
1994       vredxor_vs(tmp, src2, tmp, vm);
1995       break;
1996     case Op_MaxReductionV:
1997       vredmax_vs(tmp, src2, tmp, vm);
1998       break;
1999     case Op_MinReductionV:
2000       vredmin_vs(tmp, src2, tmp, vm);
2001       break;
2002     default:
2003       ShouldNotReachHere();
2004   }
2005   vmv_x_s(dst, tmp);
2006 }
2007 
2008 // Set vl and vtype for full and partial vector operations.
2009 // (vma = mu, vta = tu, vill = false)
2010 void C2_MacroAssembler::vsetvli_helper(BasicType bt, int vector_length, LMUL vlmul, Register tmp) {
2011   Assembler::SEW sew = Assembler::elemtype_to_sew(bt);
2012   if (vector_length <= 31) {
2013     vsetivli(tmp, vector_length, sew, vlmul);
2014   } else if (vector_length == (MaxVectorSize / type2aelembytes(bt))) {
2015     vsetvli(tmp, x0, sew, vlmul);
2016   } else {
2017     mv(tmp, vector_length);
2018     vsetvli(tmp, tmp, sew, vlmul);
2019   }
2020 }
2021 
2022 void C2_MacroAssembler::compare_integral_v(VectorRegister vd, VectorRegister src1, VectorRegister src2,
2023                                            int cond, BasicType bt, int vector_length, VectorMask vm) {
2024   assert(is_integral_type(bt), "unsupported element type");
2025   assert(vm == Assembler::v0_t ? vd != v0 : true, "should be different registers");
2026   vsetvli_helper(bt, vector_length);
2027   vmclr_m(vd);
2028   switch (cond) {
2029     case BoolTest::eq: vmseq_vv(vd, src1, src2, vm); break;
2030     case BoolTest::ne: vmsne_vv(vd, src1, src2, vm); break;
2031     case BoolTest::le: vmsle_vv(vd, src1, src2, vm); break;
2032     case BoolTest::ge: vmsge_vv(vd, src1, src2, vm); break;
2033     case BoolTest::lt: vmslt_vv(vd, src1, src2, vm); break;
2034     case BoolTest::gt: vmsgt_vv(vd, src1, src2, vm); break;
2035     default:
2036       assert(false, "unsupported compare condition");
2037       ShouldNotReachHere();
2038   }
2039 }
2040 
2041 void C2_MacroAssembler::compare_fp_v(VectorRegister vd, VectorRegister src1, VectorRegister src2,
2042                                      int cond, BasicType bt, int vector_length, VectorMask vm) {
2043   assert(is_floating_point_type(bt), "unsupported element type");
2044   assert(vm == Assembler::v0_t ? vd != v0 : true, "should be different registers");
2045   vsetvli_helper(bt, vector_length);
2046   vmclr_m(vd);
2047   switch (cond) {
2048     case BoolTest::eq: vmfeq_vv(vd, src1, src2, vm); break;
2049     case BoolTest::ne: vmfne_vv(vd, src1, src2, vm); break;
2050     case BoolTest::le: vmfle_vv(vd, src1, src2, vm); break;
2051     case BoolTest::ge: vmfge_vv(vd, src1, src2, vm); break;
2052     case BoolTest::lt: vmflt_vv(vd, src1, src2, vm); break;
2053     case BoolTest::gt: vmfgt_vv(vd, src1, src2, vm); break;
2054     default:
2055       assert(false, "unsupported compare condition");
2056       ShouldNotReachHere();
2057   }
2058 }
2059 
2060 void C2_MacroAssembler::integer_extend_v(VectorRegister dst, BasicType dst_bt, int vector_length,
2061                                          VectorRegister src, BasicType src_bt) {
2062   assert(type2aelembytes(dst_bt) > type2aelembytes(src_bt) && type2aelembytes(dst_bt) <= 8 && type2aelembytes(src_bt) <= 4, "invalid element size");
2063   assert(dst_bt != T_FLOAT && dst_bt != T_DOUBLE && src_bt != T_FLOAT && src_bt != T_DOUBLE, "unsupported element type");
2064   // https://github.com/riscv/riscv-v-spec/blob/master/v-spec.adoc#52-vector-operands
2065   // The destination EEW is greater than the source EEW, the source EMUL is at least 1,
2066   // and the overlap is in the highest-numbered part of the destination register group.
2067   // Since LMUL=1, vd and vs cannot be the same.
2068   assert_different_registers(dst, src);
2069 
2070   vsetvli_helper(dst_bt, vector_length);
2071   if (src_bt == T_BYTE) {
2072     switch (dst_bt) {
2073     case T_SHORT:
2074       vsext_vf2(dst, src);
2075       break;
2076     case T_INT:
2077       vsext_vf4(dst, src);
2078       break;
2079     case T_LONG:
2080       vsext_vf8(dst, src);
2081       break;
2082     default:
2083       ShouldNotReachHere();
2084     }
2085   } else if (src_bt == T_SHORT) {
2086     if (dst_bt == T_INT) {
2087       vsext_vf2(dst, src);
2088     } else {
2089       vsext_vf4(dst, src);
2090     }
2091   } else if (src_bt == T_INT) {
2092     vsext_vf2(dst, src);
2093   }
2094 }
2095 
2096 // Vector narrow from src to dst with specified element sizes.
2097 // High part of dst vector will be filled with zero.
2098 void C2_MacroAssembler::integer_narrow_v(VectorRegister dst, BasicType dst_bt, int vector_length,
2099                                          VectorRegister src, BasicType src_bt) {
2100   assert(type2aelembytes(dst_bt) < type2aelembytes(src_bt) && type2aelembytes(dst_bt) <= 4 && type2aelembytes(src_bt) <= 8, "invalid element size");
2101   assert(dst_bt != T_FLOAT && dst_bt != T_DOUBLE && src_bt != T_FLOAT && src_bt != T_DOUBLE, "unsupported element type");
2102   mv(t0, vector_length);
2103   if (src_bt == T_LONG) {
2104     // https://github.com/riscv/riscv-v-spec/blob/master/v-spec.adoc#117-vector-narrowing-integer-right-shift-instructions
2105     // Future extensions might add support for versions that narrow to a destination that is 1/4 the width of the source.
2106     // So we can currently only scale down by 1/2 the width at a time.
2107     vsetvli(t0, t0, Assembler::e32, Assembler::mf2);
2108     vncvt_x_x_w(dst, src);
2109     if (dst_bt == T_SHORT || dst_bt == T_BYTE) {
2110       vsetvli(t0, t0, Assembler::e16, Assembler::mf2);
2111       vncvt_x_x_w(dst, dst);
2112       if (dst_bt == T_BYTE) {
2113         vsetvli(t0, t0, Assembler::e8, Assembler::mf2);
2114         vncvt_x_x_w(dst, dst);
2115       }
2116     }
2117   } else if (src_bt == T_INT) {
2118     // T_SHORT
2119     vsetvli(t0, t0, Assembler::e16, Assembler::mf2);
2120     vncvt_x_x_w(dst, src);
2121     if (dst_bt == T_BYTE) {
2122       vsetvli(t0, t0, Assembler::e8, Assembler::mf2);
2123       vncvt_x_x_w(dst, dst);
2124     }
2125   } else if (src_bt == T_SHORT) {
2126     vsetvli(t0, t0, Assembler::e8, Assembler::mf2);
2127     vncvt_x_x_w(dst, src);
2128   }
2129 }
2130 
2131 #define VFCVT_SAFE(VFLOATCVT)                                                      \
2132 void C2_MacroAssembler::VFLOATCVT##_safe(VectorRegister dst, VectorRegister src) { \
2133   assert_different_registers(dst, src);                                            \
2134   vxor_vv(dst, dst, dst);                                                          \
2135   vmfeq_vv(v0, src, src);                                                          \
2136   VFLOATCVT(dst, src, Assembler::v0_t);                                            \
2137 }
2138 
2139 VFCVT_SAFE(vfcvt_rtz_x_f_v);
2140 
2141 #undef VFCVT_SAFE
2142 
2143 // Extract a scalar element from an vector at position 'idx'.
2144 // The input elements in src are expected to be of integral type.
2145 void C2_MacroAssembler::extract_v(Register dst, VectorRegister src, BasicType bt,
2146                                   int idx, VectorRegister tmp) {
2147   assert(is_integral_type(bt), "unsupported element type");
2148   assert(idx >= 0, "idx cannot be negative");
2149   // Only need the first element after vector slidedown
2150   vsetvli_helper(bt, 1);
2151   if (idx == 0) {
2152     vmv_x_s(dst, src);
2153   } else if (idx <= 31) {
2154     vslidedown_vi(tmp, src, idx);
2155     vmv_x_s(dst, tmp);
2156   } else {
2157     mv(t0, idx);
2158     vslidedown_vx(tmp, src, t0);
2159     vmv_x_s(dst, tmp);
2160   }
2161 }
2162 
2163 // Extract a scalar element from an vector at position 'idx'.
2164 // The input elements in src are expected to be of floating point type.
2165 void C2_MacroAssembler::extract_fp_v(FloatRegister dst, VectorRegister src, BasicType bt,
2166                                      int idx, VectorRegister tmp) {
2167   assert(is_floating_point_type(bt), "unsupported element type");
2168   assert(idx >= 0, "idx cannot be negative");
2169   // Only need the first element after vector slidedown
2170   vsetvli_helper(bt, 1);
2171   if (idx == 0) {
2172     vfmv_f_s(dst, src);
2173   } else if (idx <= 31) {
2174     vslidedown_vi(tmp, src, idx);
2175     vfmv_f_s(dst, tmp);
2176   } else {
2177     mv(t0, idx);
2178     vslidedown_vx(tmp, src, t0);
2179     vfmv_f_s(dst, tmp);
2180   }
2181 }
--- EOF ---