/* Copyright 2023 Dual Tachyon * https://github.com/DualTachyon * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <string.h> #include "app/dtmf.h" #if defined(ENABLE_FMRADIO) #include "app/fm.h" #endif #include "audio.h" #include "bsp/dp32g030/gpio.h" #include "dcs.h" #include "driver/backlight.h" #if defined(ENABLE_FMRADIO) #include "driver/bk1080.h" #endif #include "driver/bk4819.h" #include "driver/gpio.h" #include "driver/system.h" #include "driver/st7565.h" #include "frequencies.h" #include "functions.h" #include "helper/battery.h" #include "misc.h" #include "radio.h" #include "settings.h" #include "ui/status.h" #include "ui/ui.h" FUNCTION_Type_t gCurrentFunction; void FUNCTION_Init(void) { #ifdef ENABLE_NOAA if (!IS_NOAA_CHANNEL(gRxVfo->CHANNEL_SAVE)) #endif { gCurrentCodeType = (gRxVfo->Modulation != MODULATION_FM) ? CODE_TYPE_OFF : gRxVfo->pRX->CodeType; } #ifdef ENABLE_NOAA else gCurrentCodeType = CODE_TYPE_CONTINUOUS_TONE; #endif #ifdef ENABLE_DTMF_CALLING DTMF_clear_RX(); #endif g_CxCSS_TAIL_Found = false; g_CDCSS_Lost = false; g_CTCSS_Lost = false; #ifdef ENABLE_VOX g_VOX_Lost = false; #endif g_SquelchLost = false; gFlagTailNoteEliminationComplete = false; gTailNoteEliminationCountdown_10ms = 0; gFoundCTCSS = false; gFoundCDCSS = false; gFoundCTCSSCountdown_10ms = 0; gFoundCDCSSCountdown_10ms = 0; gEndOfRxDetectedMaybe = false; #ifdef ENABLE_NOAA gNOAACountdown_10ms = 0; #endif gUpdateStatus = true; } void FUNCTION_Select(FUNCTION_Type_t Function) { const FUNCTION_Type_t PreviousFunction = gCurrentFunction; const bool bWasPowerSave = (PreviousFunction == FUNCTION_POWER_SAVE); gCurrentFunction = Function; if (bWasPowerSave && Function != FUNCTION_POWER_SAVE) { BK4819_Conditional_RX_TurnOn_and_GPIO6_Enable(); gRxIdleMode = false; UI_DisplayStatus(); } switch (Function) { case FUNCTION_FOREGROUND: #ifdef ENABLE_DTMF_CALLING if (gDTMF_ReplyState != DTMF_REPLY_NONE) RADIO_PrepareCssTX(); #endif if (PreviousFunction == FUNCTION_TRANSMIT) { ST7565_FixInterfGlitch(); gVFO_RSSI_bar_level[0] = 0; gVFO_RSSI_bar_level[1] = 0; } else if (PreviousFunction != FUNCTION_RECEIVE) break; #if defined(ENABLE_FMRADIO) if (gFmRadioMode) gFM_RestoreCountdown_10ms = fm_restore_countdown_10ms; #endif #ifdef ENABLE_DTMF_CALLING if (gDTMF_CallState == DTMF_CALL_STATE_CALL_OUT || gDTMF_CallState == DTMF_CALL_STATE_RECEIVED || gDTMF_CallState == DTMF_CALL_STATE_RECEIVED_STAY) { gDTMF_auto_reset_time_500ms = gEeprom.DTMF_auto_reset_time * 2; } #endif gUpdateStatus = true; return; case FUNCTION_MONITOR: gMonitor = true; break; case FUNCTION_INCOMING: case FUNCTION_RECEIVE: break; case FUNCTION_POWER_SAVE: gPowerSave_10ms = gEeprom.BATTERY_SAVE * 10; gPowerSaveCountdownExpired = false; gRxIdleMode = true; gMonitor = false; BK4819_DisableVox(); BK4819_Sleep(); BK4819_ToggleGpioOut(BK4819_GPIO0_PIN28_RX_ENABLE, false); gUpdateStatus = true; if (gScreenToDisplay != DISPLAY_MENU) // 1of11 .. don't close the menu GUI_SelectNextDisplay(DISPLAY_MAIN); return; case FUNCTION_TRANSMIT: // if DTMF is enabled when TX'ing, it changes the TX audio filtering !! .. 1of11 BK4819_DisableDTMF(); #ifdef ENABLE_DTMF_CALLING // clear the DTMF RX buffer DTMF_clear_RX(); #endif // clear the DTMF RX live decoder buffer gDTMF_RX_live_timeout = 0; gDTMF_RX_live_timeout = 0; memset(gDTMF_RX_live, 0, sizeof(gDTMF_RX_live)); #if defined(ENABLE_FMRADIO) if (gFmRadioMode) BK1080_Init(0, false); #endif #ifdef ENABLE_ALARM if (gAlarmState == ALARM_STATE_TXALARM && gEeprom.ALARM_MODE != ALARM_MODE_TONE) { gAlarmState = ALARM_STATE_ALARM; GUI_DisplayScreen(); AUDIO_AudioPathOff(); SYSTEM_DelayMs(20); BK4819_PlayTone(500, 0); SYSTEM_DelayMs(2); AUDIO_AudioPathOn(); gEnableSpeaker = true; SYSTEM_DelayMs(60); BK4819_ExitTxMute(); gAlarmToneCounter = 0; break; } #endif gUpdateStatus = true; GUI_DisplayScreen(); RADIO_SetTxParameters(); // turn the RED LED on BK4819_ToggleGpioOut(BK4819_GPIO5_PIN1_RED, true); DTMF_Reply(); if (gCurrentVfo->DTMF_PTT_ID_TX_MODE == PTT_ID_APOLLO) BK4819_PlaySingleTone(2525, 250, 0, gEeprom.DTMF_SIDE_TONE); #if defined(ENABLE_ALARM) || defined(ENABLE_TX1750) if (gAlarmState != ALARM_STATE_OFF) { #ifdef ENABLE_TX1750 if (gAlarmState == ALARM_STATE_TX1750) BK4819_TransmitTone(true, 1750); #endif #ifdef ENABLE_ALARM if (gAlarmState == ALARM_STATE_TXALARM) BK4819_TransmitTone(true, 500); #endif SYSTEM_DelayMs(2); AUDIO_AudioPathOn(); #ifdef ENABLE_ALARM gAlarmToneCounter = 0; #endif gEnableSpeaker = true; break; } #endif if (gCurrentVfo->SCRAMBLING_TYPE > 0 && gSetting_ScrambleEnable) BK4819_EnableScramble(gCurrentVfo->SCRAMBLING_TYPE - 1); else BK4819_DisableScramble(); // if (gSetting_backlight_on_tx_rx == BACKLIGHT_ON_TR_TX || // gSetting_backlight_on_tx_rx == BACKLIGHT_ON_TR_TXRX) BACKLIGHT_TurnOn(); break; case FUNCTION_BAND_SCOPE: break; } gBatterySaveCountdown_10ms = battery_save_count_10ms; gSchedulePowerSave = false; #if defined(ENABLE_FMRADIO) gFM_RestoreCountdown_10ms = 0; #endif }