1 //===--- RISCV.cpp - RISCV Helpers for Tools --------------------*- 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 "RISCV.h" 10 #include "clang/Basic/CharInfo.h" 11 #include "clang/Driver/Driver.h" 12 #include "clang/Driver/DriverDiagnostic.h" 13 #include "clang/Driver/Options.h" 14 #include "llvm/Option/ArgList.h" 15 #include "llvm/ADT/Optional.h" 16 #include "llvm/Support/TargetParser.h" 17 #include "llvm/Support/raw_ostream.h" 18 #include "ToolChains/CommonArgs.h" 19 20 using namespace clang::driver; 21 using namespace clang::driver::tools; 22 using namespace clang; 23 using namespace llvm::opt; 24 25 static StringRef getExtensionTypeDesc(StringRef Ext) { 26 if (Ext.startswith("sx")) 27 return "non-standard supervisor-level extension"; 28 if (Ext.startswith("s")) 29 return "standard supervisor-level extension"; 30 if (Ext.startswith("x")) 31 return "non-standard user-level extension"; 32 return StringRef(); 33 } 34 35 static StringRef getExtensionType(StringRef Ext) { 36 if (Ext.startswith("sx")) 37 return "sx"; 38 if (Ext.startswith("s")) 39 return "s"; 40 if (Ext.startswith("x")) 41 return "x"; 42 return StringRef(); 43 } 44 45 static bool isSupportedExtension(StringRef Ext) { 46 // LLVM does not support "sx", "s" nor "x" extensions. 47 return false; 48 } 49 50 // Extensions may have a version number, and may be separated by 51 // an underscore '_' e.g.: rv32i2_m2. 52 // Version number is divided into major and minor version numbers, 53 // separated by a 'p'. If the minor version is 0 then 'p0' can be 54 // omitted from the version string. E.g., rv32i2p0, rv32i2, rv32i2p1. 55 static bool getExtensionVersion(const Driver &D, StringRef MArch, 56 StringRef Ext, StringRef In, 57 std::string &Major, std::string &Minor) { 58 Major = In.take_while(isDigit); 59 In = In.substr(Major.size()); 60 if (Major.empty()) 61 return true; 62 63 if (In.consume_front("p")) { 64 Minor = In.take_while(isDigit); 65 In = In.substr(Major.size()); 66 67 // Expected 'p' to be followed by minor version number. 68 if (Minor.empty()) { 69 std::string Error = 70 "minor version number missing after 'p' for extension"; 71 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 72 << MArch << Error << Ext; 73 return false; 74 } 75 } 76 77 // TODO: Handle extensions with version number. 78 std::string Error = "unsupported version number " + Major; 79 if (!Minor.empty()) 80 Error += "." + Minor; 81 Error += " for extension"; 82 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) << MArch << Error << Ext; 83 84 return false; 85 } 86 87 // Handle other types of extensions other than the standard 88 // general purpose and standard user-level extensions. 89 // Parse the ISA string containing non-standard user-level 90 // extensions, standard supervisor-level extensions and 91 // non-standard supervisor-level extensions. 92 // These extensions start with 'x', 's', 'sx' prefixes, follow a 93 // canonical order, might have a version number (major, minor) 94 // and are separated by a single underscore '_'. 95 // Set the hardware features for the extensions that are supported. 96 static void getExtensionFeatures(const Driver &D, 97 const ArgList &Args, 98 std::vector<StringRef> &Features, 99 StringRef &MArch, StringRef &Exts) { 100 if (Exts.empty()) 101 return; 102 103 // Multi-letter extensions are seperated by a single underscore 104 // as described in RISC-V User-Level ISA V2.2. 105 SmallVector<StringRef, 8> Split; 106 Exts.split(Split, StringRef("_")); 107 108 SmallVector<StringRef, 3> Prefix{"x", "s", "sx"}; 109 auto I = Prefix.begin(); 110 auto E = Prefix.end(); 111 112 SmallVector<StringRef, 8> AllExts; 113 114 for (StringRef Ext : Split) { 115 if (Ext.empty()) { 116 D.Diag(diag::err_drv_invalid_riscv_arch_name) << MArch 117 << "extension name missing after separator '_'"; 118 return; 119 } 120 121 StringRef Type = getExtensionType(Ext); 122 StringRef Name(Ext.substr(Type.size())); 123 StringRef Desc = getExtensionTypeDesc(Ext); 124 125 if (Type.empty()) { 126 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 127 << MArch << "invalid extension prefix" << Ext; 128 return; 129 } 130 131 // Check ISA extensions are specified in the canonical order. 132 while (I != E && *I != Type) 133 ++I; 134 135 if (I == E) { 136 std::string Error = Desc; 137 Error += " not given in canonical order"; 138 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 139 << MArch << Error << Ext; 140 return; 141 } 142 143 // The order is OK, do not advance I to the next prefix 144 // to allow repeated extension type, e.g.: rv32ixabc_xdef. 145 146 if (Name.empty()) { 147 std::string Error = Desc; 148 Error += " name missing after"; 149 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 150 << MArch << Error << Ext; 151 return; 152 } 153 154 std::string Major, Minor; 155 auto Pos = Name.find_if(isDigit); 156 if (Pos != StringRef::npos) { 157 auto Next = Name.substr(Pos); 158 Name = Name.substr(0, Pos); 159 if (!getExtensionVersion(D, MArch, Ext, Next, Major, Minor)) 160 return; 161 } 162 163 // Check if duplicated extension. 164 if (llvm::is_contained(AllExts, Ext)) { 165 std::string Error = "duplicated "; 166 Error += Desc; 167 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 168 << MArch << Error << Ext; 169 return; 170 } 171 172 // Extension format is correct, keep parsing the extensions. 173 // TODO: Save Type, Name, Major, Minor to avoid parsing them later. 174 AllExts.push_back(Ext); 175 } 176 177 // Set target features. 178 // TODO: Hardware features to be handled in Support/TargetParser.cpp. 179 // TODO: Use version number when setting target features. 180 for (auto Ext : AllExts) { 181 if (!isSupportedExtension(Ext)) { 182 StringRef Desc = getExtensionTypeDesc(getExtensionType(Ext)); 183 std::string Error = "unsupported "; 184 Error += Desc; 185 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 186 << MArch << Error << Ext; 187 return; 188 } 189 Features.push_back(Args.MakeArgString("+" + Ext)); 190 } 191 } 192 193 // Returns false if an error is diagnosed. 194 static bool getArchFeatures(const Driver &D, StringRef MArch, 195 std::vector<StringRef> &Features, 196 const ArgList &Args) { 197 // RISC-V ISA strings must be lowercase. 198 if (llvm::any_of(MArch, [](char c) { return isupper(c); })) { 199 D.Diag(diag::err_drv_invalid_riscv_arch_name) 200 << MArch << "string must be lowercase"; 201 return false; 202 } 203 204 // ISA string must begin with rv32 or rv64. 205 if (!(MArch.startswith("rv32") || MArch.startswith("rv64")) || 206 (MArch.size() < 5)) { 207 D.Diag(diag::err_drv_invalid_riscv_arch_name) 208 << MArch << "string must begin with rv32{i,e,g} or rv64{i,g}"; 209 return false; 210 } 211 212 bool HasRV64 = MArch.startswith("rv64"); 213 214 // The canonical order specified in ISA manual. 215 // Ref: Table 22.1 in RISC-V User-Level ISA V2.2 216 StringRef StdExts = "mafdqlcbjtpvn"; 217 bool HasF = false, HasD = false; 218 char Baseline = MArch[4]; 219 220 // First letter should be 'e', 'i' or 'g'. 221 switch (Baseline) { 222 default: 223 D.Diag(diag::err_drv_invalid_riscv_arch_name) 224 << MArch << "first letter should be 'e', 'i' or 'g'"; 225 return false; 226 case 'e': { 227 StringRef Error; 228 // Currently LLVM does not support 'e'. 229 // Extension 'e' is not allowed in rv64. 230 if (HasRV64) 231 Error = "standard user-level extension 'e' requires 'rv32'"; 232 else 233 Error = "unsupported standard user-level extension 'e'"; 234 D.Diag(diag::err_drv_invalid_riscv_arch_name) << MArch << Error; 235 return false; 236 } 237 case 'i': 238 break; 239 case 'g': 240 // g = imafd 241 StdExts = StdExts.drop_front(4); 242 Features.push_back("+m"); 243 Features.push_back("+a"); 244 Features.push_back("+f"); 245 Features.push_back("+d"); 246 HasF = true; 247 HasD = true; 248 break; 249 } 250 251 // Skip rvxxx 252 StringRef Exts = MArch.substr(5); 253 254 // Remove non-standard extensions and supervisor-level extensions. 255 // They have 'x', 's', 'sx' prefixes. Parse them at the end. 256 // Find the very first occurrence of 's' or 'x'. 257 StringRef OtherExts; 258 size_t Pos = Exts.find_first_of("sx"); 259 if (Pos != StringRef::npos) { 260 OtherExts = Exts.substr(Pos); 261 Exts = Exts.substr(0, Pos); 262 } 263 264 std::string Major, Minor; 265 if (!getExtensionVersion(D, MArch, std::string(1, Baseline), Exts, Major, 266 Minor)) 267 return false; 268 269 // TODO: Use version number when setting target features 270 // and consume the underscore '_' that might follow. 271 272 auto StdExtsItr = StdExts.begin(); 273 auto StdExtsEnd = StdExts.end(); 274 275 for (auto I = Exts.begin(), E = Exts.end(); I != E; ++I) { 276 char c = *I; 277 278 // Check ISA extensions are specified in the canonical order. 279 while (StdExtsItr != StdExtsEnd && *StdExtsItr != c) 280 ++StdExtsItr; 281 282 if (StdExtsItr == StdExtsEnd) { 283 // Either c contains a valid extension but it was not given in 284 // canonical order or it is an invalid extension. 285 StringRef Error; 286 if (StdExts.contains(c)) 287 Error = "standard user-level extension not given in canonical order"; 288 else 289 Error = "invalid standard user-level extension"; 290 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 291 << MArch << Error << std::string(1, c); 292 return false; 293 } 294 295 // Move to next char to prevent repeated letter. 296 ++StdExtsItr; 297 298 if (std::next(I) != E) { 299 // Skip c. 300 std::string Next = std::string(std::next(I), E); 301 std::string Major, Minor; 302 if (!getExtensionVersion(D, MArch, std::string(1, c), Next, Major, Minor)) 303 return false; 304 305 // TODO: Use version number when setting target features 306 // and consume the underscore '_' that might follow. 307 } 308 309 // The order is OK, then push it into features. 310 switch (c) { 311 default: 312 // Currently LLVM supports only "mafdc". 313 D.Diag(diag::err_drv_invalid_riscv_ext_arch_name) 314 << MArch << "unsupported standard user-level extension" 315 << std::string(1, c); 316 return false; 317 case 'm': 318 Features.push_back("+m"); 319 break; 320 case 'a': 321 Features.push_back("+a"); 322 break; 323 case 'f': 324 Features.push_back("+f"); 325 HasF = true; 326 break; 327 case 'd': 328 Features.push_back("+d"); 329 HasD = true; 330 break; 331 case 'c': 332 Features.push_back("+c"); 333 break; 334 } 335 } 336 337 // Dependency check. 338 // It's illegal to specify the 'd' (double-precision floating point) 339 // extension without also specifying the 'f' (single precision 340 // floating-point) extension. 341 if (HasD && !HasF) { 342 D.Diag(diag::err_drv_invalid_riscv_arch_name) 343 << MArch << "d requires f extension to also be specified"; 344 return false; 345 } 346 347 // Additional dependency checks. 348 // TODO: The 'q' extension requires rv64. 349 // TODO: It is illegal to specify 'e' extensions with 'f' and 'd'. 350 351 // Handle all other types of extensions. 352 getExtensionFeatures(D, Args, Features, MArch, OtherExts); 353 354 return true; 355 } 356 357 void riscv::getRISCVTargetFeatures(const Driver &D, const llvm::Triple &Triple, 358 const ArgList &Args, 359 std::vector<StringRef> &Features) { 360 StringRef MArch = getRISCVArch(Args, Triple); 361 362 if (!getArchFeatures(D, MArch, Features, Args)) 363 return; 364 365 // Handle features corresponding to "-ffixed-X" options 366 if (Args.hasArg(options::OPT_ffixed_x1)) 367 Features.push_back("+reserve-x1"); 368 if (Args.hasArg(options::OPT_ffixed_x2)) 369 Features.push_back("+reserve-x2"); 370 if (Args.hasArg(options::OPT_ffixed_x3)) 371 Features.push_back("+reserve-x3"); 372 if (Args.hasArg(options::OPT_ffixed_x4)) 373 Features.push_back("+reserve-x4"); 374 if (Args.hasArg(options::OPT_ffixed_x5)) 375 Features.push_back("+reserve-x5"); 376 if (Args.hasArg(options::OPT_ffixed_x6)) 377 Features.push_back("+reserve-x6"); 378 if (Args.hasArg(options::OPT_ffixed_x7)) 379 Features.push_back("+reserve-x7"); 380 if (Args.hasArg(options::OPT_ffixed_x8)) 381 Features.push_back("+reserve-x8"); 382 if (Args.hasArg(options::OPT_ffixed_x9)) 383 Features.push_back("+reserve-x9"); 384 if (Args.hasArg(options::OPT_ffixed_x10)) 385 Features.push_back("+reserve-x10"); 386 if (Args.hasArg(options::OPT_ffixed_x11)) 387 Features.push_back("+reserve-x11"); 388 if (Args.hasArg(options::OPT_ffixed_x12)) 389 Features.push_back("+reserve-x12"); 390 if (Args.hasArg(options::OPT_ffixed_x13)) 391 Features.push_back("+reserve-x13"); 392 if (Args.hasArg(options::OPT_ffixed_x14)) 393 Features.push_back("+reserve-x14"); 394 if (Args.hasArg(options::OPT_ffixed_x15)) 395 Features.push_back("+reserve-x15"); 396 if (Args.hasArg(options::OPT_ffixed_x16)) 397 Features.push_back("+reserve-x16"); 398 if (Args.hasArg(options::OPT_ffixed_x17)) 399 Features.push_back("+reserve-x17"); 400 if (Args.hasArg(options::OPT_ffixed_x18)) 401 Features.push_back("+reserve-x18"); 402 if (Args.hasArg(options::OPT_ffixed_x19)) 403 Features.push_back("+reserve-x19"); 404 if (Args.hasArg(options::OPT_ffixed_x20)) 405 Features.push_back("+reserve-x20"); 406 if (Args.hasArg(options::OPT_ffixed_x21)) 407 Features.push_back("+reserve-x21"); 408 if (Args.hasArg(options::OPT_ffixed_x22)) 409 Features.push_back("+reserve-x22"); 410 if (Args.hasArg(options::OPT_ffixed_x23)) 411 Features.push_back("+reserve-x23"); 412 if (Args.hasArg(options::OPT_ffixed_x24)) 413 Features.push_back("+reserve-x24"); 414 if (Args.hasArg(options::OPT_ffixed_x25)) 415 Features.push_back("+reserve-x25"); 416 if (Args.hasArg(options::OPT_ffixed_x26)) 417 Features.push_back("+reserve-x26"); 418 if (Args.hasArg(options::OPT_ffixed_x27)) 419 Features.push_back("+reserve-x27"); 420 if (Args.hasArg(options::OPT_ffixed_x28)) 421 Features.push_back("+reserve-x28"); 422 if (Args.hasArg(options::OPT_ffixed_x29)) 423 Features.push_back("+reserve-x29"); 424 if (Args.hasArg(options::OPT_ffixed_x30)) 425 Features.push_back("+reserve-x30"); 426 if (Args.hasArg(options::OPT_ffixed_x31)) 427 Features.push_back("+reserve-x31"); 428 429 // -mrelax is default, unless -mno-relax is specified. 430 if (Args.hasFlag(options::OPT_mrelax, options::OPT_mno_relax, true)) 431 Features.push_back("+relax"); 432 else 433 Features.push_back("-relax"); 434 435 // GCC Compatibility: -mno-save-restore is default, unless -msave-restore is 436 // specified... 437 if (Args.hasFlag(options::OPT_msave_restore, options::OPT_mno_save_restore, false)) { 438 // ... but we don't support -msave-restore, so issue a warning. 439 D.Diag(diag::warn_drv_clang_unsupported) 440 << Args.getLastArg(options::OPT_msave_restore)->getAsString(Args); 441 } 442 443 // Now add any that the user explicitly requested on the command line, 444 // which may override the defaults. 445 handleTargetFeaturesGroup(Args, Features, options::OPT_m_riscv_Features_Group); 446 } 447 448 StringRef riscv::getRISCVABI(const ArgList &Args, const llvm::Triple &Triple) { 449 assert((Triple.getArch() == llvm::Triple::riscv32 || 450 Triple.getArch() == llvm::Triple::riscv64) && 451 "Unexpected triple"); 452 453 // GCC's logic around choosing a default `-mabi=` is complex. If GCC is not 454 // configured using `--with-abi=`, then the logic for the default choice is 455 // defined in config.gcc. This function is based on the logic in GCC 9.2.0. We 456 // deviate from GCC's default only on baremetal targets (UnknownOS) where 457 // neither `-march` nor `-mabi` is specified. 458 // 459 // The logic uses the following, in order: 460 // 1. Explicit choices using `--with-abi=` 461 // 2. A default based on `--with-arch=`, if provided 462 // 3. A default based on the target triple's arch 463 // 464 // The logic in config.gcc is a little circular but it is not inconsistent. 465 // 466 // Clang does not have `--with-arch=` or `--with-abi=`, so we use `-march=` 467 // and `-mabi=` respectively instead. 468 469 // 1. If `-mabi=` is specified, use it. 470 if (const Arg *A = Args.getLastArg(options::OPT_mabi_EQ)) 471 return A->getValue(); 472 473 // 2. Choose a default based on `-march=` 474 // 475 // rv32g | rv32*d -> ilp32d 476 // rv32e -> ilp32e 477 // rv32* -> ilp32 478 // rv64g | rv64*d -> lp64d 479 // rv64* -> lp64 480 if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) { 481 StringRef MArch = A->getValue(); 482 483 if (MArch.startswith_lower("rv32")) { 484 // FIXME: parse `March` to find `D` extension properly 485 if (MArch.substr(4).contains_lower("d") || 486 MArch.startswith_lower("rv32g")) 487 return "ilp32d"; 488 else if (MArch.startswith_lower("rv32e")) 489 return "ilp32e"; 490 else 491 return "ilp32"; 492 } else if (MArch.startswith_lower("rv64")) { 493 // FIXME: parse `March` to find `D` extension properly 494 if (MArch.substr(4).contains_lower("d") || 495 MArch.startswith_lower("rv64g")) 496 return "lp64d"; 497 else 498 return "lp64"; 499 } 500 } 501 502 // 3. Choose a default based on the triple 503 // 504 // We deviate from GCC's defaults here: 505 // - On `riscv{XLEN}-unknown-elf` we use the integer calling convention only. 506 // - On all other OSs we use the double floating point calling convention. 507 if (Triple.getArch() == llvm::Triple::riscv32) { 508 if (Triple.getOS() == llvm::Triple::UnknownOS) 509 return "ilp32"; 510 else 511 return "ilp32d"; 512 } else { 513 if (Triple.getOS() == llvm::Triple::UnknownOS) 514 return "lp64"; 515 else 516 return "lp64d"; 517 } 518 } 519 520 StringRef riscv::getRISCVArch(const llvm::opt::ArgList &Args, 521 const llvm::Triple &Triple) { 522 assert((Triple.getArch() == llvm::Triple::riscv32 || 523 Triple.getArch() == llvm::Triple::riscv64) && 524 "Unexpected triple"); 525 526 // GCC's logic around choosing a default `-march=` is complex. If GCC is not 527 // configured using `--with-arch=`, then the logic for the default choice is 528 // defined in config.gcc. This function is based on the logic in GCC 9.2.0. We 529 // deviate from GCC's default only on baremetal targets (UnknownOS) where 530 // neither `-march` nor `-mabi` is specified. 531 // 532 // The logic uses the following, in order: 533 // 1. Explicit choices using `--with-arch=` 534 // 2. A default based on `--with-abi=`, if provided 535 // 3. A default based on the target triple's arch 536 // 537 // The logic in config.gcc is a little circular but it is not inconsistent. 538 // 539 // Clang does not have `--with-arch=` or `--with-abi=`, so we use `-march=` 540 // and `-mabi=` respectively instead. 541 // 542 // Clang does not yet support MULTILIB_REUSE, so we use `rv{XLEN}imafdc` 543 // instead of `rv{XLEN}gc` though they are (currently) equivalent. 544 545 // 1. If `-march=` is specified, use it. 546 if (const Arg *A = Args.getLastArg(options::OPT_march_EQ)) 547 return A->getValue(); 548 549 // 2. Choose a default based on `-mabi=` 550 // 551 // ilp32e -> rv32e 552 // ilp32 | ilp32f | ilp32d -> rv32imafdc 553 // lp64 | lp64f | lp64d -> rv64imafdc 554 if (const Arg *A = Args.getLastArg(options::OPT_mabi_EQ)) { 555 StringRef MABI = A->getValue(); 556 557 if (MABI.equals_lower("ilp32e")) 558 return "rv32e"; 559 else if (MABI.startswith_lower("ilp32")) 560 return "rv32imafdc"; 561 else if (MABI.startswith_lower("lp64")) 562 return "rv64imafdc"; 563 } 564 565 // 3. Choose a default based on the triple 566 // 567 // We deviate from GCC's defaults here: 568 // - On `riscv{XLEN}-unknown-elf` we default to `rv{XLEN}imac` 569 // - On all other OSs we use `rv{XLEN}imafdc` (equivalent to `rv{XLEN}gc`) 570 if (Triple.getArch() == llvm::Triple::riscv32) { 571 if (Triple.getOS() == llvm::Triple::UnknownOS) 572 return "rv32imac"; 573 else 574 return "rv32imafdc"; 575 } else { 576 if (Triple.getOS() == llvm::Triple::UnknownOS) 577 return "rv64imac"; 578 else 579 return "rv64imafdc"; 580 } 581 } 582