// SPDX-License-Identifier: GPL-2.0 //! The custom target specification file generator for `rustc`. //! //! To configure a target from scratch, a JSON-encoded file has to be passed //! to `rustc` (introduced in [RFC 131]). These options and the file itself are //! unstable. Eventually, `rustc` should provide a way to do this in a stable //! manner. For instance, via command-line arguments. Therefore, this file //! should avoid using keys which can be set via `-C` or `-Z` options. //! //! [RFC 131]: https://rust-lang.github.io/rfcs/0131-target-specification.html use std::{ collections::HashMap, fmt::{Display, Formatter, Result}, io::BufRead, }; enum Value { Boolean(bool), Number(i32), String(String), Object(Object), } type Object = Vec<(String, Value)>; /// Minimal "almost JSON" generator (e.g. no `null`s, no arrays, no escaping), /// enough for this purpose. impl Display for Value { fn fmt(&self, formatter: &mut Formatter<'_>) -> Result { match self { Value::Boolean(boolean) => write!(formatter, "{}", boolean), Value::Number(number) => write!(formatter, "{}", number), Value::String(string) => write!(formatter, "\"{}\"", string), Value::Object(object) => { formatter.write_str("{")?; if let [ref rest @ .., ref last] = object[..] { for (key, value) in rest { write!(formatter, "\"{}\": {},", key, value)?; } write!(formatter, "\"{}\": {}", last.0, last.1)?; } formatter.write_str("}") } } } } struct TargetSpec(Object); impl TargetSpec { fn new() -> TargetSpec { TargetSpec(Vec::new()) } } trait Push { fn push(&mut self, key: &str, value: T); } impl Push for TargetSpec { fn push(&mut self, key: &str, value: bool) { self.0.push((key.to_string(), Value::Boolean(value))); } } impl Push for TargetSpec { fn push(&mut self, key: &str, value: i32) { self.0.push((key.to_string(), Value::Number(value))); } } impl Push for TargetSpec { fn push(&mut self, key: &str, value: String) { self.0.push((key.to_string(), Value::String(value))); } } impl Push<&str> for TargetSpec { fn push(&mut self, key: &str, value: &str) { self.push(key, value.to_string()); } } impl Push for TargetSpec { fn push(&mut self, key: &str, value: Object) { self.0.push((key.to_string(), Value::Object(value))); } } impl Display for TargetSpec { fn fmt(&self, formatter: &mut Formatter<'_>) -> Result { // We add some newlines for clarity. formatter.write_str("{\n")?; if let [ref rest @ .., ref last] = self.0[..] { for (key, value) in rest { write!(formatter, " \"{}\": {},\n", key, value)?; } write!(formatter, " \"{}\": {}\n", last.0, last.1)?; } formatter.write_str("}") } } struct KernelConfig(HashMap); impl KernelConfig { /// Parses `include/config/auto.conf` from `stdin`. fn from_stdin() -> KernelConfig { let mut result = HashMap::new(); let stdin = std::io::stdin(); let mut handle = stdin.lock(); let mut line = String::new(); loop { line.clear(); if handle.read_line(&mut line).unwrap() == 0 { break; } if line.starts_with('#') { continue; } let (key, value) = line.split_once('=').expect("Missing `=` in line."); result.insert(key.to_string(), value.trim_end_matches('\n').to_string()); } KernelConfig(result) } /// Does the option exist in the configuration (any value)? /// /// The argument must be passed without the `CONFIG_` prefix. /// This avoids repetition and it also avoids `fixdep` making us /// depend on it. fn has(&self, option: &str) -> bool { let option = "CONFIG_".to_owned() + option; self.0.contains_key(&option) } } fn main() { let cfg = KernelConfig::from_stdin(); let mut ts = TargetSpec::new(); // `llvm-target`s are taken from `scripts/Makefile.clang`. if cfg.has("ARM") { ts.push("arch", "arm"); ts.push( "data-layout", "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64", ); ts.push("features", "+strict-align,+v6"); ts.push("llvm-target", "arm-linux-gnueabi"); ts.push("max-atomic-width", 64); ts.push("target-mcount", "\\u0001__gnu_mcount_nc"); ts.push("target-pointer-width", "32"); } else if cfg.has("ARM64") { ts.push("arch", "aarch64"); ts.push( "data-layout", "e-m:e-i8:8:32-i16:16:32-i64:64-i128:128-n32:64-S128", ); ts.push("disable-redzone", true); ts.push("features", "+strict-align,-neon,-fp-armv8"); ts.push("llvm-target", "aarch64-linux-gnu"); ts.push("max-atomic-width", 128); ts.push("target-pointer-width", "64"); } else if cfg.has("PPC") { ts.push("arch", "powerpc64"); ts.push("code-model", "large"); ts.push("data-layout", "e-m:e-i64:64-n32:64"); ts.push("features", "-altivec,-vsx,-hard-float"); ts.push("llvm-target", "powerpc64le-linux-gnu"); ts.push("max-atomic-width", 64); ts.push("target-mcount", "_mcount"); ts.push("target-pointer-width", "64"); } else if cfg.has("RISCV") { if cfg.has("64BIT") { ts.push("arch", "riscv64"); ts.push("data-layout", "e-m:e-p:64:64-i64:64-i128:128-n64-S128"); ts.push("llvm-target", "riscv64-linux-gnu"); ts.push("target-pointer-width", "64"); } else { ts.push("arch", "riscv32"); ts.push("data-layout", "e-m:e-p:32:32-i64:64-n32-S128"); ts.push("llvm-target", "riscv32-linux-gnu"); ts.push("target-pointer-width", "32"); } ts.push("code-model", "medium"); ts.push("disable-redzone", true); let mut features = "+m,+a".to_string(); if cfg.has("RISCV_ISA_C") { features += ",+c"; } ts.push("features", features); } else if cfg.has("X86_64") { ts.push("arch", "x86_64"); ts.push( "data-layout", "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128", ); ts.push("llvm-target", "x86_64-linux-gnu"); ts.push("target-pointer-width", "64"); } else { panic!("Unsupported architecture"); } ts.push("emit-debug-gdb-scripts", false); ts.push("frame-pointer", "may-omit"); ts.push( "stack-probes", vec![("kind".to_string(), Value::String("none".to_string()))], ); // Everything else is LE, whether `CPU_LITTLE_ENDIAN` is declared or not // (e.g. x86). It is also `rustc`'s default. if cfg.has("CPU_BIG_ENDIAN") { ts.push("target-endian", "big"); } println!("{}", ts); }