diff --git a/firmware/main.c b/firmware/main.c index 18a5dce46..86ec80640 100644 --- a/firmware/main.c +++ b/firmware/main.c @@ -54,6 +54,9 @@ #include "mag3110.h" #endif #include "ms5611.h" +#if defined(I2C0_RV3129) +#include "rv3129.h" +#endif #if defined(I2C1_ADS122) #include "ads122.h" #elif defined(I2C1_ELVR) @@ -301,6 +304,9 @@ void main() { mag3110_init(); #endif ms5611_init(); +#if defined(I2C0_RV3129) + rv3129_init(); +#endif #if defined(I2C1_ADS122) ads122_init(); #elif defined(I2C1_ELVR) @@ -333,6 +339,9 @@ void main() { mag3110_polling(); #endif ms5611_polling(); +#if defined(I2C0_RV3129) + rv3129_polling(); +#endif #if defined(I2C1_ADS122) ads122_polling(); #elif defined(I2C1_ELVR) diff --git a/firmware/rv3129.c b/firmware/rv3129.c new file mode 100644 index 000000000..8ee01cbc9 --- /dev/null +++ b/firmware/rv3129.c @@ -0,0 +1,215 @@ +/* + * Copyright (c) 2024, M.Naruoka (fenrir) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without modification, + * are permitted provided that the following conditions are met: + * + * - Redistributions of source code must retain the above copyright notice, + * this list of conditions and the following disclaimer. + * - Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * - Neither the name of the naruoka.org nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, + * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, + * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#include "c8051f380.h" + +#include "rv3129.h" +#include "main.h" +#include "gps.h" + +#include "f38x_i2c.h" + +#define I2C_ADDRESS_RW (0x56 << 1) + +typedef enum { + // Control page + ADDR_Control_1 = 0, + ADDR_Control_INT, + ADDR_Control_INT_Flag, + ADDR_Control_Status, + ADDR_Control_Reset, + + // Clock page + ADDR_Seconds = 0x08, + ADDR_Minutes, + ADDR_Hours, + ADDR_Days, + ADDR_Weekdays, + ADDR_Months, + ADDR_Years, + + // Alarm page + ADDR_Second_Alarm = 0x10, + ADDR_Minute_Alarm, + ADDR_Hour_Alarm, + ADDR_Days_Alarm, + ADDR_Weekday_Alarm, + ADDR_Months_Alarm, + ADDR_Year_Alarm, + + // Timer page + ADDR_Timer_Low = 0x18, + ADDR_Timer_High, + + // Temperature page + ADDR_Temperature = 0x20, + + // EEPROM User + ADDR_EEPROM_User0 = 0x28, + ADDR_EEPROM_User1, + + // EEPROM Control page + ADDR_EEPROM_Contr = 0x30, + ADDR_Xtal_Offset, + ADDR_Xtal_Coef, + ADDR_Xtal_T0, + + // RAM page + //User RAM = 0x38, +} address_t; + +static void read_data(u8 addr, u8 *buf, u8 buf_size){ + i2c0_read_write(I2C_WRITE(I2C_ADDRESS_RW), &addr, 1); + i2c0_read_write(I2C_READ(I2C_ADDRESS_RW), buf, buf_size); +} + +static void update_register(u8 addr, u8 v, u8 v_mask){ + u8 buf[2] = {addr}; + read_data(addr, &buf[1], 1); + buf[1] &= ~v_mask; + buf[1] |= (v & v_mask); + i2c0_read_write(I2C_WRITE(I2C_ADDRESS_RW), buf, 2); +} + +static void start_watch(){ + update_register(ADDR_Control_1, 0x01, 0x01); // WE = 1 +} +static void stop_watch(){ + update_register(ADDR_Control_1, 0x00, 0x01); // WE = 0 +} +static void start_timer(){ + update_register(ADDR_Control_1, 0x02, 0x02); // TE = 1 +} +static void stop_timer(){ + update_register(ADDR_Control_1, 0x00, 0x02); // TE = 0 +} + +#define BCD2NUM(bcd) ((((bcd & 0xF0) >> 4) * 10) + (bcd & 0x0F)) +#define NUM2BCD(num) ((((num) / 10) << 4) + (num % 10)) + +static void get_watch(struct tm *t){ + u8 buf[7]; + read_data(ADDR_Seconds, buf, sizeof(buf)); + i2c0_read_write(I2C_READ(I2C_ADDRESS_RW), buf, sizeof(buf)); + t->tm_sec = BCD2NUM(buf[0]); // Seconds [0-60] + t->tm_min = BCD2NUM(buf[1]); // Minutes [0-59] + t->tm_hour = BCD2NUM(buf[2]); // Hours [0-23] + t->tm_mday = BCD2NUM(buf[3]); // Day [1-31] + t->tm_mon = BCD2NUM(buf[5]) - 1; // Month [1-12] -> [0-11] + t->tm_year = BCD2NUM(buf[6]) + 100; // Year since 2000 -> 1900 +} + +static void set_watch(struct tm *t){ + u8 mon = t->tm_mon + 1; // Month [0-11] -> [1-12] + u8 year = t->tm_year - 100; // Year since 1900 -> 2000 + { // Recalculate weekday + // @see https://en.wikipedia.org/wiki/Determination_of_the_day_of_the_week#Sakamoto's_methods + // modified base date from 1753/1/1(1=Monday) -> 1980/1/1(2=Tuesday) + // valid from 1980/1/1 to 2099/12/31 + u8 y = t->tm_year - 76; // y > 1980 + static const u8 tbl[] = {5, 1, 0, 3, 5, 1, 3, 6, 2, 4, 0, 2}; + if(t->tm_mon < 2){y -= 1;} + t->tm_wday = ((y + y/4 + tbl[t->tm_mon] + (u8)t->tm_mday) % 7); + } + { + u8 buf[] = { + ADDR_Seconds, + NUM2BCD(t->tm_sec), // Seconds [0-60] + NUM2BCD(t->tm_min), // Minutes [0-59] + NUM2BCD(t->tm_hour), // Hours [0-23] + NUM2BCD(t->tm_mday), // Day [1-31] + NUM2BCD(t->tm_wday), // weekday [0:Sunday-6] + NUM2BCD(mon), + NUM2BCD(year), + }; + i2c0_read_write(I2C_WRITE(I2C_ADDRESS_RW), buf, sizeof(buf)); + } +} + +static void get_watch2(u32 *tow_ms){ + struct tm t; + get_watch(&t); + *tow_ms = t.tm_wday; // weekday [0:Sunday-6] + (*tow_ms) *= 7; + (*tow_ms) += t.tm_hour; // Hours [0-23] + (*tow_ms) *= 24; + (*tow_ms) += t.tm_min; // Minutes [0-59] + (*tow_ms) *= 60; + (*tow_ms) += t.tm_sec; // Seconds [0-60] + (*tow_ms) *= 1000; +} + +static void set_watch2(u32 tow_ms){ + struct tm t = {0}; + tow_ms = (tow_ms / 1000) + 1; + t.tm_sec = tow_ms % 60; // Seconds [0-60] + t.tm_min = (tow_ms /= 60) % 60; // Minutes [0-59] + t.tm_hour = (tow_ms /= 60) % 24;// Hours [0-23] + t.tm_wday = tow_ms / 24; // weekday [0:Sunday-6] + t.tm_mday = 9 + t.tm_wday; // Day [1-31] + t.tm_mon = 0; // 2000/1/9 = Sunday + t.tm_year = 100; + set_watch(&t); +} + +static u16 get_timer(){ + WORD_t v; + read_data(ADDR_Timer_Low, v.c, sizeof(v)); + return v.i; +} + +static void set_timer(u16 v){ + u8 buf[] = { + ADDR_Timer_Low, (u8)(v & 0xFF), (u8)(v >> 8) + }; + i2c0_read_write(I2C_WRITE(I2C_ADDRESS_RW), buf, sizeof(buf)); +} + +__bit watch_started = FALSE; + +void rv3129_init(){ + u8 buf; + read_data(ADDR_Control_1, &buf, 1); + if(buf & 0x01){ // watch started + watch_started = TRUE; + get_watch2(&(gps_time.itow_ms)); + gps_time_modified = TRUE; + IE0 = 1; // invoke -INT0 + } +} + +void rv3129_polling(){ + if(watch_started){return;} + if(!gps_utc_valid){return;} + set_watch(&gps_utc); + start_watch(); + watch_started = TRUE; +} diff --git a/firmware/rv3129.h b/firmware/rv3129.h new file mode 100644 index 000000000..e622f6a80 --- /dev/null +++ b/firmware/rv3129.h @@ -0,0 +1,41 @@ +/* + * Copyright (c) 2024, M.Naruoka (fenrir) + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without modification, + * are permitted provided that the following conditions are met: + * + * - Redistributions of source code must retain the above copyright notice, + * this list of conditions and the following disclaimer. + * - Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * - Neither the name of the naruoka.org nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, + * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS + * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, + * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, + * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef __RV3129_H__ +#define __RV3129_H__ + +#include +#include "type.h" + +void rv3129_init(); +void rv3129_polling(); + +#endif /* __RV3129_H__ */ diff --git a/tool/test/misc_spec.rb b/tool/test/misc_spec.rb new file mode 100644 index 000000000..051bda0ed --- /dev/null +++ b/tool/test/misc_spec.rb @@ -0,0 +1,27 @@ +require 'rspec' +require 'inline' +require 'date' + +module Miscellaneous + inline{|builder| + #builder.prefix("") + builder.c_singleton <<-__C_CODE__ + unsigned int dow(unsigned int y1900, unsigned int m1, unsigned int d){ + unsigned int y = y1900 - 76; // y > 1980 + static const unsigned int tbl[] = {5, 1, 0, 3, 5, 1, 3, 6, 2, 4, 0, 2}; + if(m1 < 2){y -= 1;} + return ((y + y/4 + tbl[m1] + d) % 7); + } + __C_CODE__ + } +end + +describe Miscellaneous do + let "dow" do + (1980..2099).each{|y| + (1..12).each{|m| + expect(Miscellaneous::dow(y - 1900, m - 1, 1)).to equal(Date::new(y, m, 1).wday) + } + } + end +end