|  | // 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<T> { | 
|  | fn push(&mut self, key: &str, value: T); | 
|  | } | 
|  |  | 
|  | impl Push<bool> for TargetSpec { | 
|  | fn push(&mut self, key: &str, value: bool) { | 
|  | self.0.push((key.to_string(), Value::Boolean(value))); | 
|  | } | 
|  | } | 
|  |  | 
|  | impl Push<i32> for TargetSpec { | 
|  | fn push(&mut self, key: &str, value: i32) { | 
|  | self.0.push((key.to_string(), Value::Number(value))); | 
|  | } | 
|  | } | 
|  |  | 
|  | impl Push<String> 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<Object> 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<String, String>); | 
|  |  | 
|  | 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("ARM64") { | 
|  | panic!("arm64 uses the builtin rustc aarch64-unknown-none target"); | 
|  | } else if cfg.has("RISCV") { | 
|  | if cfg.has("64BIT") { | 
|  | panic!("64-bit RISC-V uses the builtin rustc riscv64-unknown-none-elf target"); | 
|  | } else { | 
|  | panic!("32-bit RISC-V is an unsupported architecture"); | 
|  | } | 
|  | } 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-i128:128-f80:128-n8:16:32:64-S128", | 
|  | ); | 
|  | let mut features = "-mmx,+soft-float".to_string(); | 
|  | if cfg.has("MITIGATION_RETPOLINE") { | 
|  | features += ",+retpoline-external-thunk"; | 
|  | } | 
|  | ts.push("features", features); | 
|  | ts.push("llvm-target", "x86_64-linux-gnu"); | 
|  | ts.push("target-pointer-width", "64"); | 
|  | } else if cfg.has("X86_32") { | 
|  | // This only works on UML, as i386 otherwise needs regparm support in rustc | 
|  | if !cfg.has("UML") { | 
|  | panic!("32-bit x86 only works under UML"); | 
|  | } | 
|  | ts.push("arch", "x86"); | 
|  | ts.push( | 
|  | "data-layout", | 
|  | "e-m:e-p:32:32-p270:32:32-p271:32:32-p272:64:64-i128:128-f64:32:64-f80:32-n8:16:32-S128", | 
|  | ); | 
|  | let mut features = "-mmx,+soft-float".to_string(); | 
|  | if cfg.has("MITIGATION_RETPOLINE") { | 
|  | features += ",+retpoline-external-thunk"; | 
|  | } | 
|  | ts.push("features", features); | 
|  | ts.push("llvm-target", "i386-unknown-linux-gnu"); | 
|  | ts.push("target-pointer-width", "32"); | 
|  | } else if cfg.has("LOONGARCH") { | 
|  | panic!("loongarch uses the builtin rustc loongarch64-unknown-none-softfloat target"); | 
|  | } 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); | 
|  | } |