/* * MIT License * * Copyright (c) 2020-2024 Joey Castillo * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include "pins.h" #include "watch_slcd.h" #include "watch_common_display.h" #include "slcd.h" ////////////////////////////////////////////////////////////////////////////////////////// // Segmented Display static uint16_t _slcd_framerate = 0; static uint16_t _slcd_fc_min_ms_bypass = 0; void watch_enable_display(void) { HAL_GPIO_SLCD0_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD1_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD2_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD3_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD4_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD5_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD6_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD7_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD8_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD9_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD10_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD11_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD12_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD13_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD14_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD15_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD16_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD17_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD18_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD19_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD20_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD21_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD22_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD23_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD24_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD25_pmuxen(HAL_GPIO_PMUX_B); HAL_GPIO_SLCD26_pmuxen(HAL_GPIO_PMUX_B); #ifdef USE_CUSTOM_LCD // Original famous Casio LCD: 1/3 bias, 1/4 duty with a frame rate of 32 Hz slcd_init(LCD_PIN_ENABLE, SLCD_BIAS_THIRD, SLCD_DUTY_4_COMMON, SLCD_CLOCKSOURCE_XOSC, SLCD_PRESCALER_DIV64, SLCD_CLOCKDIV_4); // exact frame rate is: 32768 / (4 * 64 * 4) ≈ 32 Hz _slcd_framerate = 32; #else // Original famous Casio LCD: 1/3 bias, 1/3 duty with a frame rate of ~34 Hz slcd_init(LCD_PIN_ENABLE, SLCD_BIAS_THIRD, SLCD_DUTY_3_COMMON, SLCD_CLOCKSOURCE_XOSC, SLCD_PRESCALER_DIV64, SLCD_CLOCKDIV_5); // exact frame rate is: 32768 / (3 * 64 * 5) ≈ 34.13 Hz _slcd_framerate = 34; #endif // calculate the smallest duration we can time before we have to engage the frame counter prescaler bypass _slcd_fc_min_ms_bypass = 32 * (1000 / _slcd_framerate); slcd_clear(); #ifdef USE_CUSTOM_LCD slcd_set_contrast(6); #else slcd_set_contrast(9); #endif slcd_enable(); } inline void watch_set_pixel(uint8_t com, uint8_t seg) { slcd_set_segment(com, seg); } inline void watch_clear_pixel(uint8_t com, uint8_t seg) { slcd_clear_segment(com, seg); } void watch_clear_display(void) { slcd_clear(); } void watch_start_character_blink(char character, uint32_t duration) { slcd_set_frame_counter_enabled(0, false); if (duration <= _slcd_fc_min_ms_bypass) { slcd_configure_frame_counter(0, (duration / (1000 / _slcd_framerate)) - 1, false); } else { slcd_configure_frame_counter(0, ((duration / (1000 / _slcd_framerate)) / 8 - 1), true); } slcd_set_frame_counter_enabled(0, true); watch_display_character(character, 7); watch_clear_pixel(2, 10); // clear segment B of position 7 since it can't blink slcd_disable(); slcd_set_blink_enabled(false); slcd_configure_blink(false, 0x0F, 0x0F, 0); slcd_set_blink_enabled(true); slcd_enable(); } void watch_start_indicator_blink_if_possible(watch_indicator_t indicator, uint32_t duration) { #ifdef USE_CUSTOM_LCD uint8_t mask = 0; switch (indicator) { case WATCH_INDICATOR_COLON: mask = 0b0001; break; case WATCH_INDICATOR_LAP: mask = 0b0010; break; case WATCH_INDICATOR_BATTERY: mask = 0b0100; break; case WATCH_INDICATOR_SLEEP: mask = 0b1000; break; default: return; } watch_set_indicator(indicator); if (duration <= _slcd_fc_min_ms_bypass) { slcd_configure_frame_counter(0, (duration / (1000 / _slcd_framerate)) - 1, false); } else { slcd_configure_frame_counter(0, ((duration / (1000 / _slcd_framerate)) / 8 - 1), true); } slcd_set_frame_counter_enabled(0, true); slcd_disable(); slcd_set_blink_enabled(false); slcd_configure_blink(false, mask, 0, 0); slcd_set_blink_enabled(true); slcd_enable(); #else (void) indicator; (void) duration; #endif } void watch_stop_blink(void) { slcd_set_frame_counter_enabled(0, false); slcd_set_blink_enabled(false); } void watch_start_sleep_animation(uint32_t duration) { #ifdef USE_CUSTOM_LCD (void) duration; watch_set_indicator(WATCH_INDICATOR_SLEEP); #else watch_display_character(' ', 8); watch_display_character(' ', 9); slcd_disable(); slcd_set_frame_counter_enabled(1, false); slcd_set_circular_shift_animation_enabled(false); if (duration <= _slcd_fc_min_ms_bypass) { slcd_configure_frame_counter(1, (duration / (1000 / _slcd_framerate)) - 1, false); } else { slcd_configure_frame_counter(1, ((duration / (1000 / _slcd_framerate)) / 8 - 1), true); } slcd_set_frame_counter_enabled(1, true); slcd_configure_circular_shift_animation(0b00000001, 1, SLCD_CSRSHIFT_LEFT, 1); slcd_set_circular_shift_animation_enabled(true); slcd_enable(); #endif } bool watch_sleep_animation_is_running(void) { // TODO: wrap this in gossamer call return SLCD->CTRLD.bit.CSREN; } void watch_stop_sleep_animation(void) { #ifdef USE_CUSTOM_LCD watch_clear_indicator(WATCH_INDICATOR_SLEEP); #else slcd_set_circular_shift_animation_enabled(false); watch_display_character(' ', 8); #endif }