mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 16:48:26 +02:00 
			
		
		
		
	 4f05e82003
			
		
	
	
		4f05e82003
		
	
	
	
	
		
			
			1, Select some options in Kconfig; 2, Give a chance to build with !CONFIG_SMP; 3, Switch to use built-in rustc target; 4, Add new supported device nodes to dts; 5, Some bug fixes and other small changes; 6, Update the default config file. -----BEGIN PGP SIGNATURE----- iQJKBAABCAA0FiEEzOlt8mkP+tbeiYy5AoYrw/LiJnoFAmZKCycWHGNoZW5odWFj YWlAa2VybmVsLm9yZwAKCRAChivD8uImeoXWD/9pFhbbJj49T1xiwc2j/XgQL8HI s88/h4z5AXEbHFO8XIG1Cpw/Z3a1DsCiWBsOkCogagILzYuN0r7UqcrI02ZoeY6N fbuDatB3i+hJWCBzcl1HPkFy/9av4j4EktZs0+X/wVgKkd0aIh78qs8+1RwKhshf FoOv+cMu7zFS8Jrt+w16diNCY1JsDv7TCkCVhvJxAodrtGg4oo2NPfrGOrKAP8Dq LClvFEqDcXq1kKcipw3Q7BwDlBpJEvLZ0iAl19BnLAmBzI3Wfze9ouoYv8WiUyaY br0GPShGf16I3DKtTdHsHH/zmayQ7JSmFzZ9JEHzcBrE4AprfWLuwsUjd2WXDD6U wK+p4tWd0AUFf+/h4u1yQB9/rlt+JZ2ny/A2u4YR/BPtthiYqp8SDSH62vpCSFOE dByDeTbfjTdJsWr+bsI2gOO0sVwDYpph9SJfAyBn4miKw7v8w+2rI1oqo/ZQkP59 0SczM9C9jzpgXSGDc4yQbnqoA4KA9U6zljd12mYL5HV/AjhD19va3FmENgByZUuE Z7A0RZsiU5T401xEZiOUhwzy9m/USc1O2ivCmeowx9kP/gWic0KeAsmlMiro0jeR y9jthci8iOgjjLmCEVC06GWGUojP2roXI/38We6enVevy2GXbEEDRa1QGbQ5ndoJ MEPm4NvW1wsBgWIYmg== =WR15 -----END PGP SIGNATURE----- Merge tag 'loongarch-6.10' of git://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson Pull LoongArch updates from Huacai Chen: - Select some options in Kconfig - Give a chance to build with !CONFIG_SMP - Switch to use built-in rustc target - Add new supported device nodes to dts - Some bug fixes and other small changes - Update the default config file * tag 'loongarch-6.10' of git://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson: LoongArch: Update Loongson-3 default config file LoongArch: dts: Add new supported device nodes to Loongson-2K2000 LoongArch: dts: Add new supported device nodes to Loongson-2K0500 LoongArch: dts: Remove "disabled" state of clock controller node LoongArch: rust: Switch to use built-in rustc target LoongArch: Fix callchain parse error with kernel tracepoint events again LoongArch: Give a chance to build with !CONFIG_SMP LoongArch: Select THP_SWAP if HAVE_ARCH_TRANSPARENT_HUGEPAGE LoongArch: Select ARCH_WANT_DEFAULT_BPF_JIT LoongArch: Select ARCH_SUPPORTS_INT128 if CC_HAS_INT128 LoongArch: Select ARCH_HAS_FAST_MULTIPLIER
		
			
				
	
	
		
			186 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| // 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("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 = "-3dnow,-3dnowa,-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("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);
 | |
| }
 |