1 //===- RISCVVIntrinsicUtils.cpp - RISC-V Vector Intrinsic Utils -*- C++ -*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "clang/Support/RISCVVIntrinsicUtils.h" 10 #include "llvm/ADT/ArrayRef.h" 11 #include "llvm/ADT/SmallSet.h" 12 #include "llvm/ADT/StringExtras.h" 13 #include "llvm/ADT/StringSet.h" 14 #include "llvm/ADT/Twine.h" 15 #include "llvm/Support/ErrorHandling.h" 16 #include "llvm/Support/raw_ostream.h" 17 #include <numeric> 18 #include <optional> 19 20 using namespace llvm; 21 22 namespace clang { 23 namespace RISCV { 24 25 const PrototypeDescriptor PrototypeDescriptor::Mask = PrototypeDescriptor( 26 BaseTypeModifier::Vector, VectorTypeModifier::MaskVector); 27 const PrototypeDescriptor PrototypeDescriptor::VL = 28 PrototypeDescriptor(BaseTypeModifier::SizeT); 29 const PrototypeDescriptor PrototypeDescriptor::Vector = 30 PrototypeDescriptor(BaseTypeModifier::Vector); 31 32 //===----------------------------------------------------------------------===// 33 // Type implementation 34 //===----------------------------------------------------------------------===// 35 36 LMULType::LMULType(int NewLog2LMUL) { 37 // Check Log2LMUL is -3, -2, -1, 0, 1, 2, 3 38 assert(NewLog2LMUL <= 3 && NewLog2LMUL >= -3 && "Bad LMUL number!"); 39 Log2LMUL = NewLog2LMUL; 40 } 41 42 std::string LMULType::str() const { 43 if (Log2LMUL < 0) 44 return "mf" + utostr(1ULL << (-Log2LMUL)); 45 return "m" + utostr(1ULL << Log2LMUL); 46 } 47 48 VScaleVal LMULType::getScale(unsigned ElementBitwidth) const { 49 int Log2ScaleResult = 0; 50 switch (ElementBitwidth) { 51 default: 52 break; 53 case 8: 54 Log2ScaleResult = Log2LMUL + 3; 55 break; 56 case 16: 57 Log2ScaleResult = Log2LMUL + 2; 58 break; 59 case 32: 60 Log2ScaleResult = Log2LMUL + 1; 61 break; 62 case 64: 63 Log2ScaleResult = Log2LMUL; 64 break; 65 } 66 // Illegal vscale result would be less than 1 67 if (Log2ScaleResult < 0) 68 return std::nullopt; 69 return 1 << Log2ScaleResult; 70 } 71 72 void LMULType::MulLog2LMUL(int log2LMUL) { Log2LMUL += log2LMUL; } 73 74 RVVType::RVVType(BasicType BT, int Log2LMUL, 75 const PrototypeDescriptor &prototype) 76 : BT(BT), LMUL(LMULType(Log2LMUL)) { 77 applyBasicType(); 78 applyModifier(prototype); 79 Valid = verifyType(); 80 if (Valid) { 81 initBuiltinStr(); 82 initTypeStr(); 83 if (isVector()) { 84 initClangBuiltinStr(); 85 } 86 } 87 } 88 89 // clang-format off 90 // boolean type are encoded the ratio of n (SEW/LMUL) 91 // SEW/LMUL | 1 | 2 | 4 | 8 | 16 | 32 | 64 92 // c type | vbool64_t | vbool32_t | vbool16_t | vbool8_t | vbool4_t | vbool2_t | vbool1_t 93 // IR type | nxv1i1 | nxv2i1 | nxv4i1 | nxv8i1 | nxv16i1 | nxv32i1 | nxv64i1 94 95 // type\lmul | 1/8 | 1/4 | 1/2 | 1 | 2 | 4 | 8 96 // -------- |------ | -------- | ------- | ------- | -------- | -------- | -------- 97 // i64 | N/A | N/A | N/A | nxv1i64 | nxv2i64 | nxv4i64 | nxv8i64 98 // i32 | N/A | N/A | nxv1i32 | nxv2i32 | nxv4i32 | nxv8i32 | nxv16i32 99 // i16 | N/A | nxv1i16 | nxv2i16 | nxv4i16 | nxv8i16 | nxv16i16 | nxv32i16 100 // i8 | nxv1i8 | nxv2i8 | nxv4i8 | nxv8i8 | nxv16i8 | nxv32i8 | nxv64i8 101 // double | N/A | N/A | N/A | nxv1f64 | nxv2f64 | nxv4f64 | nxv8f64 102 // float | N/A | N/A | nxv1f32 | nxv2f32 | nxv4f32 | nxv8f32 | nxv16f32 103 // half | N/A | nxv1f16 | nxv2f16 | nxv4f16 | nxv8f16 | nxv16f16 | nxv32f16 104 // bfloat16 | N/A | nxv1bf16 | nxv2bf16| nxv4bf16| nxv8bf16 | nxv16bf16| nxv32bf16 105 // clang-format on 106 107 bool RVVType::verifyType() const { 108 if (ScalarType == Invalid) 109 return false; 110 if (isScalar()) 111 return true; 112 if (!Scale) 113 return false; 114 if (isFloat() && ElementBitwidth == 8) 115 return false; 116 if (isBFloat() && ElementBitwidth != 16) 117 return false; 118 if (IsTuple && (NF == 1 || NF > 8)) 119 return false; 120 if (IsTuple && (1 << std::max(0, LMUL.Log2LMUL)) * NF > 8) 121 return false; 122 unsigned V = *Scale; 123 switch (ElementBitwidth) { 124 case 1: 125 case 8: 126 // Check Scale is 1,2,4,8,16,32,64 127 return (V <= 64 && isPowerOf2_32(V)); 128 case 16: 129 // Check Scale is 1,2,4,8,16,32 130 return (V <= 32 && isPowerOf2_32(V)); 131 case 32: 132 // Check Scale is 1,2,4,8,16 133 return (V <= 16 && isPowerOf2_32(V)); 134 case 64: 135 // Check Scale is 1,2,4,8 136 return (V <= 8 && isPowerOf2_32(V)); 137 } 138 return false; 139 } 140 141 void RVVType::initBuiltinStr() { 142 assert(isValid() && "RVVType is invalid"); 143 switch (ScalarType) { 144 case ScalarTypeKind::Void: 145 BuiltinStr = "v"; 146 return; 147 case ScalarTypeKind::Size_t: 148 BuiltinStr = "z"; 149 if (IsImmediate) 150 BuiltinStr = "I" + BuiltinStr; 151 if (IsPointer) 152 BuiltinStr += "*"; 153 return; 154 case ScalarTypeKind::Ptrdiff_t: 155 BuiltinStr = "Y"; 156 return; 157 case ScalarTypeKind::UnsignedLong: 158 BuiltinStr = "ULi"; 159 return; 160 case ScalarTypeKind::SignedLong: 161 BuiltinStr = "Li"; 162 return; 163 case ScalarTypeKind::Boolean: 164 assert(ElementBitwidth == 1); 165 BuiltinStr += "b"; 166 break; 167 case ScalarTypeKind::SignedInteger: 168 case ScalarTypeKind::UnsignedInteger: 169 switch (ElementBitwidth) { 170 case 8: 171 BuiltinStr += "c"; 172 break; 173 case 16: 174 BuiltinStr += "s"; 175 break; 176 case 32: 177 BuiltinStr += "i"; 178 break; 179 case 64: 180 BuiltinStr += "Wi"; 181 break; 182 default: 183 llvm_unreachable("Unhandled ElementBitwidth!"); 184 } 185 if (isSignedInteger()) 186 BuiltinStr = "S" + BuiltinStr; 187 else 188 BuiltinStr = "U" + BuiltinStr; 189 break; 190 case ScalarTypeKind::Float: 191 switch (ElementBitwidth) { 192 case 16: 193 BuiltinStr += "x"; 194 break; 195 case 32: 196 BuiltinStr += "f"; 197 break; 198 case 64: 199 BuiltinStr += "d"; 200 break; 201 default: 202 llvm_unreachable("Unhandled ElementBitwidth!"); 203 } 204 break; 205 case ScalarTypeKind::BFloat: 206 BuiltinStr += "y"; 207 break; 208 default: 209 llvm_unreachable("ScalarType is invalid!"); 210 } 211 if (IsImmediate) 212 BuiltinStr = "I" + BuiltinStr; 213 if (isScalar()) { 214 if (IsConstant) 215 BuiltinStr += "C"; 216 if (IsPointer) 217 BuiltinStr += "*"; 218 return; 219 } 220 BuiltinStr = "q" + utostr(*Scale) + BuiltinStr; 221 // Pointer to vector types. Defined for segment load intrinsics. 222 // segment load intrinsics have pointer type arguments to store the loaded 223 // vector values. 224 if (IsPointer) 225 BuiltinStr += "*"; 226 227 if (IsTuple) 228 BuiltinStr = "T" + utostr(NF) + BuiltinStr; 229 } 230 231 void RVVType::initClangBuiltinStr() { 232 assert(isValid() && "RVVType is invalid"); 233 assert(isVector() && "Handle Vector type only"); 234 235 ClangBuiltinStr = "__rvv_"; 236 switch (ScalarType) { 237 case ScalarTypeKind::Boolean: 238 ClangBuiltinStr += "bool" + utostr(64 / *Scale) + "_t"; 239 return; 240 case ScalarTypeKind::Float: 241 ClangBuiltinStr += "float"; 242 break; 243 case ScalarTypeKind::BFloat: 244 ClangBuiltinStr += "bfloat"; 245 break; 246 case ScalarTypeKind::SignedInteger: 247 ClangBuiltinStr += "int"; 248 break; 249 case ScalarTypeKind::UnsignedInteger: 250 ClangBuiltinStr += "uint"; 251 break; 252 default: 253 llvm_unreachable("ScalarTypeKind is invalid"); 254 } 255 ClangBuiltinStr += utostr(ElementBitwidth) + LMUL.str() + 256 (IsTuple ? "x" + utostr(NF) : "") + "_t"; 257 } 258 259 void RVVType::initTypeStr() { 260 assert(isValid() && "RVVType is invalid"); 261 262 if (IsConstant) 263 Str += "const "; 264 265 auto getTypeString = [&](StringRef TypeStr) { 266 if (isScalar()) 267 return Twine(TypeStr + Twine(ElementBitwidth) + "_t").str(); 268 return Twine("v" + TypeStr + Twine(ElementBitwidth) + LMUL.str() + 269 (IsTuple ? "x" + utostr(NF) : "") + "_t") 270 .str(); 271 }; 272 273 switch (ScalarType) { 274 case ScalarTypeKind::Void: 275 Str = "void"; 276 return; 277 case ScalarTypeKind::Size_t: 278 Str = "size_t"; 279 if (IsPointer) 280 Str += " *"; 281 return; 282 case ScalarTypeKind::Ptrdiff_t: 283 Str = "ptrdiff_t"; 284 return; 285 case ScalarTypeKind::UnsignedLong: 286 Str = "unsigned long"; 287 return; 288 case ScalarTypeKind::SignedLong: 289 Str = "long"; 290 return; 291 case ScalarTypeKind::Boolean: 292 if (isScalar()) 293 Str += "bool"; 294 else 295 // Vector bool is special case, the formulate is 296 // `vbool<N>_t = MVT::nxv<64/N>i1` ex. vbool16_t = MVT::4i1 297 Str += "vbool" + utostr(64 / *Scale) + "_t"; 298 break; 299 case ScalarTypeKind::Float: 300 if (isScalar()) { 301 if (ElementBitwidth == 64) 302 Str += "double"; 303 else if (ElementBitwidth == 32) 304 Str += "float"; 305 else if (ElementBitwidth == 16) 306 Str += "_Float16"; 307 else 308 llvm_unreachable("Unhandled floating type."); 309 } else 310 Str += getTypeString("float"); 311 break; 312 case ScalarTypeKind::BFloat: 313 if (isScalar()) { 314 if (ElementBitwidth == 16) 315 Str += "__bf16"; 316 else 317 llvm_unreachable("Unhandled floating type."); 318 } else 319 Str += getTypeString("bfloat"); 320 break; 321 case ScalarTypeKind::SignedInteger: 322 Str += getTypeString("int"); 323 break; 324 case ScalarTypeKind::UnsignedInteger: 325 Str += getTypeString("uint"); 326 break; 327 default: 328 llvm_unreachable("ScalarType is invalid!"); 329 } 330 if (IsPointer) 331 Str += " *"; 332 } 333 334 void RVVType::initShortStr() { 335 switch (ScalarType) { 336 case ScalarTypeKind::Boolean: 337 assert(isVector()); 338 ShortStr = "b" + utostr(64 / *Scale); 339 return; 340 case ScalarTypeKind::Float: 341 ShortStr = "f" + utostr(ElementBitwidth); 342 break; 343 case ScalarTypeKind::BFloat: 344 ShortStr = "bf" + utostr(ElementBitwidth); 345 break; 346 case ScalarTypeKind::SignedInteger: 347 ShortStr = "i" + utostr(ElementBitwidth); 348 break; 349 case ScalarTypeKind::UnsignedInteger: 350 ShortStr = "u" + utostr(ElementBitwidth); 351 break; 352 default: 353 llvm_unreachable("Unhandled case!"); 354 } 355 if (isVector()) 356 ShortStr += LMUL.str(); 357 if (isTuple()) 358 ShortStr += "x" + utostr(NF); 359 } 360 361 static VectorTypeModifier getTupleVTM(unsigned NF) { 362 assert(2 <= NF && NF <= 8 && "2 <= NF <= 8"); 363 return static_cast<VectorTypeModifier>( 364 static_cast<uint8_t>(VectorTypeModifier::Tuple2) + (NF - 2)); 365 } 366 367 void RVVType::applyBasicType() { 368 switch (BT) { 369 case BasicType::Int8: 370 ElementBitwidth = 8; 371 ScalarType = ScalarTypeKind::SignedInteger; 372 break; 373 case BasicType::Int16: 374 ElementBitwidth = 16; 375 ScalarType = ScalarTypeKind::SignedInteger; 376 break; 377 case BasicType::Int32: 378 ElementBitwidth = 32; 379 ScalarType = ScalarTypeKind::SignedInteger; 380 break; 381 case BasicType::Int64: 382 ElementBitwidth = 64; 383 ScalarType = ScalarTypeKind::SignedInteger; 384 break; 385 case BasicType::Float16: 386 ElementBitwidth = 16; 387 ScalarType = ScalarTypeKind::Float; 388 break; 389 case BasicType::Float32: 390 ElementBitwidth = 32; 391 ScalarType = ScalarTypeKind::Float; 392 break; 393 case BasicType::Float64: 394 ElementBitwidth = 64; 395 ScalarType = ScalarTypeKind::Float; 396 break; 397 case BasicType::BFloat16: 398 ElementBitwidth = 16; 399 ScalarType = ScalarTypeKind::BFloat; 400 break; 401 default: 402 llvm_unreachable("Unhandled type code!"); 403 } 404 assert(ElementBitwidth != 0 && "Bad element bitwidth!"); 405 } 406 407 std::optional<PrototypeDescriptor> 408 PrototypeDescriptor::parsePrototypeDescriptor( 409 llvm::StringRef PrototypeDescriptorStr) { 410 PrototypeDescriptor PD; 411 BaseTypeModifier PT = BaseTypeModifier::Invalid; 412 VectorTypeModifier VTM = VectorTypeModifier::NoModifier; 413 414 if (PrototypeDescriptorStr.empty()) 415 return PD; 416 417 // Handle base type modifier 418 auto PType = PrototypeDescriptorStr.back(); 419 switch (PType) { 420 case 'e': 421 PT = BaseTypeModifier::Scalar; 422 break; 423 case 'v': 424 PT = BaseTypeModifier::Vector; 425 break; 426 case 'w': 427 PT = BaseTypeModifier::Vector; 428 VTM = VectorTypeModifier::Widening2XVector; 429 break; 430 case 'q': 431 PT = BaseTypeModifier::Vector; 432 VTM = VectorTypeModifier::Widening4XVector; 433 break; 434 case 'o': 435 PT = BaseTypeModifier::Vector; 436 VTM = VectorTypeModifier::Widening8XVector; 437 break; 438 case 'm': 439 PT = BaseTypeModifier::Vector; 440 VTM = VectorTypeModifier::MaskVector; 441 break; 442 case '0': 443 PT = BaseTypeModifier::Void; 444 break; 445 case 'z': 446 PT = BaseTypeModifier::SizeT; 447 break; 448 case 't': 449 PT = BaseTypeModifier::Ptrdiff; 450 break; 451 case 'u': 452 PT = BaseTypeModifier::UnsignedLong; 453 break; 454 case 'l': 455 PT = BaseTypeModifier::SignedLong; 456 break; 457 case 'f': 458 PT = BaseTypeModifier::Float32; 459 break; 460 default: 461 llvm_unreachable("Illegal primitive type transformers!"); 462 } 463 PD.PT = static_cast<uint8_t>(PT); 464 PrototypeDescriptorStr = PrototypeDescriptorStr.drop_back(); 465 466 // Compute the vector type transformers, it can only appear one time. 467 if (PrototypeDescriptorStr.starts_with("(")) { 468 assert(VTM == VectorTypeModifier::NoModifier && 469 "VectorTypeModifier should only have one modifier"); 470 size_t Idx = PrototypeDescriptorStr.find(')'); 471 assert(Idx != StringRef::npos); 472 StringRef ComplexType = PrototypeDescriptorStr.slice(1, Idx); 473 PrototypeDescriptorStr = PrototypeDescriptorStr.drop_front(Idx + 1); 474 assert(!PrototypeDescriptorStr.contains('(') && 475 "Only allow one vector type modifier"); 476 477 auto ComplexTT = ComplexType.split(":"); 478 if (ComplexTT.first == "Log2EEW") { 479 uint32_t Log2EEW; 480 if (ComplexTT.second.getAsInteger(10, Log2EEW)) { 481 llvm_unreachable("Invalid Log2EEW value!"); 482 return std::nullopt; 483 } 484 switch (Log2EEW) { 485 case 3: 486 VTM = VectorTypeModifier::Log2EEW3; 487 break; 488 case 4: 489 VTM = VectorTypeModifier::Log2EEW4; 490 break; 491 case 5: 492 VTM = VectorTypeModifier::Log2EEW5; 493 break; 494 case 6: 495 VTM = VectorTypeModifier::Log2EEW6; 496 break; 497 default: 498 llvm_unreachable("Invalid Log2EEW value, should be [3-6]"); 499 return std::nullopt; 500 } 501 } else if (ComplexTT.first == "FixedSEW") { 502 uint32_t NewSEW; 503 if (ComplexTT.second.getAsInteger(10, NewSEW)) { 504 llvm_unreachable("Invalid FixedSEW value!"); 505 return std::nullopt; 506 } 507 switch (NewSEW) { 508 case 8: 509 VTM = VectorTypeModifier::FixedSEW8; 510 break; 511 case 16: 512 VTM = VectorTypeModifier::FixedSEW16; 513 break; 514 case 32: 515 VTM = VectorTypeModifier::FixedSEW32; 516 break; 517 case 64: 518 VTM = VectorTypeModifier::FixedSEW64; 519 break; 520 default: 521 llvm_unreachable("Invalid FixedSEW value, should be 8, 16, 32 or 64"); 522 return std::nullopt; 523 } 524 } else if (ComplexTT.first == "LFixedLog2LMUL") { 525 int32_t Log2LMUL; 526 if (ComplexTT.second.getAsInteger(10, Log2LMUL)) { 527 llvm_unreachable("Invalid LFixedLog2LMUL value!"); 528 return std::nullopt; 529 } 530 switch (Log2LMUL) { 531 case -3: 532 VTM = VectorTypeModifier::LFixedLog2LMULN3; 533 break; 534 case -2: 535 VTM = VectorTypeModifier::LFixedLog2LMULN2; 536 break; 537 case -1: 538 VTM = VectorTypeModifier::LFixedLog2LMULN1; 539 break; 540 case 0: 541 VTM = VectorTypeModifier::LFixedLog2LMUL0; 542 break; 543 case 1: 544 VTM = VectorTypeModifier::LFixedLog2LMUL1; 545 break; 546 case 2: 547 VTM = VectorTypeModifier::LFixedLog2LMUL2; 548 break; 549 case 3: 550 VTM = VectorTypeModifier::LFixedLog2LMUL3; 551 break; 552 default: 553 llvm_unreachable("Invalid LFixedLog2LMUL value, should be [-3, 3]"); 554 return std::nullopt; 555 } 556 } else if (ComplexTT.first == "SFixedLog2LMUL") { 557 int32_t Log2LMUL; 558 if (ComplexTT.second.getAsInteger(10, Log2LMUL)) { 559 llvm_unreachable("Invalid SFixedLog2LMUL value!"); 560 return std::nullopt; 561 } 562 switch (Log2LMUL) { 563 case -3: 564 VTM = VectorTypeModifier::SFixedLog2LMULN3; 565 break; 566 case -2: 567 VTM = VectorTypeModifier::SFixedLog2LMULN2; 568 break; 569 case -1: 570 VTM = VectorTypeModifier::SFixedLog2LMULN1; 571 break; 572 case 0: 573 VTM = VectorTypeModifier::SFixedLog2LMUL0; 574 break; 575 case 1: 576 VTM = VectorTypeModifier::SFixedLog2LMUL1; 577 break; 578 case 2: 579 VTM = VectorTypeModifier::SFixedLog2LMUL2; 580 break; 581 case 3: 582 VTM = VectorTypeModifier::SFixedLog2LMUL3; 583 break; 584 default: 585 llvm_unreachable("Invalid LFixedLog2LMUL value, should be [-3, 3]"); 586 return std::nullopt; 587 } 588 589 } else if (ComplexTT.first == "SEFixedLog2LMUL") { 590 int32_t Log2LMUL; 591 if (ComplexTT.second.getAsInteger(10, Log2LMUL)) { 592 llvm_unreachable("Invalid SEFixedLog2LMUL value!"); 593 return std::nullopt; 594 } 595 switch (Log2LMUL) { 596 case -3: 597 VTM = VectorTypeModifier::SEFixedLog2LMULN3; 598 break; 599 case -2: 600 VTM = VectorTypeModifier::SEFixedLog2LMULN2; 601 break; 602 case -1: 603 VTM = VectorTypeModifier::SEFixedLog2LMULN1; 604 break; 605 case 0: 606 VTM = VectorTypeModifier::SEFixedLog2LMUL0; 607 break; 608 case 1: 609 VTM = VectorTypeModifier::SEFixedLog2LMUL1; 610 break; 611 case 2: 612 VTM = VectorTypeModifier::SEFixedLog2LMUL2; 613 break; 614 case 3: 615 VTM = VectorTypeModifier::SEFixedLog2LMUL3; 616 break; 617 default: 618 llvm_unreachable("Invalid LFixedLog2LMUL value, should be [-3, 3]"); 619 return std::nullopt; 620 } 621 } else if (ComplexTT.first == "Tuple") { 622 unsigned NF = 0; 623 if (ComplexTT.second.getAsInteger(10, NF)) { 624 llvm_unreachable("Invalid NF value!"); 625 return std::nullopt; 626 } 627 VTM = getTupleVTM(NF); 628 } else { 629 llvm_unreachable("Illegal complex type transformers!"); 630 } 631 } 632 PD.VTM = static_cast<uint8_t>(VTM); 633 634 // Compute the remain type transformers 635 TypeModifier TM = TypeModifier::NoModifier; 636 for (char I : PrototypeDescriptorStr) { 637 switch (I) { 638 case 'P': 639 if ((TM & TypeModifier::Const) == TypeModifier::Const) 640 llvm_unreachable("'P' transformer cannot be used after 'C'"); 641 if ((TM & TypeModifier::Pointer) == TypeModifier::Pointer) 642 llvm_unreachable("'P' transformer cannot be used twice"); 643 TM |= TypeModifier::Pointer; 644 break; 645 case 'C': 646 TM |= TypeModifier::Const; 647 break; 648 case 'K': 649 TM |= TypeModifier::Immediate; 650 break; 651 case 'U': 652 TM |= TypeModifier::UnsignedInteger; 653 break; 654 case 'I': 655 TM |= TypeModifier::SignedInteger; 656 break; 657 case 'F': 658 TM |= TypeModifier::Float; 659 break; 660 case 'S': 661 TM |= TypeModifier::LMUL1; 662 break; 663 default: 664 llvm_unreachable("Illegal non-primitive type transformer!"); 665 } 666 } 667 PD.TM = static_cast<uint8_t>(TM); 668 669 return PD; 670 } 671 672 void RVVType::applyModifier(const PrototypeDescriptor &Transformer) { 673 // Handle primitive type transformer 674 switch (static_cast<BaseTypeModifier>(Transformer.PT)) { 675 case BaseTypeModifier::Scalar: 676 Scale = 0; 677 break; 678 case BaseTypeModifier::Vector: 679 Scale = LMUL.getScale(ElementBitwidth); 680 break; 681 case BaseTypeModifier::Void: 682 ScalarType = ScalarTypeKind::Void; 683 break; 684 case BaseTypeModifier::SizeT: 685 ScalarType = ScalarTypeKind::Size_t; 686 break; 687 case BaseTypeModifier::Ptrdiff: 688 ScalarType = ScalarTypeKind::Ptrdiff_t; 689 break; 690 case BaseTypeModifier::UnsignedLong: 691 ScalarType = ScalarTypeKind::UnsignedLong; 692 break; 693 case BaseTypeModifier::SignedLong: 694 ScalarType = ScalarTypeKind::SignedLong; 695 break; 696 case BaseTypeModifier::Float32: 697 ElementBitwidth = 32; 698 ScalarType = ScalarTypeKind::Float; 699 break; 700 case BaseTypeModifier::Invalid: 701 ScalarType = ScalarTypeKind::Invalid; 702 return; 703 } 704 705 switch (static_cast<VectorTypeModifier>(Transformer.VTM)) { 706 case VectorTypeModifier::Widening2XVector: 707 ElementBitwidth *= 2; 708 LMUL.MulLog2LMUL(1); 709 Scale = LMUL.getScale(ElementBitwidth); 710 break; 711 case VectorTypeModifier::Widening4XVector: 712 ElementBitwidth *= 4; 713 LMUL.MulLog2LMUL(2); 714 Scale = LMUL.getScale(ElementBitwidth); 715 break; 716 case VectorTypeModifier::Widening8XVector: 717 ElementBitwidth *= 8; 718 LMUL.MulLog2LMUL(3); 719 Scale = LMUL.getScale(ElementBitwidth); 720 break; 721 case VectorTypeModifier::MaskVector: 722 ScalarType = ScalarTypeKind::Boolean; 723 Scale = LMUL.getScale(ElementBitwidth); 724 ElementBitwidth = 1; 725 break; 726 case VectorTypeModifier::Log2EEW3: 727 applyLog2EEW(3); 728 break; 729 case VectorTypeModifier::Log2EEW4: 730 applyLog2EEW(4); 731 break; 732 case VectorTypeModifier::Log2EEW5: 733 applyLog2EEW(5); 734 break; 735 case VectorTypeModifier::Log2EEW6: 736 applyLog2EEW(6); 737 break; 738 case VectorTypeModifier::FixedSEW8: 739 applyFixedSEW(8); 740 break; 741 case VectorTypeModifier::FixedSEW16: 742 applyFixedSEW(16); 743 break; 744 case VectorTypeModifier::FixedSEW32: 745 applyFixedSEW(32); 746 break; 747 case VectorTypeModifier::FixedSEW64: 748 applyFixedSEW(64); 749 break; 750 case VectorTypeModifier::LFixedLog2LMULN3: 751 applyFixedLog2LMUL(-3, FixedLMULType::LargerThan); 752 break; 753 case VectorTypeModifier::LFixedLog2LMULN2: 754 applyFixedLog2LMUL(-2, FixedLMULType::LargerThan); 755 break; 756 case VectorTypeModifier::LFixedLog2LMULN1: 757 applyFixedLog2LMUL(-1, FixedLMULType::LargerThan); 758 break; 759 case VectorTypeModifier::LFixedLog2LMUL0: 760 applyFixedLog2LMUL(0, FixedLMULType::LargerThan); 761 break; 762 case VectorTypeModifier::LFixedLog2LMUL1: 763 applyFixedLog2LMUL(1, FixedLMULType::LargerThan); 764 break; 765 case VectorTypeModifier::LFixedLog2LMUL2: 766 applyFixedLog2LMUL(2, FixedLMULType::LargerThan); 767 break; 768 case VectorTypeModifier::LFixedLog2LMUL3: 769 applyFixedLog2LMUL(3, FixedLMULType::LargerThan); 770 break; 771 case VectorTypeModifier::SFixedLog2LMULN3: 772 applyFixedLog2LMUL(-3, FixedLMULType::SmallerThan); 773 break; 774 case VectorTypeModifier::SFixedLog2LMULN2: 775 applyFixedLog2LMUL(-2, FixedLMULType::SmallerThan); 776 break; 777 case VectorTypeModifier::SFixedLog2LMULN1: 778 applyFixedLog2LMUL(-1, FixedLMULType::SmallerThan); 779 break; 780 case VectorTypeModifier::SFixedLog2LMUL0: 781 applyFixedLog2LMUL(0, FixedLMULType::SmallerThan); 782 break; 783 case VectorTypeModifier::SFixedLog2LMUL1: 784 applyFixedLog2LMUL(1, FixedLMULType::SmallerThan); 785 break; 786 case VectorTypeModifier::SFixedLog2LMUL2: 787 applyFixedLog2LMUL(2, FixedLMULType::SmallerThan); 788 break; 789 case VectorTypeModifier::SFixedLog2LMUL3: 790 applyFixedLog2LMUL(3, FixedLMULType::SmallerThan); 791 break; 792 case VectorTypeModifier::SEFixedLog2LMULN3: 793 applyFixedLog2LMUL(-3, FixedLMULType::SmallerOrEqual); 794 break; 795 case VectorTypeModifier::SEFixedLog2LMULN2: 796 applyFixedLog2LMUL(-2, FixedLMULType::SmallerOrEqual); 797 break; 798 case VectorTypeModifier::SEFixedLog2LMULN1: 799 applyFixedLog2LMUL(-1, FixedLMULType::SmallerOrEqual); 800 break; 801 case VectorTypeModifier::SEFixedLog2LMUL0: 802 applyFixedLog2LMUL(0, FixedLMULType::SmallerOrEqual); 803 break; 804 case VectorTypeModifier::SEFixedLog2LMUL1: 805 applyFixedLog2LMUL(1, FixedLMULType::SmallerOrEqual); 806 break; 807 case VectorTypeModifier::SEFixedLog2LMUL2: 808 applyFixedLog2LMUL(2, FixedLMULType::SmallerOrEqual); 809 break; 810 case VectorTypeModifier::SEFixedLog2LMUL3: 811 applyFixedLog2LMUL(3, FixedLMULType::SmallerOrEqual); 812 break; 813 case VectorTypeModifier::Tuple2: 814 case VectorTypeModifier::Tuple3: 815 case VectorTypeModifier::Tuple4: 816 case VectorTypeModifier::Tuple5: 817 case VectorTypeModifier::Tuple6: 818 case VectorTypeModifier::Tuple7: 819 case VectorTypeModifier::Tuple8: { 820 IsTuple = true; 821 NF = 2 + static_cast<uint8_t>(Transformer.VTM) - 822 static_cast<uint8_t>(VectorTypeModifier::Tuple2); 823 break; 824 } 825 case VectorTypeModifier::NoModifier: 826 break; 827 } 828 829 // Early return if the current type modifier is already invalid. 830 if (ScalarType == Invalid) 831 return; 832 833 for (unsigned TypeModifierMaskShift = 0; 834 TypeModifierMaskShift <= static_cast<unsigned>(TypeModifier::MaxOffset); 835 ++TypeModifierMaskShift) { 836 unsigned TypeModifierMask = 1 << TypeModifierMaskShift; 837 if ((static_cast<unsigned>(Transformer.TM) & TypeModifierMask) != 838 TypeModifierMask) 839 continue; 840 switch (static_cast<TypeModifier>(TypeModifierMask)) { 841 case TypeModifier::Pointer: 842 IsPointer = true; 843 break; 844 case TypeModifier::Const: 845 IsConstant = true; 846 break; 847 case TypeModifier::Immediate: 848 IsImmediate = true; 849 IsConstant = true; 850 break; 851 case TypeModifier::UnsignedInteger: 852 ScalarType = ScalarTypeKind::UnsignedInteger; 853 break; 854 case TypeModifier::SignedInteger: 855 ScalarType = ScalarTypeKind::SignedInteger; 856 break; 857 case TypeModifier::Float: 858 ScalarType = ScalarTypeKind::Float; 859 break; 860 case TypeModifier::BFloat: 861 ScalarType = ScalarTypeKind::BFloat; 862 break; 863 case TypeModifier::LMUL1: 864 LMUL = LMULType(0); 865 // Update ElementBitwidth need to update Scale too. 866 Scale = LMUL.getScale(ElementBitwidth); 867 break; 868 default: 869 llvm_unreachable("Unknown type modifier mask!"); 870 } 871 } 872 } 873 874 void RVVType::applyLog2EEW(unsigned Log2EEW) { 875 // update new elmul = (eew/sew) * lmul 876 LMUL.MulLog2LMUL(Log2EEW - Log2_32(ElementBitwidth)); 877 // update new eew 878 ElementBitwidth = 1 << Log2EEW; 879 ScalarType = ScalarTypeKind::SignedInteger; 880 Scale = LMUL.getScale(ElementBitwidth); 881 } 882 883 void RVVType::applyFixedSEW(unsigned NewSEW) { 884 // Set invalid type if src and dst SEW are same. 885 if (ElementBitwidth == NewSEW) { 886 ScalarType = ScalarTypeKind::Invalid; 887 return; 888 } 889 // Update new SEW 890 ElementBitwidth = NewSEW; 891 Scale = LMUL.getScale(ElementBitwidth); 892 } 893 894 void RVVType::applyFixedLog2LMUL(int Log2LMUL, enum FixedLMULType Type) { 895 switch (Type) { 896 case FixedLMULType::LargerThan: 897 if (Log2LMUL <= LMUL.Log2LMUL) { 898 ScalarType = ScalarTypeKind::Invalid; 899 return; 900 } 901 break; 902 case FixedLMULType::SmallerThan: 903 if (Log2LMUL >= LMUL.Log2LMUL) { 904 ScalarType = ScalarTypeKind::Invalid; 905 return; 906 } 907 break; 908 case FixedLMULType::SmallerOrEqual: 909 if (Log2LMUL > LMUL.Log2LMUL) { 910 ScalarType = ScalarTypeKind::Invalid; 911 return; 912 } 913 break; 914 } 915 916 // Update new LMUL 917 LMUL = LMULType(Log2LMUL); 918 Scale = LMUL.getScale(ElementBitwidth); 919 } 920 921 std::optional<RVVTypes> 922 RVVTypeCache::computeTypes(BasicType BT, int Log2LMUL, unsigned NF, 923 ArrayRef<PrototypeDescriptor> Prototype) { 924 RVVTypes Types; 925 for (const PrototypeDescriptor &Proto : Prototype) { 926 auto T = computeType(BT, Log2LMUL, Proto); 927 if (!T) 928 return std::nullopt; 929 // Record legal type index 930 Types.push_back(*T); 931 } 932 return Types; 933 } 934 935 // Compute the hash value of RVVType, used for cache the result of computeType. 936 static uint64_t computeRVVTypeHashValue(BasicType BT, int Log2LMUL, 937 PrototypeDescriptor Proto) { 938 // Layout of hash value: 939 // 0 8 16 24 32 40 940 // | Log2LMUL + 3 | BT | Proto.PT | Proto.TM | Proto.VTM | 941 assert(Log2LMUL >= -3 && Log2LMUL <= 3); 942 return (Log2LMUL + 3) | (static_cast<uint64_t>(BT) & 0xff) << 8 | 943 ((uint64_t)(Proto.PT & 0xff) << 16) | 944 ((uint64_t)(Proto.TM & 0xff) << 24) | 945 ((uint64_t)(Proto.VTM & 0xff) << 32); 946 } 947 948 std::optional<RVVTypePtr> RVVTypeCache::computeType(BasicType BT, int Log2LMUL, 949 PrototypeDescriptor Proto) { 950 uint64_t Idx = computeRVVTypeHashValue(BT, Log2LMUL, Proto); 951 // Search first 952 auto It = LegalTypes.find(Idx); 953 if (It != LegalTypes.end()) 954 return &(It->second); 955 956 if (IllegalTypes.count(Idx)) 957 return std::nullopt; 958 959 // Compute type and record the result. 960 RVVType T(BT, Log2LMUL, Proto); 961 if (T.isValid()) { 962 // Record legal type index and value. 963 std::pair<std::unordered_map<uint64_t, RVVType>::iterator, bool> 964 InsertResult = LegalTypes.insert({Idx, T}); 965 return &(InsertResult.first->second); 966 } 967 // Record illegal type index. 968 IllegalTypes.insert(Idx); 969 return std::nullopt; 970 } 971 972 //===----------------------------------------------------------------------===// 973 // RVVIntrinsic implementation 974 //===----------------------------------------------------------------------===// 975 RVVIntrinsic::RVVIntrinsic( 976 StringRef NewName, StringRef Suffix, StringRef NewOverloadedName, 977 StringRef OverloadedSuffix, StringRef IRName, bool IsMasked, 978 bool HasMaskedOffOperand, bool HasVL, PolicyScheme Scheme, 979 bool SupportOverloading, bool HasBuiltinAlias, StringRef ManualCodegen, 980 const RVVTypes &OutInTypes, const std::vector<int64_t> &NewIntrinsicTypes, 981 unsigned NF, Policy NewPolicyAttrs, bool HasFRMRoundModeOp) 982 : IRName(IRName), IsMasked(IsMasked), 983 HasMaskedOffOperand(HasMaskedOffOperand), HasVL(HasVL), Scheme(Scheme), 984 SupportOverloading(SupportOverloading), HasBuiltinAlias(HasBuiltinAlias), 985 ManualCodegen(ManualCodegen.str()), NF(NF), PolicyAttrs(NewPolicyAttrs) { 986 987 // Init BuiltinName, Name and OverloadedName 988 BuiltinName = NewName.str(); 989 Name = BuiltinName; 990 if (NewOverloadedName.empty()) 991 OverloadedName = NewName.split("_").first.str(); 992 else 993 OverloadedName = NewOverloadedName.str(); 994 if (!Suffix.empty()) 995 Name += "_" + Suffix.str(); 996 if (!OverloadedSuffix.empty()) 997 OverloadedName += "_" + OverloadedSuffix.str(); 998 999 updateNamesAndPolicy(IsMasked, hasPolicy(), Name, BuiltinName, OverloadedName, 1000 PolicyAttrs, HasFRMRoundModeOp); 1001 1002 // Init OutputType and InputTypes 1003 OutputType = OutInTypes[0]; 1004 InputTypes.assign(OutInTypes.begin() + 1, OutInTypes.end()); 1005 1006 // IntrinsicTypes is unmasked TA version index. Need to update it 1007 // if there is merge operand (It is always in first operand). 1008 IntrinsicTypes = NewIntrinsicTypes; 1009 if ((IsMasked && hasMaskedOffOperand()) || 1010 (!IsMasked && hasPassthruOperand())) { 1011 for (auto &I : IntrinsicTypes) { 1012 if (I >= 0) 1013 I += NF; 1014 } 1015 } 1016 } 1017 1018 std::string RVVIntrinsic::getBuiltinTypeStr() const { 1019 std::string S; 1020 S += OutputType->getBuiltinStr(); 1021 for (const auto &T : InputTypes) { 1022 S += T->getBuiltinStr(); 1023 } 1024 return S; 1025 } 1026 1027 std::string RVVIntrinsic::getSuffixStr( 1028 RVVTypeCache &TypeCache, BasicType Type, int Log2LMUL, 1029 llvm::ArrayRef<PrototypeDescriptor> PrototypeDescriptors) { 1030 SmallVector<std::string> SuffixStrs; 1031 for (auto PD : PrototypeDescriptors) { 1032 auto T = TypeCache.computeType(Type, Log2LMUL, PD); 1033 SuffixStrs.push_back((*T)->getShortStr()); 1034 } 1035 return join(SuffixStrs, "_"); 1036 } 1037 1038 llvm::SmallVector<PrototypeDescriptor> RVVIntrinsic::computeBuiltinTypes( 1039 llvm::ArrayRef<PrototypeDescriptor> Prototype, bool IsMasked, 1040 bool HasMaskedOffOperand, bool HasVL, unsigned NF, 1041 PolicyScheme DefaultScheme, Policy PolicyAttrs, bool IsTuple) { 1042 SmallVector<PrototypeDescriptor> NewPrototype(Prototype.begin(), 1043 Prototype.end()); 1044 bool HasPassthruOp = DefaultScheme == PolicyScheme::HasPassthruOperand; 1045 if (IsMasked) { 1046 // If HasMaskedOffOperand, insert result type as first input operand if 1047 // need. 1048 if (HasMaskedOffOperand && !PolicyAttrs.isTAMAPolicy()) { 1049 if (NF == 1) { 1050 NewPrototype.insert(NewPrototype.begin() + 1, NewPrototype[0]); 1051 } else if (NF > 1) { 1052 if (IsTuple) { 1053 PrototypeDescriptor BasePtrOperand = Prototype[1]; 1054 PrototypeDescriptor MaskoffType = PrototypeDescriptor( 1055 static_cast<uint8_t>(BaseTypeModifier::Vector), 1056 static_cast<uint8_t>(getTupleVTM(NF)), 1057 BasePtrOperand.TM & ~static_cast<uint8_t>(TypeModifier::Pointer)); 1058 NewPrototype.insert(NewPrototype.begin() + 1, MaskoffType); 1059 } else { 1060 // Convert 1061 // (void, op0 address, op1 address, ...) 1062 // to 1063 // (void, op0 address, op1 address, ..., maskedoff0, maskedoff1, ...) 1064 PrototypeDescriptor MaskoffType = NewPrototype[1]; 1065 MaskoffType.TM &= ~static_cast<uint8_t>(TypeModifier::Pointer); 1066 NewPrototype.insert(NewPrototype.begin() + NF + 1, NF, MaskoffType); 1067 } 1068 } 1069 } 1070 if (HasMaskedOffOperand && NF > 1) { 1071 // Convert 1072 // (void, op0 address, op1 address, ..., maskedoff0, maskedoff1, ...) 1073 // to 1074 // (void, op0 address, op1 address, ..., mask, maskedoff0, maskedoff1, 1075 // ...) 1076 if (IsTuple) 1077 NewPrototype.insert(NewPrototype.begin() + 1, 1078 PrototypeDescriptor::Mask); 1079 else 1080 NewPrototype.insert(NewPrototype.begin() + NF + 1, 1081 PrototypeDescriptor::Mask); 1082 } else { 1083 // If IsMasked, insert PrototypeDescriptor:Mask as first input operand. 1084 NewPrototype.insert(NewPrototype.begin() + 1, PrototypeDescriptor::Mask); 1085 } 1086 } else { 1087 if (NF == 1) { 1088 if (PolicyAttrs.isTUPolicy() && HasPassthruOp) 1089 NewPrototype.insert(NewPrototype.begin(), NewPrototype[0]); 1090 } else if (PolicyAttrs.isTUPolicy() && HasPassthruOp) { 1091 if (IsTuple) { 1092 PrototypeDescriptor BasePtrOperand = Prototype[0]; 1093 PrototypeDescriptor MaskoffType = PrototypeDescriptor( 1094 static_cast<uint8_t>(BaseTypeModifier::Vector), 1095 static_cast<uint8_t>(getTupleVTM(NF)), 1096 BasePtrOperand.TM & ~static_cast<uint8_t>(TypeModifier::Pointer)); 1097 NewPrototype.insert(NewPrototype.begin(), MaskoffType); 1098 } else { 1099 // NF > 1 cases for segment load operations. 1100 // Convert 1101 // (void, op0 address, op1 address, ...) 1102 // to 1103 // (void, op0 address, op1 address, maskedoff0, maskedoff1, ...) 1104 PrototypeDescriptor MaskoffType = Prototype[1]; 1105 MaskoffType.TM &= ~static_cast<uint8_t>(TypeModifier::Pointer); 1106 NewPrototype.insert(NewPrototype.begin() + NF + 1, NF, MaskoffType); 1107 } 1108 } 1109 } 1110 1111 // If HasVL, append PrototypeDescriptor:VL to last operand 1112 if (HasVL) 1113 NewPrototype.push_back(PrototypeDescriptor::VL); 1114 1115 return NewPrototype; 1116 } 1117 1118 llvm::SmallVector<Policy> RVVIntrinsic::getSupportedUnMaskedPolicies() { 1119 return {Policy(Policy::PolicyType::Undisturbed)}; // TU 1120 } 1121 1122 llvm::SmallVector<Policy> 1123 RVVIntrinsic::getSupportedMaskedPolicies(bool HasTailPolicy, 1124 bool HasMaskPolicy) { 1125 if (HasTailPolicy && HasMaskPolicy) 1126 return {Policy(Policy::PolicyType::Undisturbed, 1127 Policy::PolicyType::Agnostic), // TUM 1128 Policy(Policy::PolicyType::Undisturbed, 1129 Policy::PolicyType::Undisturbed), // TUMU 1130 Policy(Policy::PolicyType::Agnostic, 1131 Policy::PolicyType::Undisturbed)}; // MU 1132 if (HasTailPolicy && !HasMaskPolicy) 1133 return {Policy(Policy::PolicyType::Undisturbed, 1134 Policy::PolicyType::Agnostic)}; // TU 1135 if (!HasTailPolicy && HasMaskPolicy) 1136 return {Policy(Policy::PolicyType::Agnostic, 1137 Policy::PolicyType::Undisturbed)}; // MU 1138 llvm_unreachable("An RVV instruction should not be without both tail policy " 1139 "and mask policy"); 1140 } 1141 1142 void RVVIntrinsic::updateNamesAndPolicy( 1143 bool IsMasked, bool HasPolicy, std::string &Name, std::string &BuiltinName, 1144 std::string &OverloadedName, Policy &PolicyAttrs, bool HasFRMRoundModeOp) { 1145 1146 auto appendPolicySuffix = [&](const std::string &suffix) { 1147 Name += suffix; 1148 BuiltinName += suffix; 1149 OverloadedName += suffix; 1150 }; 1151 1152 if (HasFRMRoundModeOp) { 1153 Name += "_rm"; 1154 BuiltinName += "_rm"; 1155 } 1156 1157 if (IsMasked) { 1158 if (PolicyAttrs.isTUMUPolicy()) 1159 appendPolicySuffix("_tumu"); 1160 else if (PolicyAttrs.isTUMAPolicy()) 1161 appendPolicySuffix("_tum"); 1162 else if (PolicyAttrs.isTAMUPolicy()) 1163 appendPolicySuffix("_mu"); 1164 else if (PolicyAttrs.isTAMAPolicy()) { 1165 Name += "_m"; 1166 BuiltinName += "_m"; 1167 } else 1168 llvm_unreachable("Unhandled policy condition"); 1169 } else { 1170 if (PolicyAttrs.isTUPolicy()) 1171 appendPolicySuffix("_tu"); 1172 else if (PolicyAttrs.isTAPolicy()) // no suffix needed 1173 return; 1174 else 1175 llvm_unreachable("Unhandled policy condition"); 1176 } 1177 } 1178 1179 SmallVector<PrototypeDescriptor> parsePrototypes(StringRef Prototypes) { 1180 SmallVector<PrototypeDescriptor> PrototypeDescriptors; 1181 const StringRef Primaries("evwqom0ztulf"); 1182 while (!Prototypes.empty()) { 1183 size_t Idx = 0; 1184 // Skip over complex prototype because it could contain primitive type 1185 // character. 1186 if (Prototypes[0] == '(') 1187 Idx = Prototypes.find_first_of(')'); 1188 Idx = Prototypes.find_first_of(Primaries, Idx); 1189 assert(Idx != StringRef::npos); 1190 auto PD = PrototypeDescriptor::parsePrototypeDescriptor( 1191 Prototypes.slice(0, Idx + 1)); 1192 if (!PD) 1193 llvm_unreachable("Error during parsing prototype."); 1194 PrototypeDescriptors.push_back(*PD); 1195 Prototypes = Prototypes.drop_front(Idx + 1); 1196 } 1197 return PrototypeDescriptors; 1198 } 1199 1200 raw_ostream &operator<<(raw_ostream &OS, const RVVIntrinsicRecord &Record) { 1201 OS << "{"; 1202 OS << "\"" << Record.Name << "\","; 1203 if (Record.OverloadedName == nullptr || 1204 StringRef(Record.OverloadedName).empty()) 1205 OS << "nullptr,"; 1206 else 1207 OS << "\"" << Record.OverloadedName << "\","; 1208 OS << Record.PrototypeIndex << ","; 1209 OS << Record.SuffixIndex << ","; 1210 OS << Record.OverloadedSuffixIndex << ","; 1211 OS << (int)Record.PrototypeLength << ","; 1212 OS << (int)Record.SuffixLength << ","; 1213 OS << (int)Record.OverloadedSuffixSize << ","; 1214 OS << Record.RequiredExtensions << ","; 1215 OS << (int)Record.TypeRangeMask << ","; 1216 OS << (int)Record.Log2LMULMask << ","; 1217 OS << (int)Record.NF << ","; 1218 OS << (int)Record.HasMasked << ","; 1219 OS << (int)Record.HasVL << ","; 1220 OS << (int)Record.HasMaskedOffOperand << ","; 1221 OS << (int)Record.HasTailPolicy << ","; 1222 OS << (int)Record.HasMaskPolicy << ","; 1223 OS << (int)Record.HasFRMRoundModeOp << ","; 1224 OS << (int)Record.IsTuple << ","; 1225 OS << (int)Record.UnMaskedPolicyScheme << ","; 1226 OS << (int)Record.MaskedPolicyScheme << ","; 1227 OS << "},\n"; 1228 return OS; 1229 } 1230 1231 } // end namespace RISCV 1232 } // end namespace clang 1233