|
1 |
| -use std::ffi::{CStr, CString, c_char, c_void}; |
| 1 | +use std::collections::VecDeque; |
| 2 | +use std::ffi::{CStr, CString}; |
2 | 3 | use std::fmt::Write;
|
3 | 4 | use std::path::Path;
|
4 | 5 | use std::sync::Once;
|
@@ -387,7 +388,65 @@ fn llvm_target_features(tm: &llvm::TargetMachine) -> Vec<(&str, &str)> {
|
387 | 388 | ret
|
388 | 389 | }
|
389 | 390 |
|
390 |
| -fn print_target_features(out: &mut String, sess: &Session, tm: &llvm::TargetMachine) { |
| 391 | +pub(crate) fn print(req: &PrintRequest, out: &mut String, sess: &Session) { |
| 392 | + require_inited(); |
| 393 | + let tm = create_informational_target_machine(sess, false); |
| 394 | + match req.kind { |
| 395 | + PrintKind::TargetCPUs => print_target_cpus(sess, &tm, out), |
| 396 | + PrintKind::TargetFeatures => print_target_features(sess, &tm, out), |
| 397 | + _ => bug!("rustc_codegen_llvm can't handle print request: {:?}", req), |
| 398 | + } |
| 399 | +} |
| 400 | + |
| 401 | +fn print_target_cpus(sess: &Session, tm: &llvm::TargetMachine, out: &mut String) { |
| 402 | + let cpu_names = llvm::build_string(|s| unsafe { |
| 403 | + llvm::LLVMRustPrintTargetCPUs(&tm, s); |
| 404 | + }) |
| 405 | + .unwrap(); |
| 406 | + |
| 407 | + struct Cpu<'a> { |
| 408 | + cpu_name: &'a str, |
| 409 | + remark: String, |
| 410 | + } |
| 411 | + // Compare CPU against current target to label the default. |
| 412 | + let target_cpu = handle_native(&sess.target.cpu); |
| 413 | + let make_remark = |cpu_name| { |
| 414 | + if cpu_name == target_cpu { |
| 415 | + // FIXME(#132514): This prints the LLVM target string, which can be |
| 416 | + // different from the Rust target string. Is that intended? |
| 417 | + let target = &sess.target.llvm_target; |
| 418 | + format!( |
| 419 | + " - This is the default target CPU for the current build target (currently {target})." |
| 420 | + ) |
| 421 | + } else { |
| 422 | + "".to_owned() |
| 423 | + } |
| 424 | + }; |
| 425 | + let mut cpus = cpu_names |
| 426 | + .lines() |
| 427 | + .map(|cpu_name| Cpu { cpu_name, remark: make_remark(cpu_name) }) |
| 428 | + .collect::<VecDeque<_>>(); |
| 429 | + |
| 430 | + // Only print the "native" entry when host and target are the same arch, |
| 431 | + // since otherwise it could be wrong or misleading. |
| 432 | + if sess.host.arch == sess.target.arch { |
| 433 | + let host = get_host_cpu_name(); |
| 434 | + cpus.push_front(Cpu { |
| 435 | + cpu_name: "native", |
| 436 | + remark: format!(" - Select the CPU of the current host (currently {host})."), |
| 437 | + }); |
| 438 | + } |
| 439 | + |
| 440 | + let max_name_width = cpus.iter().map(|cpu| cpu.cpu_name.len()).max().unwrap_or(0); |
| 441 | + writeln!(out, "Available CPUs for this target:").unwrap(); |
| 442 | + for Cpu { cpu_name, remark } in cpus { |
| 443 | + // Only pad the CPU name if there's a remark to print after it. |
| 444 | + let width = if remark.is_empty() { 0 } else { max_name_width }; |
| 445 | + writeln!(out, " {cpu_name:<width$}{remark}").unwrap(); |
| 446 | + } |
| 447 | +} |
| 448 | + |
| 449 | +fn print_target_features(sess: &Session, tm: &llvm::TargetMachine, out: &mut String) { |
391 | 450 | let mut llvm_target_features = llvm_target_features(tm);
|
392 | 451 | let mut known_llvm_target_features = FxHashSet::<&'static str>::default();
|
393 | 452 | let mut rustc_target_features = sess
|
@@ -447,52 +506,31 @@ fn print_target_features(out: &mut String, sess: &Session, tm: &llvm::TargetMach
|
447 | 506 | writeln!(out, "and may be renamed or removed in a future version of LLVM or rustc.\n").unwrap();
|
448 | 507 | }
|
449 | 508 |
|
450 |
| -pub(crate) fn print(req: &PrintRequest, mut out: &mut String, sess: &Session) { |
451 |
| - require_inited(); |
452 |
| - let tm = create_informational_target_machine(sess, false); |
453 |
| - match req.kind { |
454 |
| - PrintKind::TargetCPUs => { |
455 |
| - // SAFETY generate a C compatible string from a byte slice to pass |
456 |
| - // the target CPU name into LLVM, the lifetime of the reference is |
457 |
| - // at least as long as the C function |
458 |
| - let cpu_cstring = CString::new(handle_native(sess.target.cpu.as_ref())) |
459 |
| - .unwrap_or_else(|e| bug!("failed to convert to cstring: {}", e)); |
460 |
| - unsafe extern "C" fn callback(out: *mut c_void, string: *const c_char, len: usize) { |
461 |
| - let out = unsafe { &mut *(out as *mut &mut String) }; |
462 |
| - let bytes = unsafe { slice::from_raw_parts(string as *const u8, len) }; |
463 |
| - write!(out, "{}", String::from_utf8_lossy(bytes)).unwrap(); |
464 |
| - } |
465 |
| - unsafe { |
466 |
| - llvm::LLVMRustPrintTargetCPUs( |
467 |
| - &tm, |
468 |
| - cpu_cstring.as_ptr(), |
469 |
| - callback, |
470 |
| - (&raw mut out) as *mut c_void, |
471 |
| - ); |
472 |
| - } |
473 |
| - } |
474 |
| - PrintKind::TargetFeatures => print_target_features(out, sess, &tm), |
475 |
| - _ => bug!("rustc_codegen_llvm can't handle print request: {:?}", req), |
476 |
| - } |
| 509 | +/// Returns the host CPU name, according to LLVM. |
| 510 | +fn get_host_cpu_name() -> &'static str { |
| 511 | + let mut len = 0; |
| 512 | + // SAFETY: The underlying C++ global function returns a `StringRef` that |
| 513 | + // isn't tied to any particular backing buffer, so it must be 'static. |
| 514 | + let slice: &'static [u8] = unsafe { |
| 515 | + let ptr = llvm::LLVMRustGetHostCPUName(&mut len); |
| 516 | + assert!(!ptr.is_null()); |
| 517 | + slice::from_raw_parts(ptr, len) |
| 518 | + }; |
| 519 | + str::from_utf8(slice).expect("host CPU name should be UTF-8") |
477 | 520 | }
|
478 | 521 |
|
479 |
| -fn handle_native(name: &str) -> &str { |
480 |
| - if name != "native" { |
481 |
| - return name; |
482 |
| - } |
483 |
| - |
484 |
| - unsafe { |
485 |
| - let mut len = 0; |
486 |
| - let ptr = llvm::LLVMRustGetHostCPUName(&mut len); |
487 |
| - str::from_utf8(slice::from_raw_parts(ptr as *const u8, len)).unwrap() |
| 522 | +/// If the given string is `"native"`, returns the host CPU name according to |
| 523 | +/// LLVM. Otherwise, the string is returned as-is. |
| 524 | +fn handle_native(cpu_name: &str) -> &str { |
| 525 | + match cpu_name { |
| 526 | + "native" => get_host_cpu_name(), |
| 527 | + _ => cpu_name, |
488 | 528 | }
|
489 | 529 | }
|
490 | 530 |
|
491 | 531 | pub(crate) fn target_cpu(sess: &Session) -> &str {
|
492 |
| - match sess.opts.cg.target_cpu { |
493 |
| - Some(ref name) => handle_native(name), |
494 |
| - None => handle_native(sess.target.cpu.as_ref()), |
495 |
| - } |
| 532 | + let cpu_name = sess.opts.cg.target_cpu.as_deref().unwrap_or_else(|| &sess.target.cpu); |
| 533 | + handle_native(cpu_name) |
496 | 534 | }
|
497 | 535 |
|
498 | 536 | /// The list of LLVM features computed from CLI flags (`-Ctarget-cpu`, `-Ctarget-feature`,
|
|
0 commit comments