audio.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471
  1. /* Copyright 2023 Dual Tachyon
  2. * https://github.com/DualTachyon
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #ifdef ENABLE_FMRADIO
  17. #include "app/fm.h"
  18. #endif
  19. #include "audio.h"
  20. #include "bsp/dp32g030/gpio.h"
  21. #ifdef ENABLE_FMRADIO
  22. #include "driver/bk1080.h"
  23. #endif
  24. #include "driver/bk4819.h"
  25. #include "driver/gpio.h"
  26. #include "driver/system.h"
  27. #include "driver/systick.h"
  28. #include "functions.h"
  29. #include "misc.h"
  30. #include "settings.h"
  31. #include "ui/ui.h"
  32. #include "utils/music.h"
  33. BEEP_Type_t gBeepToPlay = BEEP_NONE;
  34. void AUDIO_meep(void)
  35. {
  36. AUDIO_AudioPathOff();
  37. SYSTEM_DelayMs(20);
  38. uint16_t ToneConfig = BK4819_ReadRegister(BK4819_REG_71);
  39. AUDIO_AudioPathOn();
  40. // for (uint16_t ToneFrequency = 220; ToneFrequency < 1500; ToneFrequency += 100)
  41. // BK4819_PlaySingleTone(ToneFrequency, 60, 127, true);
  42. for (const uint8_t *musicp = musicdata; *musicp != 255; musicp += 4)
  43. {
  44. int freq = musicp[1] + (musicp[0] << 8); /* in Hz */
  45. int dur = musicp[3] + (musicp[2] << 8); /* in 1/100s */
  46. dur *= 5;
  47. BK4819_PlaySingleTone(freq, dur, 64, true);
  48. }
  49. AUDIO_AudioPathOff();
  50. SYSTEM_DelayMs(5);
  51. BK4819_TurnsOffTones_TurnsOnRX();
  52. SYSTEM_DelayMs(5);
  53. BK4819_WriteRegister(BK4819_REG_71, ToneConfig);
  54. }
  55. void AUDIO_PlayBeep(BEEP_Type_t Beep)
  56. {
  57. if (Beep != BEEP_880HZ_60MS_TRIPLE_BEEP &&
  58. Beep != BEEP_500HZ_60MS_DOUBLE_BEEP &&
  59. Beep != BEEP_440HZ_500MS &&
  60. Beep != BEEP_880HZ_200MS &&
  61. Beep != BEEP_880HZ_500MS &&
  62. !gEeprom.BEEP_CONTROL)
  63. return;
  64. #ifdef ENABLE_AIRCOPY
  65. if (gScreenToDisplay == DISPLAY_AIRCOPY)
  66. return;
  67. #endif
  68. if (gCurrentFunction == FUNCTION_RECEIVE)
  69. return;
  70. if (gCurrentFunction == FUNCTION_MONITOR)
  71. return;
  72. #ifdef ENABLE_FMRADIO
  73. if (gFmRadioMode)
  74. BK1080_Mute(true);
  75. #endif
  76. AUDIO_AudioPathOff();
  77. if (gCurrentFunction == FUNCTION_POWER_SAVE && gRxIdleMode)
  78. BK4819_RX_TurnOn();
  79. SYSTEM_DelayMs(20);
  80. uint16_t ToneConfig = BK4819_ReadRegister(BK4819_REG_71);
  81. uint16_t ToneFrequency;
  82. switch (Beep)
  83. {
  84. default:
  85. case BEEP_NONE:
  86. ToneFrequency = 220;
  87. break;
  88. case BEEP_1KHZ_60MS_OPTIONAL:
  89. ToneFrequency = 1000;
  90. break;
  91. case BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL:
  92. case BEEP_500HZ_60MS_DOUBLE_BEEP:
  93. ToneFrequency = 500;
  94. break;
  95. case BEEP_440HZ_40MS_OPTIONAL:
  96. case BEEP_440HZ_500MS:
  97. ToneFrequency = 440;
  98. break;
  99. case BEEP_880HZ_40MS_OPTIONAL:
  100. case BEEP_880HZ_60MS_TRIPLE_BEEP:
  101. case BEEP_880HZ_200MS:
  102. case BEEP_880HZ_500MS:
  103. ToneFrequency = 880;
  104. break;
  105. }
  106. BK4819_PlayTone(ToneFrequency, true);
  107. SYSTEM_DelayMs(2);
  108. AUDIO_AudioPathOn();
  109. SYSTEM_DelayMs(60);
  110. uint16_t Duration;
  111. switch (Beep)
  112. {
  113. case BEEP_880HZ_60MS_TRIPLE_BEEP:
  114. BK4819_ExitTxMute();
  115. SYSTEM_DelayMs(60);
  116. BK4819_EnterTxMute();
  117. SYSTEM_DelayMs(20);
  118. [[fallthrough]];
  119. case BEEP_500HZ_60MS_DOUBLE_BEEP_OPTIONAL:
  120. case BEEP_500HZ_60MS_DOUBLE_BEEP:
  121. BK4819_ExitTxMute();
  122. SYSTEM_DelayMs(60);
  123. BK4819_EnterTxMute();
  124. SYSTEM_DelayMs(20);
  125. [[fallthrough]];
  126. case BEEP_1KHZ_60MS_OPTIONAL:
  127. BK4819_ExitTxMute();
  128. Duration = 60;
  129. break;
  130. case BEEP_880HZ_40MS_OPTIONAL:
  131. case BEEP_440HZ_40MS_OPTIONAL:
  132. BK4819_ExitTxMute();
  133. Duration = 40;
  134. break;
  135. case BEEP_880HZ_200MS:
  136. BK4819_ExitTxMute();
  137. Duration = 200;
  138. break;
  139. case BEEP_440HZ_500MS:
  140. case BEEP_880HZ_500MS:
  141. default:
  142. BK4819_ExitTxMute();
  143. Duration = 500;
  144. break;
  145. }
  146. SYSTEM_DelayMs(Duration);
  147. BK4819_EnterTxMute();
  148. SYSTEM_DelayMs(20);
  149. AUDIO_AudioPathOff();
  150. SYSTEM_DelayMs(5);
  151. BK4819_TurnsOffTones_TurnsOnRX();
  152. SYSTEM_DelayMs(5);
  153. BK4819_WriteRegister(BK4819_REG_71, ToneConfig);
  154. if (gEnableSpeaker)
  155. AUDIO_AudioPathOn();
  156. #ifdef ENABLE_FMRADIO
  157. if (gFmRadioMode)
  158. BK1080_Mute(false);
  159. #endif
  160. if (gCurrentFunction == FUNCTION_POWER_SAVE && gRxIdleMode)
  161. BK4819_Sleep();
  162. #ifdef ENABLE_VOX
  163. gVoxResumeCountdown = 80;
  164. #endif
  165. }
  166. #ifdef ENABLE_VOICE
  167. static const uint8_t VoiceClipLengthChinese[58] =
  168. {
  169. 0x32, 0x32, 0x32, 0x37, 0x37, 0x32, 0x32, 0x32,
  170. 0x32, 0x37, 0x37, 0x32, 0x64, 0x64, 0x64, 0x64,
  171. 0x64, 0x69, 0x64, 0x69, 0x5A, 0x5F, 0x5F, 0x64,
  172. 0x64, 0x69, 0x64, 0x64, 0x69, 0x69, 0x69, 0x64,
  173. 0x64, 0x6E, 0x69, 0x5F, 0x64, 0x64, 0x64, 0x69,
  174. 0x69, 0x69, 0x64, 0x69, 0x64, 0x64, 0x55, 0x5F,
  175. 0x5A, 0x4B, 0x4B, 0x46, 0x46, 0x69, 0x64, 0x6E,
  176. 0x5A, 0x64,
  177. };
  178. static const uint8_t VoiceClipLengthEnglish[76] =
  179. {
  180. 0x50, 0x32, 0x2D, 0x2D, 0x2D, 0x37, 0x37, 0x37,
  181. 0x32, 0x32, 0x3C, 0x37, 0x46, 0x46, 0x4B, 0x82,
  182. 0x82, 0x6E, 0x82, 0x46, 0x96, 0x64, 0x46, 0x6E,
  183. 0x78, 0x6E, 0x87, 0x64, 0x96, 0x96, 0x46, 0x9B,
  184. 0x91, 0x82, 0x82, 0x73, 0x78, 0x64, 0x82, 0x6E,
  185. 0x78, 0x82, 0x87, 0x6E, 0x55, 0x78, 0x64, 0x69,
  186. 0x9B, 0x5A, 0x50, 0x3C, 0x32, 0x55, 0x64, 0x64,
  187. 0x50, 0x46, 0x46, 0x46, 0x4B, 0x4B, 0x50, 0x50,
  188. 0x55, 0x4B, 0x4B, 0x32, 0x32, 0x32, 0x32, 0x37,
  189. 0x41, 0x32, 0x3C, 0x37,
  190. };
  191. VOICE_ID_t gVoiceID[8];
  192. uint8_t gVoiceReadIndex;
  193. uint8_t gVoiceWriteIndex;
  194. volatile uint16_t gCountdownToPlayNextVoice_10ms;
  195. volatile bool gFlagPlayQueuedVoice;
  196. VOICE_ID_t gAnotherVoiceID = VOICE_ID_INVALID;
  197. static void AUDIO_PlayVoice(uint8_t VoiceID)
  198. {
  199. unsigned int i;
  200. GPIO_SetBit(&GPIOA->DATA, GPIOA_PIN_VOICE_0);
  201. SYSTEM_DelayMs(20);
  202. GPIO_ClearBit(&GPIOA->DATA, GPIOA_PIN_VOICE_0);
  203. for (i = 0; i < 8; i++)
  204. {
  205. if ((VoiceID & 0x80U) == 0)
  206. GPIO_ClearBit(&GPIOA->DATA, GPIOA_PIN_VOICE_1);
  207. else
  208. GPIO_SetBit(&GPIOA->DATA, GPIOA_PIN_VOICE_1);
  209. SYSTICK_DelayUs(1000);
  210. GPIO_SetBit(&GPIOA->DATA, GPIOA_PIN_VOICE_0);
  211. SYSTICK_DelayUs(1200);
  212. GPIO_ClearBit(&GPIOA->DATA, GPIOA_PIN_VOICE_0);
  213. VoiceID <<= 1;
  214. SYSTICK_DelayUs(200);
  215. }
  216. }
  217. void AUDIO_PlaySingleVoice(bool bFlag)
  218. {
  219. uint8_t VoiceID;
  220. uint8_t Delay;
  221. VoiceID = gVoiceID[0];
  222. if (gEeprom.VOICE_PROMPT != VOICE_PROMPT_OFF && gVoiceWriteIndex > 0)
  223. {
  224. if (gEeprom.VOICE_PROMPT == VOICE_PROMPT_CHINESE)
  225. { // Chinese
  226. if (VoiceID >= ARRAY_SIZE(VoiceClipLengthChinese))
  227. goto Bailout;
  228. Delay = VoiceClipLengthChinese[VoiceID];
  229. VoiceID += VOICE_ID_CHI_BASE;
  230. }
  231. else
  232. { // English
  233. if (VoiceID >= ARRAY_SIZE(VoiceClipLengthEnglish))
  234. goto Bailout;
  235. Delay = VoiceClipLengthEnglish[VoiceID];
  236. VoiceID += VOICE_ID_ENG_BASE;
  237. }
  238. if (FUNCTION_IsRx()) // 1of11
  239. BK4819_SetAF(BK4819_AF_MUTE);
  240. #ifdef ENABLE_FMRADIO
  241. if (gFmRadioMode)
  242. BK1080_Mute(true);
  243. #endif
  244. AUDIO_AudioPathOn();
  245. #ifdef ENABLE_VOX
  246. gVoxResumeCountdown = 2000;
  247. #endif
  248. SYSTEM_DelayMs(5);
  249. AUDIO_PlayVoice(VoiceID);
  250. if (gVoiceWriteIndex == 1)
  251. Delay += 3;
  252. if (bFlag)
  253. {
  254. SYSTEM_DelayMs(Delay * 10);
  255. if (FUNCTION_IsRx()) // 1of11
  256. RADIO_SetModulation(gRxVfo->Modulation);
  257. #ifdef ENABLE_FMRADIO
  258. if (gFmRadioMode)
  259. BK1080_Mute(false);
  260. #endif
  261. if (!gEnableSpeaker)
  262. AUDIO_AudioPathOff();
  263. gVoiceWriteIndex = 0;
  264. gVoiceReadIndex = 0;
  265. #ifdef ENABLE_VOX
  266. gVoxResumeCountdown = 80;
  267. #endif
  268. return;
  269. }
  270. gVoiceReadIndex = 1;
  271. gCountdownToPlayNextVoice_10ms = Delay;
  272. gFlagPlayQueuedVoice = false;
  273. return;
  274. }
  275. Bailout:
  276. gVoiceReadIndex = 0;
  277. gVoiceWriteIndex = 0;
  278. }
  279. void AUDIO_SetVoiceID(uint8_t Index, VOICE_ID_t VoiceID)
  280. {
  281. if (Index >= ARRAY_SIZE(gVoiceID))
  282. return;
  283. if (Index == 0)
  284. {
  285. gVoiceWriteIndex = 0;
  286. gVoiceReadIndex = 0;
  287. }
  288. gVoiceID[Index] = VoiceID;
  289. gVoiceWriteIndex++;
  290. }
  291. uint8_t AUDIO_SetDigitVoice(uint8_t Index, uint16_t Value)
  292. {
  293. uint16_t Remainder;
  294. uint8_t Result;
  295. uint8_t Count;
  296. if (Index == 0)
  297. {
  298. gVoiceWriteIndex = 0;
  299. gVoiceReadIndex = 0;
  300. }
  301. Count = 0;
  302. Result = Value / 1000U;
  303. Remainder = Value % 1000U;
  304. if (Remainder < 100U)
  305. {
  306. if (Remainder < 10U)
  307. goto Skip;
  308. }
  309. else
  310. {
  311. Result = Remainder / 100U;
  312. gVoiceID[gVoiceWriteIndex++] = (VOICE_ID_t)Result;
  313. Count++;
  314. Remainder -= Result * 100U;
  315. }
  316. Result = Remainder / 10U;
  317. gVoiceID[gVoiceWriteIndex++] = (VOICE_ID_t)Result;
  318. Count++;
  319. Remainder -= Result * 10U;
  320. Skip:
  321. gVoiceID[gVoiceWriteIndex++] = (VOICE_ID_t)Remainder;
  322. return Count + 1U;
  323. }
  324. void AUDIO_PlayQueuedVoice(void)
  325. {
  326. uint8_t VoiceID;
  327. uint8_t Delay;
  328. bool Skip;
  329. Skip = false;
  330. if (gVoiceReadIndex != gVoiceWriteIndex && gEeprom.VOICE_PROMPT != VOICE_PROMPT_OFF)
  331. {
  332. VoiceID = gVoiceID[gVoiceReadIndex];
  333. if (gEeprom.VOICE_PROMPT == VOICE_PROMPT_CHINESE)
  334. {
  335. if (VoiceID < ARRAY_SIZE(VoiceClipLengthChinese))
  336. {
  337. Delay = VoiceClipLengthChinese[VoiceID];
  338. VoiceID += VOICE_ID_CHI_BASE;
  339. }
  340. else
  341. Skip = true;
  342. }
  343. else
  344. {
  345. if (VoiceID < ARRAY_SIZE(VoiceClipLengthEnglish))
  346. {
  347. Delay = VoiceClipLengthEnglish[VoiceID];
  348. VoiceID += VOICE_ID_ENG_BASE;
  349. }
  350. else
  351. Skip = true;
  352. }
  353. gVoiceReadIndex++;
  354. if (!Skip)
  355. {
  356. if (gVoiceReadIndex == gVoiceWriteIndex)
  357. Delay += 3;
  358. AUDIO_PlayVoice(VoiceID);
  359. gCountdownToPlayNextVoice_10ms = Delay;
  360. gFlagPlayQueuedVoice = false;
  361. #ifdef ENABLE_VOX
  362. gVoxResumeCountdown = 2000;
  363. #endif
  364. return;
  365. }
  366. }
  367. if (FUNCTION_IsRx())
  368. {
  369. RADIO_SetModulation(gRxVfo->Modulation); // 1of11
  370. }
  371. #ifdef ENABLE_FMRADIO
  372. if (gFmRadioMode)
  373. BK1080_Mute(false);
  374. #endif
  375. if (!gEnableSpeaker)
  376. AUDIO_AudioPathOff();
  377. #ifdef ENABLE_VOX
  378. gVoxResumeCountdown = 80;
  379. #endif
  380. gVoiceWriteIndex = 0;
  381. gVoiceReadIndex = 0;
  382. }
  383. #endif