//! This is a platform agnostic Rust driver for the DS3231, DS3232 and DS3234 //! extremely accurate real-time clocks, based on the [`embedded-hal`] traits. //! //! [`embedded-hal`]: https://github.com/rust-embedded/embedded-hal //! //! This driver allows you to: //! - Read/write the seconds. //! - Read/write the minutes. //! //! ## The devices //! //! This driver is compatible with the DS3231 and DS3232 I2C devices and the //! DS3234 SPI device. //! //! ### DS3231 //! The DS3231 is a low-cost, extremely accurate I2C real-time clock (RTC) with //! an integrated temperature-compensated crystal oscillator (TCXO) and crystal. //! //! The device incorporates a battery input, and maintains accurate timekeeping //! when main power to the device is interrupted. The integration of the //! crystal resonator enhances the long-term accuracy of the device as well as //! reduces the piece-part count in a manufacturing line. //! The DS3231 is available in commercial and industrial temperature ranges, //! and is offered in a 16-pin, 300-mil SO package. //! //! The RTC maintains seconds, minutes, hours, day, date, month, and year //! information. The date at the end of the month is automatically adjusted for //! months with fewer than 31 days, including corrections for leap year. The //! clock operates in either the 24-hour or 12-hour format with an AM/PM //! indicator. Two programmable time-of-day alarms and a programmable //! square-wave output are provided. Address and data are transferred serially //! through an I2C bidirectional bus. //! //! A precision temperature-compensated voltage reference and comparator //! circuit monitors the status of VCC to detect power failures, to provide a //! reset output, and to automatically switch to the backup supply when //! necessary. Additionally, the RST pin is monitored as a pushbutton //! input for generating a μP reset. //! //! ### DS3232 //! The DS3232 is a low-cost temperature-compensated crystal oscillator (TCXO) //! with a very accurate, temperature-compensated, integrated real-time clock //! (RTC) and 236 bytes of battery-backed SRAM. //! //! Additionally, the DS3232 incorporates a battery input and maintains //! accurate timekeeping when main power to the device is interrupted. The //! integration of the crystal resonator enhances the long-term accuracy of the //! device as well as reduces the piece-part count in a manufacturing line. //! The DS3232 is available in commercial and industrial temperature ranges, //! and is offered in an industry-standard 20-pin, 300-mil SO package. //! //! The RTC maintains seconds, minutes, hours, day, date, month, and year //! information. The date at the end of the month is automatically adjusted for //! months with fewer than 31 days, including corrections for leap year. The //! clock operates in either the 24-hour or 12-hour format with an AM/PM //! indicator. Two programmable time-of-day alarms and a programmable //! square-wave output are provided. Address and data are transferred serially //! through an I2C bidirectional bus. //! //! A precision temperature-compensated voltage reference and comparator //! circuit monitors the status of VCC to detect power failures, to provide a //! reset output, and to automatically switch to the backup supply when //! necessary. Additionally, the RST pin is monitored as a pushbutton input for //! generating a μP reset. //! //! ### DS3234 //! The DS3234 is a low-cost, extremely accurate SPI bus real-time clock (RTC) //! with an integrated temperature-compensated crystal oscillator (TCXO) and //! crystal. //! //! The DS3234 incorporates a precision, temperature-compensated voltage //! reference and comparator circuit to monitor VCC. When VCC drops below the //! power-fail voltage (VPF), the device asserts the RST output and also //! disables read and write access to the part when VCC drops below both VPF //! and VBAT. The RST pin is monitored as a pushbutton input for generating a //! μP reset. The device switches to the backup supply input and maintains //! accurate timekeeping when main power to the device is interrupted. //! The integration of the crystal resonator enhances the long-term accuracy of //! the device as well as reduces the piece-part count in a manufacturing line. //! The DS3234 is available in commercial and industrial temperature ranges, //! and is offered in an industry-standard 300-mil, 20-pin SO package. //! //! The DS3234 also integrates 256 bytes of battery-backed SRAM. In the event //! of main power loss, the contents of the memory are maintained by the power //! source connected to the V BAT pin. The RTC maintains seconds, minutes, //! hours, day, date, month, and year information. The date at the end of the //! month is automatically adjusted for months with fewer than 31 days, //! including corrections for leap year. The clock operates in either the //! 24-hour or 12-hour format with AM/PM indicator. Two programmable //! time-of-day alarms and a programmable square-wave output are provided. //! Address and data are transferred serially by an SPI bidirectional bus. //! //! Datasheets: //! - [DS3231](https://datasheets.maximintegrated.com/en/ds/DS3231.pdf) //! - [DS3232](https://datasheets.maximintegrated.com/en/ds/DS3232.pdf) //! - [DS3234](https://datasheets.maximintegrated.com/en/ds/DS3234.pdf) //! //! ## Usage examples (see also examples folder) //! //! To use this driver, import this crate and an `embedded_hal` implementation, //! then instantiate the appropriate device. //! In the following 3 examples an instance of the devices DS3231, DS3232 and //! DS3234 will be created as an example. The rest of examples will use the //! DS3231 as an example, except when using features specific to another IC, //! for example, RAM access which is not available in the DS3231 device. //! //! ### Create a driver instance for the DS3231 //! //! ```no_run //! extern crate linux_embedded_hal as hal; //! extern crate ds323x; //! use ds323x::Ds323x; //! //! # fn main() { //! let dev = hal::I2cdev::new("/dev/i2c-1").unwrap(); //! let rtc = Ds323x::new_ds3231(dev); //! // do something... //! //! // get the I2C device back //! let dev = rtc.destroy_ds3231(); //! # } //! ``` //! //! ### Create a driver instance for the DS3232 //! //! ```no_run //! extern crate linux_embedded_hal as hal; //! extern crate ds323x; //! use ds323x::Ds323x; //! //! # fn main() { //! let dev = hal::I2cdev::new("/dev/i2c-1").unwrap(); //! let rtc = Ds323x::new_ds3232(dev); //! // do something... //! //! // get the I2C device back //! let dev = rtc.destroy_ds3232(); //! # } //! ``` //! //! ### Create a driver instance for the DS3234 //! //! ```no_run //! extern crate linux_embedded_hal as hal; //! extern crate ds323x; //! use ds323x::Ds323x; //! //! # fn main() { //! let dev = hal::Spidev::open("/dev/spidev0.0").unwrap(); //! let chip_select = hal::Pin::new(24); //! let rtc = Ds323x::new_ds3234(dev, chip_select); //! // do something... //! //! // get the SPI device and chip select pin back //! let (dev, chip_select) = rtc.destroy_ds3234(); //! # } //! ``` #![deny(unsafe_code)] #![deny(missing_docs)] #![deny(warnings)] #![no_std] extern crate embedded_hal as hal; use hal::blocking; use core::marker::PhantomData; /// All possible errors in this crate #[derive(Debug)] pub enum Error { /// I²C/SPI bus error Comm(E), /// Invalid input data provided InvalidInputData } struct Register; impl Register { const SECONDS : u8 = 0x00; const MINUTES : u8 = 0x01; } const DEVICE_ADDRESS: u8 = 0b110_1000; /// IC markers pub mod ic { /// DS3231 IC marker pub struct DS3231; /// DS3232 IC marker pub struct DS3232; /// DS3234 IC marker pub struct DS3234; } pub mod interface; use interface::{ I2cInterface, SpiInterface }; /// DS3231, DS3232 and DS3234 RTC driver #[derive(Debug, Default)] pub struct Ds323x { iface: DI, _ic: PhantomData } impl Ds323x, ic::DS3231> where I2C: blocking::i2c::Write + blocking::i2c::WriteRead { /// Create a new instance of the DS3231 device. pub fn new_ds3231(i2c: I2C) -> Self { Ds323x { iface: I2cInterface { i2c, }, _ic: PhantomData } } /// Destroy driver instance, return I²C bus instance. pub fn destroy_ds3231(self) -> I2C { self.iface.i2c } } impl Ds323x, ic::DS3232> where I2C: blocking::i2c::Write + blocking::i2c::WriteRead { /// Create a new instance of the DS3232 device. pub fn new_ds3232(i2c: I2C) -> Self { Ds323x { iface: I2cInterface { i2c, }, _ic: PhantomData } } /// Destroy driver instance, return I²C bus instance. pub fn destroy_ds3232(self) -> I2C { self.iface.i2c } } impl Ds323x, ic::DS3234> where SPI: blocking::spi::Transfer + blocking::spi::Write, CS: hal::digital::OutputPin { /// Create a new instance. pub fn new_ds3234(spi: SPI, chip_select: CS) -> Self { Ds323x { iface: SpiInterface { spi, cs: chip_select }, _ic: PhantomData } } /// Destroy driver instance, return SPI bus instance and CS output pin. pub fn destroy_ds3234(self) -> (SPI, CS) { (self.iface.spi, self.iface.cs) } } mod ds323x;