2023-08-11 03:37:33 +08:00
|
|
|
/* 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.
|
|
|
|
*/
|
|
|
|
|
2023-08-27 19:32:50 +08:00
|
|
|
#include <string.h>
|
2023-08-29 07:15:44 +08:00
|
|
|
#include "app/fm.h"
|
2023-08-29 07:37:06 +08:00
|
|
|
#include "audio.h"
|
2023-08-27 01:43:05 +08:00
|
|
|
#include "bsp/dp32g030/gpio.h"
|
|
|
|
#include "driver/bk1080.h"
|
2023-08-27 19:32:50 +08:00
|
|
|
#include "driver/eeprom.h"
|
2023-08-27 01:43:05 +08:00
|
|
|
#include "driver/gpio.h"
|
|
|
|
#include "misc.h"
|
2023-08-11 00:51:57 +08:00
|
|
|
#include "settings.h"
|
2023-08-29 07:37:06 +08:00
|
|
|
#include "ui/inputbox.h"
|
|
|
|
#include "ui/ui.h"
|
|
|
|
|
2023-08-29 19:46:53 +08:00
|
|
|
extern void APP_StartScan(bool bFlag);
|
2023-08-29 07:37:06 +08:00
|
|
|
extern void APP_SwitchToFM(void);
|
2023-08-11 00:51:57 +08:00
|
|
|
|
|
|
|
uint16_t gFM_Channels[20];
|
2023-08-28 08:07:07 +08:00
|
|
|
bool gFmRadioMode;
|
2023-08-11 00:51:57 +08:00
|
|
|
|
|
|
|
bool FM_CheckValidChannel(uint8_t Channel)
|
|
|
|
{
|
2023-08-30 05:55:07 +08:00
|
|
|
if (Channel < 20 && (gFM_Channels[Channel] >= 760 && gFM_Channels[Channel] < 1080)) {
|
2023-08-11 00:51:57 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t FM_FindNextChannel(uint8_t Channel, uint8_t Direction)
|
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < 20; i++) {
|
|
|
|
if (Channel == 0xFF) {
|
|
|
|
Channel = 19;
|
|
|
|
} else if (Channel >= 20) {
|
|
|
|
Channel = 0;
|
|
|
|
}
|
|
|
|
if (FM_CheckValidChannel(Channel)) {
|
|
|
|
return Channel;
|
|
|
|
}
|
|
|
|
Channel += Direction;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
int FM_ConfigureChannelState(void)
|
|
|
|
{
|
|
|
|
uint8_t Channel;
|
|
|
|
|
|
|
|
gEeprom.FM_FrequencyToPlay = gEeprom.FM_CurrentFrequency;
|
|
|
|
if (gEeprom.FM_IsChannelSelected) {
|
|
|
|
Channel = FM_FindNextChannel(gEeprom.FM_CurrentChannel, FM_CHANNEL_UP);
|
|
|
|
if (Channel == 0xFF) {
|
|
|
|
gEeprom.FM_IsChannelSelected = false;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
gEeprom.FM_CurrentChannel = Channel;
|
|
|
|
gEeprom.FM_FrequencyToPlay = gFM_Channels[Channel];
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-08-27 01:43:05 +08:00
|
|
|
void FM_TurnOff(void)
|
|
|
|
{
|
2023-08-28 08:07:07 +08:00
|
|
|
gFmRadioMode = false;
|
2023-08-29 07:37:06 +08:00
|
|
|
gFM_Step = 0;
|
2023-08-27 01:43:05 +08:00
|
|
|
g_2000038E = 0;
|
|
|
|
GPIO_ClearBit(&GPIOC->DATA, GPIOC_PIN_AUDIO_PATH);
|
|
|
|
g_2000036B = 0;
|
|
|
|
BK1080_Init(0, false);
|
2023-08-30 07:19:12 +08:00
|
|
|
gUpdateStatus = true;
|
2023-08-27 01:43:05 +08:00
|
|
|
}
|
|
|
|
|
2023-08-27 19:32:50 +08:00
|
|
|
void FM_EraseChannels(void)
|
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
uint8_t Template[8];
|
|
|
|
|
|
|
|
memset(Template, 0xFF, sizeof(Template));
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
|
|
EEPROM_WriteBuffer(0x0E40 + (i * 8), Template);
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(gFM_Channels, 0xFF, sizeof(gFM_Channels));
|
|
|
|
}
|
|
|
|
|
2023-08-29 07:37:06 +08:00
|
|
|
void FM_Tune(uint16_t Frequency, int8_t Step, bool bFlag)
|
|
|
|
{
|
|
|
|
GPIO_ClearBit(&GPIOC->DATA, GPIOC_PIN_AUDIO_PATH);
|
|
|
|
g_2000036B = 0;
|
|
|
|
if (gFM_Step == 0) {
|
2023-08-29 20:26:13 +08:00
|
|
|
gFmPlayCountdown = 120;
|
2023-08-29 07:37:06 +08:00
|
|
|
} else {
|
2023-08-29 20:26:13 +08:00
|
|
|
gFmPlayCountdown = 10;
|
2023-08-29 07:37:06 +08:00
|
|
|
}
|
2023-08-29 20:26:13 +08:00
|
|
|
gScheduleFM = false;
|
2023-08-29 07:37:06 +08:00
|
|
|
g_20000427 = 0;
|
|
|
|
gAskToSave = false;
|
|
|
|
gAskToDelete = false;
|
|
|
|
gEeprom.FM_FrequencyToPlay = Frequency;
|
|
|
|
if (!bFlag) {
|
|
|
|
Frequency += Step;
|
|
|
|
if (Frequency < gEeprom.FM_LowerLimit) {
|
|
|
|
Frequency = gEeprom.FM_UpperLimit;
|
|
|
|
} else if (Frequency > gEeprom.FM_UpperLimit) {
|
|
|
|
Frequency = gEeprom.FM_LowerLimit;
|
|
|
|
}
|
|
|
|
gEeprom.FM_FrequencyToPlay = Frequency;
|
|
|
|
}
|
|
|
|
|
|
|
|
gFM_Step = Step;
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FM_Play(void)
|
|
|
|
{
|
|
|
|
gFM_Step = 0;
|
2023-08-30 05:55:07 +08:00
|
|
|
if (gFM_AutoScan) {
|
2023-08-29 07:37:06 +08:00
|
|
|
gEeprom.FM_IsChannelSelected = true;
|
|
|
|
gEeprom.FM_CurrentChannel = 0;
|
|
|
|
}
|
|
|
|
FM_ConfigureChannelState();
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
SETTINGS_SaveFM();
|
2023-08-29 20:26:13 +08:00
|
|
|
gFmPlayCountdown = 0;
|
|
|
|
gScheduleFM = false;
|
2023-08-29 07:37:06 +08:00
|
|
|
gAskToSave = false;
|
|
|
|
GPIO_SetBit(&GPIOC->DATA, GPIOC_PIN_AUDIO_PATH);
|
|
|
|
g_2000036B = 1;
|
|
|
|
}
|
|
|
|
|
2023-08-30 06:00:24 +08:00
|
|
|
int FM_CheckFrequencyLock(uint16_t Frequency, uint16_t LowerLimit)
|
|
|
|
{
|
|
|
|
uint16_t SNR;
|
|
|
|
int16_t Deviation;
|
|
|
|
uint16_t RSSI;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
SNR = BK1080_ReadRegister(BK1080_REG_07);
|
|
|
|
// This cast fails to extend the sign because ReadReg is guaranteed to be U16.
|
|
|
|
Deviation = (int16_t)SNR >> 4;
|
|
|
|
if ((SNR & 0xF) < 2) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
|
|
|
|
RSSI = BK1080_ReadRegister(BK1080_REG_10);
|
|
|
|
if (RSSI & 0x1000 || (RSSI & 0xFF) < 10) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Deviation < 280 || Deviation > 3815) {
|
|
|
|
if ((LowerLimit < Frequency) && (Frequency - g_20000362) == 1) {
|
|
|
|
if (gFM_FrequencyDeviation & 0x800) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
if (gFM_FrequencyDeviation < 20) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((LowerLimit <= Frequency) && (g_20000362 - Frequency) == 1) {
|
|
|
|
if ((gFM_FrequencyDeviation & 0x800) == 0) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
if (4075 < gFM_FrequencyDeviation) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Bail:
|
|
|
|
gFM_FrequencyDeviation = (uint16_t)Deviation;
|
|
|
|
g_20000362 = Frequency;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-08-29 19:46:53 +08:00
|
|
|
void FM_Key_DIGITS(KEY_Code_t Key, bool bKeyPressed, bool bKeyHeld)
|
|
|
|
{
|
|
|
|
if (!bKeyHeld && bKeyPressed) {
|
|
|
|
if (!gWasFKeyPressed) {
|
|
|
|
uint8_t Value;
|
|
|
|
|
|
|
|
if (gAskToDelete) {
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (gAskToSave) {
|
|
|
|
Value = 2;
|
|
|
|
} else {
|
|
|
|
if (gFM_Step) {
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (gEeprom.FM_IsChannelSelected) {
|
|
|
|
Value = 1;
|
|
|
|
} else {
|
|
|
|
Value = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
INPUTBOX_Append(Key);
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
if (Value == 0) {
|
|
|
|
if (gInputBoxIndex == 1) {
|
|
|
|
if (1 < gInputBox[0]) {
|
|
|
|
gInputBox[1] = gInputBox[0];
|
|
|
|
gInputBox[0] = 0;
|
|
|
|
gInputBoxIndex = 2;
|
|
|
|
}
|
|
|
|
} else if (3 < gInputBoxIndex) {
|
|
|
|
uint32_t Frequency;
|
|
|
|
|
|
|
|
gInputBoxIndex = 0;
|
|
|
|
NUMBER_Get(gInputBox, &Frequency);
|
|
|
|
Frequency = Frequency / 10000;
|
|
|
|
if (Frequency < gEeprom.FM_LowerLimit || gEeprom.FM_UpperLimit < Frequency) {
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
return;
|
|
|
|
}
|
2023-08-30 02:17:01 +08:00
|
|
|
gEeprom.FM_CurrentFrequency = (uint16_t)Frequency;
|
2023-08-29 19:46:53 +08:00
|
|
|
gAnotherVoiceID = (VOICE_ID_t)Key;
|
|
|
|
gEeprom.FM_FrequencyToPlay = gEeprom.FM_CurrentFrequency;
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
gRequestSaveFM = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (gInputBoxIndex == 2) {
|
|
|
|
uint8_t Channel;
|
|
|
|
|
|
|
|
gInputBoxIndex = 0;
|
|
|
|
Channel = ((gInputBox[0] * 10) + gInputBox[1]) - 1;
|
|
|
|
if (Value == 1) {
|
|
|
|
if (FM_CheckValidChannel(Channel)) {
|
|
|
|
gAnotherVoiceID = (VOICE_ID_t)Key;
|
|
|
|
gEeprom.FM_CurrentChannel = Channel;
|
|
|
|
gEeprom.FM_FrequencyToPlay = gFM_Channels[Channel];
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
gRequestSaveFM = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (Channel < 20) {
|
|
|
|
gAnotherVoiceID = (VOICE_ID_t)Key;
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
gInputBoxIndex = 0;
|
2023-08-30 05:55:07 +08:00
|
|
|
gFM_ScanFoundIndex = Channel;
|
2023-08-29 19:46:53 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gAnotherVoiceID = (VOICE_ID_t)Key;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gBeepToPlay = BEEP_1KHZ_60MS_OPTIONAL;
|
|
|
|
gWasFKeyPressed = false;
|
2023-08-30 07:19:12 +08:00
|
|
|
gUpdateStatus = true;
|
2023-08-29 19:46:53 +08:00
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
switch (Key) {
|
|
|
|
case KEY_0:
|
|
|
|
APP_SwitchToFM();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_1:
|
|
|
|
gEeprom.FM_IsChannelSelected = !gEeprom.FM_IsChannelSelected;
|
|
|
|
if (!FM_ConfigureChannelState()) {
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
gRequestSaveFM = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_2:
|
|
|
|
APP_StartScan(true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_3:
|
|
|
|
APP_StartScan(false);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-29 07:37:06 +08:00
|
|
|
void FM_Key_EXIT(bool bKeyPressed, bool bKeyHeld)
|
|
|
|
{
|
|
|
|
if (bKeyHeld) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!bKeyPressed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gBeepToPlay = BEEP_1KHZ_60MS_OPTIONAL;
|
|
|
|
if (gFM_Step == 0) {
|
|
|
|
if (gInputBoxIndex == 0) {
|
|
|
|
if (!gAskToSave && !gAskToDelete) {
|
|
|
|
APP_SwitchToFM();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gAskToSave = false;
|
|
|
|
gAskToDelete = false;
|
|
|
|
} else {
|
|
|
|
gInputBoxIndex--;
|
|
|
|
gInputBox[gInputBoxIndex] = 10;
|
|
|
|
if (gInputBoxIndex) {
|
|
|
|
if (gInputBoxIndex != 1) {
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (gInputBox[0] != 0) {
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
gInputBoxIndex = 0;
|
|
|
|
}
|
|
|
|
gAnotherVoiceID = VOICE_ID_CANCEL;
|
|
|
|
} else {
|
|
|
|
FM_Play();
|
|
|
|
gAnotherVoiceID = VOICE_ID_SCANNING_STOP;
|
|
|
|
}
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FM_Key_UP_DOWN(bool bKeyPressed, bool bKeyHeld, int8_t Step)
|
|
|
|
{
|
|
|
|
if (bKeyHeld || !bKeyPressed) {
|
|
|
|
if (gInputBoxIndex) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!bKeyPressed) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (gInputBoxIndex) {
|
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gBeepToPlay = BEEP_1KHZ_60MS_OPTIONAL;
|
|
|
|
}
|
|
|
|
if (gAskToSave) {
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
2023-08-30 05:55:07 +08:00
|
|
|
gFM_ScanFoundIndex = NUMBER_AddWithWraparound(gFM_ScanFoundIndex, Step, 0, 19);
|
2023-08-29 07:37:06 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (gFM_Step) {
|
2023-08-30 05:55:07 +08:00
|
|
|
if (gFM_AutoScan) {
|
2023-08-29 07:37:06 +08:00
|
|
|
gBeepToPlay = BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
FM_Tune(gEeprom.FM_FrequencyToPlay, Step, false);
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (gEeprom.FM_IsChannelSelected) {
|
|
|
|
uint8_t Channel;
|
|
|
|
|
|
|
|
Channel = FM_FindNextChannel(gEeprom.FM_CurrentChannel + Step, Step);
|
|
|
|
if (Channel == 0xFF || gEeprom.FM_CurrentChannel == Channel) {
|
|
|
|
goto Bail;
|
|
|
|
}
|
|
|
|
gEeprom.FM_CurrentChannel = Channel;
|
|
|
|
gEeprom.FM_FrequencyToPlay = gFM_Channels[Channel];
|
|
|
|
} else {
|
|
|
|
uint16_t Frequency;
|
|
|
|
|
|
|
|
Frequency = gEeprom.FM_CurrentFrequency + Step;
|
|
|
|
if (Frequency < gEeprom.FM_LowerLimit) {
|
|
|
|
Frequency = gEeprom.FM_UpperLimit;
|
|
|
|
} else if (Frequency > gEeprom.FM_UpperLimit) {
|
|
|
|
Frequency = gEeprom.FM_LowerLimit;
|
|
|
|
}
|
|
|
|
gEeprom.FM_FrequencyToPlay = Frequency;
|
|
|
|
gEeprom.FM_CurrentFrequency = gEeprom.FM_FrequencyToPlay;
|
|
|
|
}
|
|
|
|
gRequestSaveFM = true;
|
|
|
|
|
|
|
|
Bail:
|
|
|
|
BK1080_SetFrequency(gEeprom.FM_FrequencyToPlay);
|
|
|
|
gRequestDisplayScreen = DISPLAY_FM;
|
|
|
|
}
|
|
|
|
|