ds3231.rs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. use field_offset::offset_of;
  2. use stm32l4xx_hal::prelude::{
  3. _embedded_hal_blocking_i2c_Write, _embedded_hal_blocking_i2c_WriteRead,
  4. };
  5. use tock_registers::{
  6. interfaces::{ReadWriteable, Readable, Writeable},
  7. register_bitfields, register_structs,
  8. registers::InMemoryRegister,
  9. };
  10. const DS3231_ADDR: u8 = 0x68;
  11. register_bitfields![
  12. u8,
  13. T_SECOND [
  14. Value OFFSET(0) NUMBITS(7) [],
  15. ],
  16. T_MINUTE [
  17. Value OFFSET(0) NUMBITS(7) [],
  18. ],
  19. T_HOUR [
  20. Value OFFSET(0) NUMBITS(6) [],
  21. n24 OFFSET(6) NUMBITS(1) [],
  22. ],
  23. T_WEEKDAY [
  24. Value OFFSET(0) NUMBITS(3) [],
  25. ],
  26. T_DAY [
  27. Value OFFSET(0) NUMBITS(6) [],
  28. ],
  29. T_MONTH [
  30. Value OFFSET(0) NUMBITS(5) [],
  31. Century OFFSET(7) NUMBITS(1) [],
  32. ],
  33. T_YEAR [
  34. Value OFFSET(0) NUMBITS(8) [],
  35. ],
  36. A_SECOND [
  37. Value OFFSET(0) NUMBITS(7) [],
  38. M1 OFFSET(7) NUMBITS(1) [],
  39. ],
  40. A_MINUTE [
  41. Value OFFSET(0) NUMBITS(7) [],
  42. M2 OFFSET(7) NUMBITS(1) [],
  43. ],
  44. A_HOUR [
  45. Value OFFSET(0) NUMBITS(6) [],
  46. n24 OFFSET(6) NUMBITS(1) [],
  47. M3 OFFSET(7) NUMBITS(1) [],
  48. ],
  49. A_DAY_DATE [
  50. Value OFFSET(0) NUMBITS(6) [],
  51. nDT OFFSET(6) NUMBITS(1) [],
  52. M4 OFFSET(7) NUMBITS(1) [],
  53. ],
  54. CONTROL_1 [
  55. A1IE 0,
  56. A2IE 1,
  57. INTCN 2,
  58. RS1 3,
  59. RS2 4,
  60. CONV 5,
  61. BBSQW 6,
  62. nEOSC 7,
  63. ],
  64. CONTROL_2 [
  65. A1F 0,
  66. A2F 1,
  67. BSY 2,
  68. EN32KHZ 3,
  69. OSF 7,
  70. ],
  71. AGING_OFFSET [
  72. Value OFFSET(0) NUMBITS(7) [],
  73. Sign OFFSET(7) NUMBITS(1) [],
  74. ],
  75. TEMP_MSB [
  76. Value OFFSET(0) NUMBITS(7) [],
  77. Sign OFFSET(7) NUMBITS(1) [],
  78. ],
  79. TEMP_LSB [
  80. Value OFFSET(6) NUMBITS(2) [],
  81. ],
  82. ];
  83. register_structs! {
  84. Ds3231Registers {
  85. (0x00 => second: InMemoryRegister<u8, T_SECOND::Register>),
  86. (0x01 => minute: InMemoryRegister<u8, T_MINUTE::Register>),
  87. (0x02 => hour: InMemoryRegister<u8, T_HOUR::Register>),
  88. (0x03 => weekday: InMemoryRegister<u8, T_WEEKDAY::Register>),
  89. (0x04 => day: InMemoryRegister<u8, T_DAY::Register>),
  90. (0x05 => month: InMemoryRegister<u8, T_MONTH::Register>),
  91. (0x06 => year: InMemoryRegister<u8, T_YEAR::Register>),
  92. (0x07 => a1_second: InMemoryRegister<u8, A_SECOND::Register>),
  93. (0x08 => a1_minute: InMemoryRegister<u8, A_MINUTE::Register>),
  94. (0x09 => a1_hour: InMemoryRegister<u8, A_HOUR::Register>),
  95. (0x0A => a1_day_date: InMemoryRegister<u8, A_DAY_DATE::Register>),
  96. (0x0B => a2_minute: InMemoryRegister<u8, A_MINUTE::Register>),
  97. (0x0C => a2_hour: InMemoryRegister<u8, A_HOUR::Register>),
  98. (0x0D => a2_day_date: InMemoryRegister<u8, A_DAY_DATE::Register>),
  99. (0x0E => control_1: InMemoryRegister<u8, CONTROL_1::Register>),
  100. (0x0F => control_2: InMemoryRegister<u8, CONTROL_2::Register>),
  101. (0x10 => aging_offset: InMemoryRegister<u8, AGING_OFFSET::Register>),
  102. (0x11 => temp_msb: InMemoryRegister<u8, TEMP_MSB::Register>),
  103. (0x12 => temp_lsb: InMemoryRegister<u8, TEMP_LSB::Register>),
  104. (0x13 => @END),
  105. }
  106. }
  107. pub enum Weekday {
  108. Sunday = 0,
  109. Monday = 1,
  110. Tuesday = 2,
  111. Wednesday = 3,
  112. Thursday = 4,
  113. Friday = 5,
  114. Saturday = 6,
  115. }
  116. impl From<u8> for Weekday {
  117. fn from(value: u8) -> Self {
  118. match value {
  119. 0 => Weekday::Sunday,
  120. 1 => Weekday::Monday,
  121. 2 => Weekday::Tuesday,
  122. 3 => Weekday::Wednesday,
  123. 4 => Weekday::Thursday,
  124. 5 => Weekday::Friday,
  125. 6 => Weekday::Saturday,
  126. _ => panic!(),
  127. }
  128. }
  129. }
  130. impl Ds3231Registers {
  131. const fn default() -> Self {
  132. Self {
  133. second: InMemoryRegister::new(0x00),
  134. minute: InMemoryRegister::new(0x00),
  135. hour: InMemoryRegister::new(0x00),
  136. weekday: InMemoryRegister::new(0x00),
  137. day: InMemoryRegister::new(0x00),
  138. month: InMemoryRegister::new(0x00),
  139. year: InMemoryRegister::new(0x00),
  140. a1_second: InMemoryRegister::new(0x00),
  141. a1_minute: InMemoryRegister::new(0x00),
  142. a1_hour: InMemoryRegister::new(0x00),
  143. a1_day_date: InMemoryRegister::new(0x00),
  144. a2_minute: InMemoryRegister::new(0x00),
  145. a2_hour: InMemoryRegister::new(0x00),
  146. a2_day_date: InMemoryRegister::new(0x00),
  147. control_1: InMemoryRegister::new(0x00),
  148. control_2: InMemoryRegister::new(0x00),
  149. aging_offset: InMemoryRegister::new(0x00),
  150. temp_msb: InMemoryRegister::new(0x00),
  151. temp_lsb: InMemoryRegister::new(0x00),
  152. }
  153. }
  154. fn i2c_write_ctrl(&self, address: u8, i2c: &mut impl _embedded_hal_blocking_i2c_Write) {
  155. let mut buffer: [u8; 3] = [0; 3];
  156. buffer[0] = offset_of!(Ds3231Registers => control_1).get_byte_offset() as u8;
  157. buffer[1] = self.control_1.get();
  158. buffer[2] = self.control_2.get();
  159. match i2c.write(address, &buffer) {
  160. Ok(_) => (),
  161. Err(_) => panic!(),
  162. }
  163. }
  164. fn i2c_write_time(&self, address: u8, i2c: &mut impl _embedded_hal_blocking_i2c_Write) {
  165. let mut buffer: [u8; 4] = [0; 4];
  166. buffer[0] = offset_of!(Ds3231Registers => second).get_byte_offset() as u8;
  167. buffer[1] = self.second.get();
  168. buffer[2] = self.minute.get();
  169. buffer[3] = self.hour.get();
  170. match i2c.write(address, &buffer) {
  171. Ok(_) => (),
  172. Err(_) => panic!(),
  173. }
  174. }
  175. fn i2c_write_date(&self, address: u8, i2c: &mut impl _embedded_hal_blocking_i2c_Write) {
  176. let mut buffer: [u8; 5] = [0; 5];
  177. buffer[0] = offset_of!(Ds3231Registers => weekday).get_byte_offset() as u8;
  178. buffer[1] = self.weekday.get();
  179. buffer[2] = self.day.get();
  180. buffer[3] = self.month.get();
  181. buffer[4] = self.year.get();
  182. match i2c.write(address, &buffer) {
  183. Ok(_) => (),
  184. Err(_) => panic!(),
  185. }
  186. }
  187. // Returns the second, minute, and hour from the RTC
  188. fn i2c_read_time(&mut self, address: u8, i2c: &mut impl _embedded_hal_blocking_i2c_WriteRead) {
  189. let mut buffer: [u8; 3] = [0; 3];
  190. match i2c.write_read(
  191. address,
  192. &[offset_of!(Ds3231Registers => second).get_byte_offset() as u8],
  193. &mut buffer,
  194. ) {
  195. Ok(_) => (),
  196. Err(_) => panic!(),
  197. }
  198. }
  199. // Returns the weekday, day, month, year, and century from the RTC
  200. fn i2c_read_date(&mut self, address: u8, i2c: &mut impl _embedded_hal_blocking_i2c_WriteRead) {
  201. let mut buffer: [u8; 4] = [0; 4];
  202. match i2c.write_read(
  203. address,
  204. &[offset_of!(Ds3231Registers => weekday).get_byte_offset() as u8],
  205. &mut buffer,
  206. ) {
  207. Ok(_) => (),
  208. Err(_) => panic!(),
  209. }
  210. }
  211. }
  212. pub fn init(i2c: &mut impl _embedded_hal_blocking_i2c_Write) {
  213. let regs = Ds3231Registers::default();
  214. regs.control_1.modify(
  215. CONTROL_1::nEOSC::CLEAR // Enable internal oscillator on VBAT
  216. + CONTROL_1::BBSQW::CLEAR // Disable outputs on VBAT
  217. + CONTROL_1::RS1::CLEAR // Set square wave output to 1Hz
  218. + CONTROL_1::RS2::CLEAR // Set square wave output to 1Hz
  219. + CONTROL_1::INTCN::CLEAR // Enable square wave output
  220. + CONTROL_1::A1IE::CLEAR // Disable alarm 1
  221. + CONTROL_1::A2IE::CLEAR, // Disable alarm 2
  222. );
  223. // Disable 32kHz output
  224. regs.control_2.modify(CONTROL_2::EN32KHZ::CLEAR);
  225. regs.i2c_write_ctrl(DS3231_ADDR, i2c);
  226. }
  227. pub fn set_time(
  228. i2c: &mut impl _embedded_hal_blocking_i2c_Write,
  229. second: u32,
  230. minute: u32,
  231. hour: u32,
  232. ) {
  233. let regs = Ds3231Registers::default();
  234. regs.second
  235. .write(T_SECOND::Value.val(decimal_to_bcd(second)));
  236. regs.minute
  237. .write(T_MINUTE::Value.val(decimal_to_bcd(minute)));
  238. regs.hour.write(T_HOUR::Value.val(decimal_to_bcd(hour)));
  239. regs.i2c_write_time(DS3231_ADDR, i2c);
  240. }
  241. pub fn set_date(
  242. i2c: &mut impl _embedded_hal_blocking_i2c_Write,
  243. weekday: Weekday,
  244. day: u32,
  245. month: u32,
  246. year: u32,
  247. century: u32,
  248. ) {
  249. let regs = Ds3231Registers::default();
  250. regs.weekday
  251. .write(T_WEEKDAY::Value.val(decimal_to_bcd(weekday as u32)));
  252. regs.day.write(T_DAY::Value.val(decimal_to_bcd(day)));
  253. regs.month.modify(
  254. T_MONTH::Value.val(decimal_to_bcd(month)) + T_MONTH::Century.val(decimal_to_bcd(century)),
  255. );
  256. regs.year.write(T_YEAR::Value.val(decimal_to_bcd(year)));
  257. regs.i2c_write_time(DS3231_ADDR, i2c);
  258. }
  259. pub fn get_time(
  260. i2c: &mut impl _embedded_hal_blocking_i2c_WriteRead,
  261. ) -> (u32, u32, u32) {
  262. let mut regs = Ds3231Registers::default();
  263. regs.i2c_read_time(DS3231_ADDR, i2c);
  264. let second = bcd_to_decimal(regs.second.read(T_SECOND::Value));
  265. let minute = bcd_to_decimal(regs.minute.read(T_MINUTE::Value));
  266. let hour = bcd_to_decimal(regs.hour.read(T_HOUR::Value));
  267. (second, minute, hour)
  268. }
  269. pub fn get_date(
  270. i2c: &mut impl _embedded_hal_blocking_i2c_WriteRead,
  271. ) -> (Weekday, u32, u32, u32, u32) {
  272. let mut regs = Ds3231Registers::default();
  273. regs.i2c_read_date(DS3231_ADDR, i2c);
  274. let weekday: Weekday = regs.weekday.read(T_WEEKDAY::Value).into();
  275. let day = bcd_to_decimal(regs.day.read(T_DAY::Value));
  276. let month = bcd_to_decimal(regs.month.read(T_MONTH::Value));
  277. let year = bcd_to_decimal(regs.year.read(T_YEAR::Value));
  278. let century = bcd_to_decimal(regs.month.read(T_MONTH::Century));
  279. (weekday, day, month, year, century)
  280. }
  281. #[inline]
  282. fn bcd_to_decimal(value: u8) -> u32 {
  283. (((value >> 4) * 10) + (value & 0xF)) as u32
  284. }
  285. #[inline]
  286. fn decimal_to_bcd(value: u32) -> u8 {
  287. (((value / 10) << 4) | (value % 10)) as u8
  288. }
  289. pub fn in_dst(weekday: Weekday, day: u32, month: u32, hour_24: u32) -> bool {
  290. let prev_sunday: i32 = day as i32 - weekday as i32;
  291. match month {
  292. 0..=2 | 12.. => false,
  293. 4..=10 => true,
  294. 3 => match prev_sunday {
  295. ..8 => false,
  296. 15.. => true,
  297. d if d == day as i32 => hour_24 >= 2,
  298. _ => true,
  299. },
  300. 11 => match prev_sunday {
  301. ..=0 => true,
  302. 8.. => false,
  303. d if d == day as i32 => hour_24 < 2,
  304. _ => false,
  305. },
  306. }
  307. }
  308. #[cfg(test)]
  309. mod test {
  310. use super::*;
  311. #[test]
  312. fn dst_test() {
  313. // 2020 - begins Mar 8th @ 2AM (Sunday), ends Nov 1st @ 2AM (Sunday)
  314. assert!(in_dst(Weekday::Sunday, 8, 3, 1) == false);
  315. assert!(in_dst(Weekday::Sunday, 8, 3, 2) == true);
  316. assert!(in_dst(Weekday::Sunday, 8, 3, 3) == true);
  317. assert!(in_dst(Weekday::Sunday, 1, 11, 1) == true);
  318. assert!(in_dst(Weekday::Sunday, 1, 11, 2) == false);
  319. assert!(in_dst(Weekday::Sunday, 1, 11, 3) == false);
  320. // 2021 - begins Mar 14th @ 2AM (Sunday), ends Nov 7th @ 2AM (Sunday)
  321. assert!(in_dst(Weekday::Sunday, 14, 3, 1) == false);
  322. assert!(in_dst(Weekday::Sunday, 14, 3, 2) == true);
  323. assert!(in_dst(Weekday::Sunday, 14, 3, 3) == true);
  324. assert!(in_dst(Weekday::Sunday, 7, 11, 1) == true);
  325. assert!(in_dst(Weekday::Sunday, 7, 11, 2) == false);
  326. assert!(in_dst(Weekday::Sunday, 7, 11, 3) == false);
  327. // 2022 - begins Mar 13th @ 2AM (Sunday), ends Nov 6th @ 2AM (Sunday)
  328. assert!(in_dst(Weekday::Sunday, 13, 3, 1) == false);
  329. assert!(in_dst(Weekday::Sunday, 13, 3, 2) == true);
  330. assert!(in_dst(Weekday::Sunday, 13, 3, 3) == true);
  331. assert!(in_dst(Weekday::Sunday, 6, 11, 1) == true);
  332. assert!(in_dst(Weekday::Sunday, 6, 11, 2) == false);
  333. assert!(in_dst(Weekday::Sunday, 6, 11, 3) == false);
  334. // 2023 - begins Mar 12th @ 2AM (Sunday), ends Nov 5th @ 2AM (Sunday)
  335. assert!(in_dst(Weekday::Sunday, 12, 3, 1) == false);
  336. assert!(in_dst(Weekday::Sunday, 12, 3, 2) == true);
  337. assert!(in_dst(Weekday::Sunday, 12, 3, 3) == true);
  338. assert!(in_dst(Weekday::Sunday, 5, 11, 1) == true);
  339. assert!(in_dst(Weekday::Sunday, 5, 11, 2) == false);
  340. assert!(in_dst(Weekday::Sunday, 5, 11, 3) == false);
  341. // Sanity check other dates in 2021
  342. assert!(in_dst(Weekday::Friday, 1, 1, 0) == false);
  343. assert!(in_dst(Weekday::Monday, 1, 2, 0) == false);
  344. assert!(in_dst(Weekday::Monday, 1, 3, 0) == false);
  345. assert!(in_dst(Weekday::Thursday, 1, 4, 0) == true);
  346. assert!(in_dst(Weekday::Saturday, 1, 5, 0) == true);
  347. assert!(in_dst(Weekday::Tuesday, 1, 6, 0) == true);
  348. assert!(in_dst(Weekday::Thursday, 1, 7, 0) == true);
  349. assert!(in_dst(Weekday::Sunday, 1, 8, 0) == true);
  350. assert!(in_dst(Weekday::Wednesday, 1, 9, 0) == true);
  351. assert!(in_dst(Weekday::Friday, 1, 10, 0) == true);
  352. assert!(in_dst(Weekday::Monday, 1, 11, 0) == true);
  353. assert!(in_dst(Weekday::Wednesday, 1, 12, 0) == false);
  354. }
  355. }