1#![allow(missing_docs)]
2#![cfg(feature = "wifi")]
6#![no_std]
7#![no_main]
8#![allow(clippy::future_not_send, reason = "single-threaded")]
9
10use core::convert::Infallible;
11use defmt::{info, warn};
12use defmt_rtt as _;
13use device_envoy::button::PressedTo;
14use device_envoy::clock_sync::UnixSeconds;
15use device_envoy::flash_array::FlashArray;
16use device_envoy::led4::{BlinkState, Led4, Led4Static, OutputArray, circular_outline_animation};
17use device_envoy::wifi_auto::fields::{
18 TextField, TextFieldStatic, TimezoneField, TimezoneFieldStatic,
19};
20use device_envoy::wifi_auto::{WifiAuto, WifiAutoEvent};
21use device_envoy::{Error, Result};
22use embassy_executor::Spawner;
23use embassy_net::{Stack, dns::DnsQueryType, udp};
24use embassy_rp::gpio::{self, Level};
25use embassy_time::Duration;
26use heapless::String;
27use panic_probe as _;
28
29#[embassy_executor::main]
30pub async fn main(spawner: Spawner) -> ! {
31 let err = inner_main(spawner).await.unwrap_err();
32 core::panic!("{err}");
33}
34
35async fn inner_main(spawner: Spawner) -> Result<Infallible> {
36 info!("Starting wifi_auto example");
37 let p = embassy_rp::init(Default::default());
38
39 let cells = OutputArray::new([
41 gpio::Output::new(p.PIN_1, Level::High),
42 gpio::Output::new(p.PIN_2, Level::High),
43 gpio::Output::new(p.PIN_3, Level::High),
44 gpio::Output::new(p.PIN_4, Level::High),
45 ]);
46
47 let segments = OutputArray::new([
48 gpio::Output::new(p.PIN_5, Level::Low),
49 gpio::Output::new(p.PIN_6, Level::Low),
50 gpio::Output::new(p.PIN_7, Level::Low),
51 gpio::Output::new(p.PIN_8, Level::Low),
52 gpio::Output::new(p.PIN_9, Level::Low),
53 gpio::Output::new(p.PIN_10, Level::Low),
54 gpio::Output::new(p.PIN_11, Level::Low),
55 gpio::Output::new(p.PIN_12, Level::Low),
56 ]);
57
58 static LED4_STATIC: Led4Static = Led4::new_static();
59 let led4 = Led4::new(&LED4_STATIC, cells, segments, spawner)?;
60
61 let [
62 wifi_credentials_flash_block,
63 timezone_flash_block,
64 device_name_flash_block,
65 location_flash_block,
66 ] = FlashArray::<4>::new(p.FLASH)?;
67
68 static TIMEZONE_FIELD_STATIC: TimezoneFieldStatic = TimezoneField::new_static();
69 let timezone_field = TimezoneField::new(&TIMEZONE_FIELD_STATIC, timezone_flash_block);
70
71 static DEVICE_NAME_FIELD_STATIC: TextFieldStatic<32> = TextField::new_static();
72 let device_name_field = TextField::new(
73 &DEVICE_NAME_FIELD_STATIC,
74 device_name_flash_block,
75 "device_name",
76 "Device Name",
77 "www.picoclock.net",
78 );
79
80 static LOCATION_FIELD_STATIC: TextFieldStatic<64> = TextField::new_static();
81 let location_field = TextField::new(
82 &LOCATION_FIELD_STATIC,
83 location_flash_block,
84 "location",
85 "Location",
86 "Living Room",
87 );
88
89 let wifi_auto = WifiAuto::new(
90 p.PIN_23, p.PIN_24, p.PIN_25, p.PIN_29, p.PIO0, p.DMA_CH0, wifi_credentials_flash_block, p.PIN_13, PressedTo::Ground, "Pico", [timezone_field, device_name_field, location_field],
101 spawner,
102 )?;
103
104 let led4_ref = &led4;
105 let (stack, mut button) = wifi_auto
106 .connect(|event| async move {
107 match event {
108 WifiAutoEvent::CaptivePortalReady => {
109 led4_ref.write_text(['C', 'O', 'N', 'N'], BlinkState::BlinkingAndOn);
110 }
111
112 WifiAutoEvent::Connecting { try_index, .. } => {
113 led4_ref.animate_text(circular_outline_animation((try_index & 1) == 0));
114 }
115
116 WifiAutoEvent::ConnectionFailed => {
117 led4_ref.write_text(['F', 'A', 'I', 'L'], BlinkState::BlinkingButOff);
118 }
119 }
120 Ok(())
121 })
122 .await?;
123
124 led4.write_text(['D', 'O', 'N', 'E'], BlinkState::Solid);
125
126 let timezone_offset_minutes = timezone_field
127 .offset_minutes()?
128 .ok_or(Error::MissingCustomWifiAutoField)?;
129 let device_name = device_name_field.text()?.unwrap_or_else(|| {
130 let mut name = String::new();
131 name.push_str("").expect("default name exceeds capacity");
132 name
133 });
134 let location = location_field.text()?.unwrap_or_else(|| {
135 let mut name = String::new();
136 name.push_str("Living Room")
137 .expect("default location exceeds capacity");
138 name
139 });
140 info!(
141 "Device '{}' in '{}' configured with timezone offset {} minutes",
142 device_name, location, timezone_offset_minutes
143 );
144
145 info!("WiFi setup complete - press button to fetch NTP time");
148 loop {
149 button.wait_for_press().await;
150 match fetch_ntp_time(stack).await {
151 Ok(unix_seconds) => info!("Current time: {}", unix_seconds.as_i64()),
152 Err(err) => warn!("Failed to fetch time: {}", err),
153 }
154 }
155}
156
157async fn fetch_ntp_time(stack: &'static Stack<'static>) -> Result<UnixSeconds, &'static str> {
158 use udp::UdpSocket;
159
160 const NTP_SERVER: &str = "pool.ntp.org";
161 const NTP_PORT: u16 = 123;
162
163 info!("Resolving {}...", NTP_SERVER);
164 let dns_result = stack
165 .dns_query(NTP_SERVER, DnsQueryType::A)
166 .await
167 .map_err(|e| {
168 warn!("DNS lookup failed: {:?}", e);
169 "DNS lookup failed"
170 })?;
171 let server_addr = dns_result.first().ok_or("No DNS results")?;
172
173 let mut rx_meta = [udp::PacketMetadata::EMPTY; 1];
174 let mut rx_buffer = [0; 128];
175 let mut tx_meta = [udp::PacketMetadata::EMPTY; 1];
176 let mut tx_buffer = [0; 128];
177 let mut socket = UdpSocket::new(
178 *stack,
179 &mut rx_meta,
180 &mut rx_buffer,
181 &mut tx_meta,
182 &mut tx_buffer,
183 );
184
185 socket.bind(0).map_err(|e| {
186 warn!("Socket bind failed: {:?}", e);
187 "Socket bind failed"
188 })?;
189
190 let mut ntp_request = [0u8; 48];
191 ntp_request[0] = 0x1B;
192 info!("Sending NTP request...");
193 socket
194 .send_to(&ntp_request, (*server_addr, NTP_PORT))
195 .await
196 .map_err(|e| {
197 warn!("NTP send failed: {:?}", e);
198 "NTP send failed"
199 })?;
200
201 let mut response = [0u8; 48];
202 let (n, _) =
203 embassy_time::with_timeout(Duration::from_secs(5), socket.recv_from(&mut response))
204 .await
205 .map_err(|_| {
206 warn!("NTP receive timeout");
207 "NTP receive timeout"
208 })?
209 .map_err(|e| {
210 warn!("NTP receive failed: {:?}", e);
211 "NTP receive failed"
212 })?;
213
214 if n < 48 {
215 warn!("NTP response too short: {} bytes", n);
216 return Err("NTP response too short");
217 }
218
219 let ntp_seconds = u32::from_be_bytes([response[40], response[41], response[42], response[43]]);
220 UnixSeconds::from_ntp_seconds(ntp_seconds).ok_or("Invalid NTP timestamp")
221}