Merge pull request #3994 from i509VCB/mspm0-generate-singletons
mspm0: generate all singletons
This commit is contained in:
		
						commit
						f15a11f4d6
					
				| @ -45,14 +45,14 @@ cortex-m = "0.7.6" | |||||||
| critical-section = "1.2.0" | critical-section = "1.2.0" | ||||||
| 
 | 
 | ||||||
| # mspm0-metapac = { version = "" } | # mspm0-metapac = { version = "" } | ||||||
| mspm0-metapac = { git = "https://github.com/mspm0-rs/mspm0-data-generated/", tag = "mspm0-data-9faa5239a8eab04946086158f2a7fdff5a6a179d" } | mspm0-metapac = { git = "https://github.com/mspm0-rs/mspm0-data-generated/", tag = "mspm0-data-119240dd23ef5748d2a7bef219ca298d37ba604a" } | ||||||
| 
 | 
 | ||||||
| [build-dependencies] | [build-dependencies] | ||||||
| proc-macro2 = "1.0.94" | proc-macro2 = "1.0.94" | ||||||
| quote = "1.0.40" | quote = "1.0.40" | ||||||
| 
 | 
 | ||||||
| # mspm0-metapac = { version = "", default-features = false, features = ["metadata"] } | # mspm0-metapac = { version = "", default-features = false, features = ["metadata"] } | ||||||
| mspm0-metapac = { git = "https://github.com/mspm0-rs/mspm0-data-generated/", tag = "mspm0-data-9faa5239a8eab04946086158f2a7fdff5a6a179d", default-features = false, features = ["metadata"] } | mspm0-metapac = { git = "https://github.com/mspm0-rs/mspm0-data-generated/", tag = "mspm0-data-119240dd23ef5748d2a7bef219ca298d37ba604a", default-features = false, features = ["metadata"] } | ||||||
| 
 | 
 | ||||||
| [features] | [features] | ||||||
| default = ["rt"] | default = ["rt"] | ||||||
| @ -107,7 +107,10 @@ time-driver-timg9 = ["_time-driver"] | |||||||
| time-driver-timg10 = ["_time-driver"] | time-driver-timg10 = ["_time-driver"] | ||||||
| ## Use TIMG11 as time driver | ## Use TIMG11 as time driver | ||||||
| time-driver-timg11 = ["_time-driver"] | time-driver-timg11 = ["_time-driver"] | ||||||
| # TODO: Support TIMG12 and TIMG13 | ## Use TIMG12 as time driver | ||||||
|  | time-driver-timg12 = ["_time-driver"] | ||||||
|  | ## Use TIMG13 as time driver | ||||||
|  | time-driver-timg13 = ["_time-driver"] | ||||||
| ## Use TIMG14 as time driver | ## Use TIMG14 as time driver | ||||||
| time-driver-timg14 = ["_time-driver"] | time-driver-timg14 = ["_time-driver"] | ||||||
| ## Use TIMA0 as time driver | ## Use TIMA0 as time driver | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| use std::collections::HashMap; | use std::cmp::Ordering; | ||||||
|  | use std::collections::{BTreeSet, HashMap}; | ||||||
| use std::io::Write; | use std::io::Write; | ||||||
| use std::path::{Path, PathBuf}; | use std::path::{Path, PathBuf}; | ||||||
| use std::process::Command; | use std::process::Command; | ||||||
| @ -23,48 +24,128 @@ fn generate_code() { | |||||||
| 
 | 
 | ||||||
|     cfgs.declare_all(&["gpio_pb", "gpio_pc", "int_group1"]); |     cfgs.declare_all(&["gpio_pb", "gpio_pc", "int_group1"]); | ||||||
| 
 | 
 | ||||||
|     let mut singletons = Vec::new(); |     let mut singletons = get_singletons(&mut cfgs); | ||||||
| 
 | 
 | ||||||
|     // Generate singletons for GPIO pins. To only consider pins available on a family, use the name of
 |     time_driver(&mut singletons, &mut cfgs); | ||||||
|     // the pins from the pincm mappings.
 |  | ||||||
|     for pincm_mapping in METADATA.pincm_mappings.iter() { |  | ||||||
|         singletons.push(pincm_mapping.pin.to_string()); |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     for peri in METADATA.peripherals { |  | ||||||
|         match peri.kind { |  | ||||||
|             // Specially generated.
 |  | ||||||
|             "gpio" => match peri.name { |  | ||||||
|                 "GPIOB" => cfgs.enable("gpio_pb"), |  | ||||||
|                 "GPIOC" => cfgs.enable("gpio_pc"), |  | ||||||
|                 _ => (), |  | ||||||
|             }, |  | ||||||
| 
 |  | ||||||
|             // These peripherals are managed internally by the hal.
 |  | ||||||
|             "iomux" | "cpuss" => {} |  | ||||||
| 
 |  | ||||||
|             _ => singletons.push(peri.name.to_string()), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     time_driver(&singletons, &mut cfgs); |  | ||||||
| 
 |  | ||||||
|     // ========
 |  | ||||||
|     // Write singletons
 |  | ||||||
|     let mut g = TokenStream::new(); |     let mut g = TokenStream::new(); | ||||||
| 
 | 
 | ||||||
|     let singleton_tokens: Vec<_> = singletons.iter().map(|s| format_ident!("{}", s)).collect(); |     g.extend(generate_singletons(&singletons)); | ||||||
|  |     g.extend(generate_pincm_mapping()); | ||||||
|  |     g.extend(generate_pin()); | ||||||
|  |     g.extend(generate_timers()); | ||||||
|  |     g.extend(generate_interrupts()); | ||||||
|  |     g.extend(generate_peripheral_instances()); | ||||||
|  |     g.extend(generate_pin_trait_impls()); | ||||||
| 
 | 
 | ||||||
|     g.extend(quote! { |     let out_dir = &PathBuf::from(env::var_os("OUT_DIR").unwrap()); | ||||||
|         embassy_hal_internal::peripherals_definition!(#(#singleton_tokens),*); |     let out_file = out_dir.join("_generated.rs").to_string_lossy().to_string(); | ||||||
|     }); |     fs::write(&out_file, g.to_string()).unwrap(); | ||||||
|  |     rustfmt(&out_file); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     g.extend(quote! { | #[derive(Debug, Clone)] | ||||||
|         embassy_hal_internal::peripherals_struct!(#(#singleton_tokens),*); | struct Singleton { | ||||||
|     }); |     name: String, | ||||||
| 
 | 
 | ||||||
|     // ========
 |     cfg: Option<TokenStream>, | ||||||
|     // Generate GPIO pincm lookup tables.
 | } | ||||||
|  | 
 | ||||||
|  | impl PartialEq for Singleton { | ||||||
|  |     fn eq(&self, other: &Self) -> bool { | ||||||
|  |         self.name == other.name | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Eq for Singleton {} | ||||||
|  | 
 | ||||||
|  | impl PartialOrd for Singleton { | ||||||
|  |     fn partial_cmp(&self, other: &Self) -> Option<Ordering> { | ||||||
|  |         Some(self.cmp(other)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Ord for Singleton { | ||||||
|  |     fn cmp(&self, other: &Self) -> Ordering { | ||||||
|  |         self.name.cmp(&other.name) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn get_singletons(cfgs: &mut common::CfgSet) -> Vec<Singleton> { | ||||||
|  |     let mut singletons = Vec::<Singleton>::new(); | ||||||
|  | 
 | ||||||
|  |     for peripheral in METADATA.peripherals { | ||||||
|  |         // Some peripherals do not generate a singleton, but generate a singleton for each pin.
 | ||||||
|  |         let skip_peripheral_singleton = match peripheral.kind { | ||||||
|  |             "gpio" => { | ||||||
|  |                 // Also enable ports that are present.
 | ||||||
|  |                 match peripheral.name { | ||||||
|  |                     "GPIOB" => cfgs.enable("gpio_pb"), | ||||||
|  |                     "GPIOC" => cfgs.enable("gpio_pc"), | ||||||
|  |                     _ => (), | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 true | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // Each channel gets a singleton, handled separately.
 | ||||||
|  |             "dma" => true, | ||||||
|  | 
 | ||||||
|  |             // These peripherals do not exist as singletons, and have no signals but are managed
 | ||||||
|  |             // by the HAL.
 | ||||||
|  |             "iomux" | "cpuss" => true, | ||||||
|  | 
 | ||||||
|  |             _ => false, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         if !skip_peripheral_singleton { | ||||||
|  |             singletons.push(Singleton { | ||||||
|  |                 name: peripheral.name.to_string(), | ||||||
|  |                 cfg: None, | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let mut signals = BTreeSet::new(); | ||||||
|  | 
 | ||||||
|  |         // Pick out each unique signal. There may be multiple instances of each signal due to
 | ||||||
|  |         // iomux mappings.
 | ||||||
|  |         for pin in peripheral.pins { | ||||||
|  |             let signal = if peripheral.name.starts_with("GPIO") | ||||||
|  |                 || peripheral.name.starts_with("VREF") | ||||||
|  |                 || peripheral.name.starts_with("RTC") | ||||||
|  |             { | ||||||
|  |                 pin.signal.to_string() | ||||||
|  |             } else { | ||||||
|  |                 format!("{}_{}", peripheral.name, pin.signal) | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             // We need to rename some signals to become valid Rust identifiers.
 | ||||||
|  |             let signal = make_valid_identifier(&signal); | ||||||
|  |             signals.insert(signal); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         singletons.extend(signals); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // DMA channels get their own singletons
 | ||||||
|  |     for dma_channel in METADATA.dma_channels.iter() { | ||||||
|  |         singletons.push(Singleton { | ||||||
|  |             name: format!("DMA_CH{}", dma_channel.number), | ||||||
|  |             cfg: None, | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     singletons.sort_by(|a, b| a.name.cmp(&b.name)); | ||||||
|  |     singletons | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn make_valid_identifier(s: &str) -> Singleton { | ||||||
|  |     let name = s.replace('+', "_P").replace("-", "_N"); | ||||||
|  | 
 | ||||||
|  |     Singleton { name, cfg: None } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_pincm_mapping() -> TokenStream { | ||||||
|     let pincms = METADATA.pincm_mappings.iter().map(|mapping| { |     let pincms = METADATA.pincm_mappings.iter().map(|mapping| { | ||||||
|         let port_letter = mapping.pin.strip_prefix("P").unwrap(); |         let port_letter = mapping.pin.strip_prefix("P").unwrap(); | ||||||
|         let port_base = (port_letter.chars().next().unwrap() as u8 - b'A') * 32; |         let port_base = (port_letter.chars().next().unwrap() as u8 - b'A') * 32; | ||||||
| @ -81,7 +162,7 @@ fn generate_code() { | |||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     g.extend(quote! { |     quote! { | ||||||
|         #[doc = "Get the mapping from GPIO pin port to IOMUX PINCM index. This is required since the mapping from IO to PINCM index is not consistent across parts."] |         #[doc = "Get the mapping from GPIO pin port to IOMUX PINCM index. This is required since the mapping from IO to PINCM index is not consistent across parts."] | ||||||
|         pub(crate) fn gpio_pincm(pin_port: u8) -> u8 { |         pub(crate) fn gpio_pincm(pin_port: u8) -> u8 { | ||||||
|             match pin_port { |             match pin_port { | ||||||
| @ -89,9 +170,11 @@ fn generate_code() { | |||||||
|                 _ => unreachable!(), |                 _ => unreachable!(), | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     }); |     } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     for pincm_mapping in METADATA.pincm_mappings.iter() { | fn generate_pin() -> TokenStream { | ||||||
|  |     let pin_impls = METADATA.pincm_mappings.iter().map(|pincm_mapping| { | ||||||
|         let name = Ident::new(&pincm_mapping.pin, Span::call_site()); |         let name = Ident::new(&pincm_mapping.pin, Span::call_site()); | ||||||
|         let port_letter = pincm_mapping.pin.strip_prefix("P").unwrap(); |         let port_letter = pincm_mapping.pin.strip_prefix("P").unwrap(); | ||||||
|         let port_letter = port_letter.chars().next().unwrap(); |         let port_letter = port_letter.chars().next().unwrap(); | ||||||
| @ -101,78 +184,19 @@ fn generate_code() { | |||||||
| 
 | 
 | ||||||
|         // TODO: Feature gate pins that can be used as NRST
 |         // TODO: Feature gate pins that can be used as NRST
 | ||||||
| 
 | 
 | ||||||
|         g.extend(quote! { |         quote! { | ||||||
|             impl_pin!(#name, crate::gpio::Port::#port, #pin_number); |             impl_pin!(#name, crate::gpio::Port::#port, #pin_number); | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Generate timers
 |  | ||||||
|     for peripheral in METADATA.peripherals.iter().filter(|p| p.name.starts_with("TIM")) { |  | ||||||
|         let name = Ident::new(&peripheral.name, Span::call_site()); |  | ||||||
|         let timers = &*TIMERS; |  | ||||||
| 
 |  | ||||||
|         let timer = timers.get(peripheral.name).expect("Timer does not exist"); |  | ||||||
|         assert!(timer.bits == 16 || timer.bits == 32); |  | ||||||
|         let bits = if timer.bits == 16 { |  | ||||||
|             quote! { Bits16 } |  | ||||||
|         } else { |  | ||||||
|             quote! { Bits32 } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         g.extend(quote! { |  | ||||||
|             impl_timer!(#name, #bits); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Generate interrupt module
 |  | ||||||
|     let interrupts: Vec<Ident> = METADATA |  | ||||||
|         .interrupts |  | ||||||
|         .iter() |  | ||||||
|         .map(|interrupt| Ident::new(interrupt.name, Span::call_site())) |  | ||||||
|         .collect(); |  | ||||||
| 
 |  | ||||||
|     g.extend(quote! { |  | ||||||
|         embassy_hal_internal::interrupt_mod! { |  | ||||||
|             #(#interrupts),* |  | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     let group_interrupt_enables = METADATA |     quote! { | ||||||
|         .interrupts |         #(#pin_impls)* | ||||||
|         .iter() |     } | ||||||
|         .filter(|interrupt| interrupt.name.contains("GROUP")) |  | ||||||
|         .map(|interrupt| { |  | ||||||
|             let name = Ident::new(interrupt.name, Span::call_site()); |  | ||||||
| 
 |  | ||||||
|             quote! { |  | ||||||
|                 crate::interrupt::typelevel::#name::enable(); |  | ||||||
|             } |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|     // Generate interrupt enables for groups
 |  | ||||||
|     g.extend(quote! { |  | ||||||
|         pub fn enable_group_interrupts(_cs: critical_section::CriticalSection) { |  | ||||||
|             use crate::interrupt::typelevel::Interrupt; |  | ||||||
| 
 |  | ||||||
|             unsafe { |  | ||||||
|                 #(#group_interrupt_enables)* |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     let out_dir = &PathBuf::from(env::var_os("OUT_DIR").unwrap()); |  | ||||||
|     let out_file = out_dir.join("_generated.rs").to_string_lossy().to_string(); |  | ||||||
|     fs::write(&out_file, g.to_string()).unwrap(); |  | ||||||
|     rustfmt(&out_file); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn time_driver(singletons: &[String], cfgs: &mut CfgSet) { | fn time_driver(singletons: &mut Vec<Singleton>, cfgs: &mut CfgSet) { | ||||||
|     // Timer features
 |     // Timer features
 | ||||||
|     for (timer, desc) in TIMERS.iter() { |     for (timer, _) in TIMERS.iter() { | ||||||
|         if desc.bits != 16 { |  | ||||||
|             continue; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let name = timer.to_lowercase(); |         let name = timer.to_lowercase(); | ||||||
|         cfgs.declare(&format!("time_driver_{}", name)); |         cfgs.declare(&format!("time_driver_{}", name)); | ||||||
|     } |     } | ||||||
| @ -192,7 +216,7 @@ fn time_driver(singletons: &[String], cfgs: &mut CfgSet) { | |||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // Verify the selected timer is available
 |     // Verify the selected timer is available
 | ||||||
|     let singleton = match time_driver.as_ref().map(|x| x.as_ref()) { |     let selected_timer = match time_driver.as_ref().map(|x| x.as_ref()) { | ||||||
|         None => "", |         None => "", | ||||||
|         Some("timg0") => "TIMG0", |         Some("timg0") => "TIMG0", | ||||||
|         Some("timg1") => "TIMG1", |         Some("timg1") => "TIMG1", | ||||||
| @ -228,14 +252,181 @@ fn time_driver(singletons: &[String], cfgs: &mut CfgSet) { | |||||||
|                 "TIMA0", "TIMA1", |                 "TIMA0", "TIMA1", | ||||||
|             ] |             ] | ||||||
|             .iter() |             .iter() | ||||||
|             .find(|tim| singletons.contains(&tim.to_string())) |             .find(|tim| singletons.iter().any(|s| s.name == **tim)) | ||||||
|             .expect("Could not find any timer") |             .expect("Could not find any timer") | ||||||
|         } |         } | ||||||
|         _ => panic!("unknown time_driver {:?}", time_driver), |         _ => panic!("unknown time_driver {:?}", time_driver), | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     if !singleton.is_empty() { |     if !selected_timer.is_empty() { | ||||||
|         cfgs.enable(format!("time_driver_{}", singleton.to_lowercase())); |         cfgs.enable(format!("time_driver_{}", selected_timer.to_lowercase())); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Apply cfgs to each timer and it's pins
 | ||||||
|  |     for singleton in singletons.iter_mut() { | ||||||
|  |         if singleton.name.starts_with("TIM") { | ||||||
|  |             // Remove suffixes for pin singletons.
 | ||||||
|  |             let name = if singleton.name.contains("_CCP") { | ||||||
|  |                 singleton.name.split_once("_CCP").unwrap().0 | ||||||
|  |             } else if singleton.name.contains("_FAULT") { | ||||||
|  |                 singleton.name.split_once("_FAULT").unwrap().0 | ||||||
|  |             } else if singleton.name.contains("_IDX") { | ||||||
|  |                 singleton.name.split_once("_IDX").unwrap().0 | ||||||
|  |             } else { | ||||||
|  |                 &singleton.name | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             let feature = format!("time-driver-{}", name.to_lowercase()); | ||||||
|  | 
 | ||||||
|  |             if singleton.name.contains(selected_timer) { | ||||||
|  |                 singleton.cfg = Some(quote! { #[cfg(not(all(feature = "time-driver-any", feature = #feature)))] }); | ||||||
|  |             } else { | ||||||
|  |                 singleton.cfg = Some(quote! { #[cfg(not(feature = #feature))] }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_singletons(singletons: &[Singleton]) -> TokenStream { | ||||||
|  |     let singletons = singletons | ||||||
|  |         .iter() | ||||||
|  |         .map(|s| { | ||||||
|  |             let cfg = s.cfg.clone().unwrap_or_default(); | ||||||
|  | 
 | ||||||
|  |             let ident = format_ident!("{}", s.name); | ||||||
|  | 
 | ||||||
|  |             quote! { | ||||||
|  |                 #cfg | ||||||
|  |                 #ident | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |         .collect::<Vec<_>>(); | ||||||
|  | 
 | ||||||
|  |     quote! { | ||||||
|  |         embassy_hal_internal::peripherals_definition!(#(#singletons),*); | ||||||
|  |         embassy_hal_internal::peripherals_struct!(#(#singletons),*); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_timers() -> TokenStream { | ||||||
|  |     // Generate timers
 | ||||||
|  |     let timer_impls = METADATA | ||||||
|  |         .peripherals | ||||||
|  |         .iter() | ||||||
|  |         .filter(|p| p.name.starts_with("TIM")) | ||||||
|  |         .map(|peripheral| { | ||||||
|  |             let name = Ident::new(&peripheral.name, Span::call_site()); | ||||||
|  |             let timers = &*TIMERS; | ||||||
|  | 
 | ||||||
|  |             let timer = timers.get(peripheral.name).expect("Timer does not exist"); | ||||||
|  |             assert!(timer.bits == 16 || timer.bits == 32); | ||||||
|  |             let bits = if timer.bits == 16 { | ||||||
|  |                 quote! { Bits16 } | ||||||
|  |             } else { | ||||||
|  |                 quote! { Bits32 } | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             quote! { | ||||||
|  |                 impl_timer!(#name, #bits); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |     quote! { | ||||||
|  |         #(#timer_impls)* | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_interrupts() -> TokenStream { | ||||||
|  |     // Generate interrupt module
 | ||||||
|  |     let interrupts: Vec<Ident> = METADATA | ||||||
|  |         .interrupts | ||||||
|  |         .iter() | ||||||
|  |         .map(|interrupt| Ident::new(interrupt.name, Span::call_site())) | ||||||
|  |         .collect(); | ||||||
|  | 
 | ||||||
|  |     let group_interrupt_enables = METADATA | ||||||
|  |         .interrupts | ||||||
|  |         .iter() | ||||||
|  |         .filter(|interrupt| interrupt.name.contains("GROUP")) | ||||||
|  |         .map(|interrupt| { | ||||||
|  |             let name = Ident::new(interrupt.name, Span::call_site()); | ||||||
|  | 
 | ||||||
|  |             quote! { | ||||||
|  |                 crate::interrupt::typelevel::#name::enable(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |     // Generate interrupt enables for groups
 | ||||||
|  |     quote! { | ||||||
|  |         embassy_hal_internal::interrupt_mod! { | ||||||
|  |             #(#interrupts),* | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         pub fn enable_group_interrupts(_cs: critical_section::CriticalSection) { | ||||||
|  |             use crate::interrupt::typelevel::Interrupt; | ||||||
|  | 
 | ||||||
|  |             unsafe { | ||||||
|  |                 #(#group_interrupt_enables)* | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_peripheral_instances() -> TokenStream { | ||||||
|  |     let mut impls = Vec::<TokenStream>::new(); | ||||||
|  | 
 | ||||||
|  |     for peripheral in METADATA.peripherals { | ||||||
|  |         let peri = format_ident!("{}", peripheral.name); | ||||||
|  | 
 | ||||||
|  |         // Will be filled in when uart implementation is finished
 | ||||||
|  |         let _ = peri; | ||||||
|  |         let tokens = match peripheral.kind { | ||||||
|  |             // "uart" => Some(quote! { impl_uart_instance!(#peri); }),
 | ||||||
|  |             _ => None, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         if let Some(tokens) = tokens { | ||||||
|  |             impls.push(tokens); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     quote! { | ||||||
|  |         #(#impls)* | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn generate_pin_trait_impls() -> TokenStream { | ||||||
|  |     let mut impls = Vec::<TokenStream>::new(); | ||||||
|  | 
 | ||||||
|  |     for peripheral in METADATA.peripherals { | ||||||
|  |         for pin in peripheral.pins { | ||||||
|  |             let key = (peripheral.kind, pin.signal); | ||||||
|  | 
 | ||||||
|  |             let pin_name = format_ident!("{}", pin.pin); | ||||||
|  |             let peri = format_ident!("{}", peripheral.name); | ||||||
|  |             let pf = pin.pf; | ||||||
|  | 
 | ||||||
|  |             // Will be filled in when uart implementation is finished
 | ||||||
|  |             let _ = pin_name; | ||||||
|  |             let _ = peri; | ||||||
|  |             let _ = pf; | ||||||
|  | 
 | ||||||
|  |             let tokens = match key { | ||||||
|  |                 // ("uart", "TX") => Some(quote! { impl_uart_tx_pin!(#peri, #pin_name, #pf); }),
 | ||||||
|  |                 // ("uart", "RX") => Some(quote! { impl_uart_rx_pin!(#peri, #pin_name, #pf); }),
 | ||||||
|  |                 // ("uart", "CTS") => Some(quote! { impl_uart_cts_pin!(#peri, #pin_name, #pf); }),
 | ||||||
|  |                 // ("uart", "RTS") => Some(quote! { impl_uart_rts_pin!(#peri, #pin_name, #pf); }),
 | ||||||
|  |                 _ => None, | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             if let Some(tokens) = tokens { | ||||||
|  |                 impls.push(tokens); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     quote! { | ||||||
|  |         #(#impls)* | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -8,6 +8,25 @@ pub(crate) mod fmt; | |||||||
| pub mod gpio; | pub mod gpio; | ||||||
| pub mod timer; | pub mod timer; | ||||||
| 
 | 
 | ||||||
|  | /// Operating modes for peripherals.
 | ||||||
|  | pub mod mode { | ||||||
|  |     trait SealedMode {} | ||||||
|  | 
 | ||||||
|  |     /// Operating mode for a peripheral.
 | ||||||
|  |     #[allow(private_bounds)] | ||||||
|  |     pub trait Mode: SealedMode {} | ||||||
|  | 
 | ||||||
|  |     /// Blocking mode.
 | ||||||
|  |     pub struct Blocking; | ||||||
|  |     impl SealedMode for Blocking {} | ||||||
|  |     impl Mode for Blocking {} | ||||||
|  | 
 | ||||||
|  |     /// Async mode.
 | ||||||
|  |     pub struct Async; | ||||||
|  |     impl SealedMode for Async {} | ||||||
|  |     impl Mode for Async {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[cfg(feature = "_time-driver")] | #[cfg(feature = "_time-driver")] | ||||||
| mod time_driver; | mod time_driver; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,6 +12,9 @@ use mspm0_metapac::tim::{Counterregs16, Tim}; | |||||||
| use crate::peripherals; | use crate::peripherals; | ||||||
| use crate::timer::SealedTimer; | use crate::timer::SealedTimer; | ||||||
| 
 | 
 | ||||||
|  | #[cfg(any(time_driver_timg12, time_driver_timg13))] | ||||||
|  | compile_error!("TIMG12 and TIMG13 are not supported by the time driver yet"); | ||||||
|  | 
 | ||||||
| // Currently TIMG12 and TIMG13 are excluded because those are 32-bit timers.
 | // Currently TIMG12 and TIMG13 are excluded because those are 32-bit timers.
 | ||||||
| #[cfg(time_driver_timg0)] | #[cfg(time_driver_timg0)] | ||||||
| type T = peripherals::TIMG0; | type T = peripherals::TIMG0; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user