settings.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815
  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. #include <string.h>
  17. #include "app/dtmf.h"
  18. #ifdef ENABLE_FMRADIO
  19. #include "app/fm.h"
  20. #endif
  21. #ifdef ENABLE_PMR_MODE
  22. #include "app/pmr.h"
  23. #endif
  24. #include "driver/bk1080.h"
  25. #include "driver/bk4819.h"
  26. #include "driver/eeprom.h"
  27. #include "misc.h"
  28. #include "settings.h"
  29. #include "ui/menu.h"
  30. static const uint32_t gDefaultFrequencyTable[] =
  31. {
  32. 14500000, //
  33. 14550000, //
  34. 43300000, //
  35. 43320000, //
  36. 43350000 //
  37. };
  38. EEPROM_Config_t gEeprom = { 0 };
  39. #ifdef ENABLE_PMR_MODE
  40. void SETTINGS_SavePMR() {
  41. uint8_t Data[8] = {0};
  42. EEPROM_ReadBuffer(0x0E70, Data, 8);
  43. Data[3] = 0xFF;
  44. if (!gPMR_Mode_Active) Data[3] &= ~(1u << 7);
  45. EEPROM_WriteBuffer(0x0E70, Data);
  46. }
  47. #endif
  48. void SETTINGS_InitEEPROM(void)
  49. {
  50. uint8_t Data[16] = {0};
  51. // 0E70..0E77
  52. EEPROM_ReadBuffer(0x0E70, Data, 8);
  53. gEeprom.CHAN_1_CALL = IS_MR_CHANNEL(Data[0]) ? Data[0] : MR_CHANNEL_FIRST;
  54. gEeprom.SQUELCH_LEVEL = (Data[1] < 10) ? Data[1] : 1;
  55. gEeprom.TX_TIMEOUT_TIMER = (Data[2] < 11) ? Data[2] : 1;
  56. #ifdef ENABLE_NOAA
  57. //gEeprom.NOAA_AUTO_SCAN = (Data[3] < 2) ? Data[3] : false;
  58. gEeprom.NOAA_AUTO_SCAN = !!(Data[3] & (1u << 1));
  59. #endif
  60. #ifdef ENABLE_PMR_MODE
  61. gPMR_Mode_Active = !!(Data[3] & (1u << 7));
  62. #endif
  63. gEeprom.KEY_LOCK = (Data[4] < 2) ? Data[4] : false;
  64. #ifdef ENABLE_VOX
  65. gEeprom.VOX_SWITCH = (Data[5] < 2) ? Data[5] : false;
  66. gEeprom.VOX_LEVEL = (Data[6] < 10) ? Data[6] : 1;
  67. #endif
  68. gEeprom.MIC_SENSITIVITY = (Data[7] < 5) ? Data[7] : 4;
  69. // 0E78..0E7F
  70. EEPROM_ReadBuffer(0x0E78, Data, 8);
  71. gEeprom.BACKLIGHT_MAX = (Data[0] & 0xF) <= 10 ? (Data[0] & 0xF) : 10;
  72. gEeprom.BACKLIGHT_MIN = (Data[0] >> 4) < gEeprom.BACKLIGHT_MAX ? (Data[0] >> 4) : 0;
  73. #ifdef ENABLE_BLMIN_TMP_OFF
  74. gEeprom.BACKLIGHT_MIN_STAT = BLMIN_STAT_ON;
  75. #endif
  76. gEeprom.CHANNEL_DISPLAY_MODE = (Data[1] < 4) ? Data[1] : MDF_FREQUENCY; // 4 instead of 3 - extra display mode
  77. gEeprom.CROSS_BAND_RX_TX = (Data[2] < 3) ? Data[2] : CROSS_BAND_OFF;
  78. gEeprom.BATTERY_SAVE = (Data[3] < 5) ? Data[3] : 4;
  79. gEeprom.DUAL_WATCH = (Data[4] < 3) ? Data[4] : DUAL_WATCH_CHAN_A;
  80. gEeprom.BACKLIGHT_TIME = (Data[5] < ARRAY_SIZE(gSubMenu_BACKLIGHT)) ? Data[5] : 3;
  81. //gEeprom.TAIL_TONE_ELIMINATION = (Data[6] < 2) ? Data[6] : false;
  82. gEeprom.VFO_OPEN = (Data[7] < 2) ? Data[7] : true;
  83. gEeprom.TAIL_TONE_ELIMINATION = (Data[6] >> 7) & 1;
  84. #ifdef ENABLE_CONTRAST
  85. gEeprom.LCD_CONTRAST = Data[6] & 0x7F;
  86. #endif
  87. // 0E80..0E87
  88. EEPROM_ReadBuffer(0x0E80, Data, 8);
  89. gEeprom.ScreenChannel[0] = IS_VALID_CHANNEL(Data[0]) ? Data[0] : (FREQ_CHANNEL_FIRST + BAND6_400MHz);
  90. gEeprom.ScreenChannel[1] = IS_VALID_CHANNEL(Data[3]) ? Data[3] : (FREQ_CHANNEL_FIRST + BAND6_400MHz);
  91. gEeprom.MrChannel[0] = IS_MR_CHANNEL(Data[1]) ? Data[1] : MR_CHANNEL_FIRST;
  92. gEeprom.MrChannel[1] = IS_MR_CHANNEL(Data[4]) ? Data[4] : MR_CHANNEL_FIRST;
  93. gEeprom.FreqChannel[0] = IS_FREQ_CHANNEL(Data[2]) ? Data[2] : (FREQ_CHANNEL_FIRST + BAND6_400MHz);
  94. gEeprom.FreqChannel[1] = IS_FREQ_CHANNEL(Data[5]) ? Data[5] : (FREQ_CHANNEL_FIRST + BAND6_400MHz);
  95. #ifdef ENABLE_NOAA
  96. gEeprom.NoaaChannel[0] = IS_NOAA_CHANNEL(Data[6]) ? Data[6] : NOAA_CHANNEL_FIRST;
  97. gEeprom.NoaaChannel[1] = IS_NOAA_CHANNEL(Data[7]) ? Data[7] : NOAA_CHANNEL_FIRST;
  98. #endif
  99. #ifdef ENABLE_FMRADIO
  100. { // 0E88..0E8F
  101. struct
  102. {
  103. uint16_t selFreq;
  104. uint8_t selChn;
  105. uint8_t isMrMode:1;
  106. uint8_t band:2;
  107. uint8_t space:2;
  108. uint8_t preemp:1;
  109. } __attribute__((packed)) fmCfg;
  110. EEPROM_ReadBuffer(0x0E88, &fmCfg, sizeof(fmCfg));
  111. gEeprom.FM_Band = fmCfg.band;
  112. gEeprom.FM_Space = fmCfg.space;
  113. gEeprom.FM_Preemp = fmCfg.preemp;
  114. gEeprom.FM_SelectedFrequency =
  115. (fmCfg.selFreq >= BK1080_GetFreqLoLimit(gEeprom.FM_Band) && fmCfg.selFreq <= BK1080_GetFreqHiLimit(gEeprom.FM_Band)) ?
  116. fmCfg.selFreq : BK1080_GetFreqLoLimit(gEeprom.FM_Band);
  117. gEeprom.FM_SelectedChannel = fmCfg.selChn;
  118. gEeprom.FM_IsMrMode = fmCfg.isMrMode;
  119. }
  120. // 0E40..0E67
  121. EEPROM_ReadBuffer(0x0E40, gFM_Channels, sizeof(gFM_Channels));
  122. FM_ConfigureChannelState();
  123. #endif
  124. // 0E90..0E97
  125. EEPROM_ReadBuffer(0x0E90, Data, 8);
  126. gEeprom.BEEP_CONTROL = Data[0] & 1;
  127. gEeprom.KEY_M_LONG_PRESS_ACTION = ((Data[0] >> 1) < ACTION_OPT_LEN) ? (Data[0] >> 1) : ACTION_OPT_NONE;
  128. gEeprom.KEY_1_SHORT_PRESS_ACTION = (Data[1] < ACTION_OPT_LEN) ? Data[1] : ACTION_OPT_MONITOR;
  129. gEeprom.KEY_1_LONG_PRESS_ACTION = (Data[2] < ACTION_OPT_LEN) ? Data[2] : ACTION_OPT_SCAN;
  130. gEeprom.KEY_2_SHORT_PRESS_ACTION = (Data[3] < ACTION_OPT_LEN) ? Data[3] : ACTION_OPT_A_B;
  131. gEeprom.KEY_2_LONG_PRESS_ACTION = (Data[4] < ACTION_OPT_LEN) ? Data[4] : ACTION_OPT_FM;
  132. gEeprom.SCAN_RESUME_MODE = (Data[5] < 3) ? Data[5] : SCAN_RESUME_CO;
  133. gEeprom.AUTO_KEYPAD_LOCK = (Data[6] < 2) ? Data[6] : false;
  134. gEeprom.POWER_ON_DISPLAY_MODE = (Data[7] < 4) ? Data[7] : POWER_ON_DISPLAY_MODE_FULL_SCREEN;
  135. // 0E98..0E9F
  136. EEPROM_ReadBuffer(0x0E98, Data, 8);
  137. memcpy(&gEeprom.POWER_ON_PASSWORD, Data, 4);
  138. // 0EA0..0EA7
  139. EEPROM_ReadBuffer(0x0EA0, Data, 8);
  140. #ifdef ENABLE_VOICE
  141. gEeprom.VOICE_PROMPT = (Data[0] < 3) ? Data[0] : VOICE_PROMPT_ENGLISH;
  142. #endif
  143. #ifdef ENABLE_RSSI_BAR
  144. if((Data[1] < 200 && Data[1] > 90) && (Data[2] < Data[1]-9 && Data[1] < 160 && Data[2] > 50)) {
  145. gEeprom.S0_LEVEL = Data[1];
  146. gEeprom.S9_LEVEL = Data[2];
  147. }
  148. else {
  149. gEeprom.S0_LEVEL = 130;
  150. gEeprom.S9_LEVEL = 76;
  151. }
  152. #endif
  153. // 0EA8..0EAF
  154. EEPROM_ReadBuffer(0x0EA8, Data, 8);
  155. #ifdef ENABLE_ALARM
  156. gEeprom.ALARM_MODE = (Data[0] < 2) ? Data[0] : true;
  157. #endif
  158. gEeprom.ROGER = (Data[1] < 3) ? Data[1] : ROGER_MODE_OFF;
  159. gEeprom.REPEATER_TAIL_TONE_ELIMINATION = (Data[2] < 11) ? Data[2] : 0;
  160. gEeprom.TX_VFO = (Data[3] < 2) ? Data[3] : 0;
  161. gEeprom.BATTERY_TYPE = (Data[4] < BATTERY_TYPE_UNKNOWN) ? Data[4] : BATTERY_TYPE_1600_MAH;
  162. // 0ED0..0ED7
  163. EEPROM_ReadBuffer(0x0ED0, Data, 8);
  164. gEeprom.DTMF_SIDE_TONE = (Data[0] < 2) ? Data[0] : true;
  165. #ifdef ENABLE_DTMF_CALLING
  166. gEeprom.DTMF_SEPARATE_CODE = DTMF_ValidateCodes((char *)(Data + 1), 1) ? Data[1] : '*';
  167. gEeprom.DTMF_GROUP_CALL_CODE = DTMF_ValidateCodes((char *)(Data + 2), 1) ? Data[2] : '#';
  168. gEeprom.DTMF_DECODE_RESPONSE = (Data[3] < 4) ? Data[3] : 0;
  169. gEeprom.DTMF_auto_reset_time = (Data[4] < 61 && Data[4] >= 5) ? Data[4] : 10;
  170. #endif
  171. gEeprom.DTMF_PRELOAD_TIME = (Data[5] < 101) ? Data[5] * 10 : 300;
  172. gEeprom.DTMF_FIRST_CODE_PERSIST_TIME = (Data[6] < 101) ? Data[6] * 10 : 100;
  173. gEeprom.DTMF_HASH_CODE_PERSIST_TIME = (Data[7] < 101) ? Data[7] * 10 : 100;
  174. // 0ED8..0EDF
  175. EEPROM_ReadBuffer(0x0ED8, Data, 8);
  176. gEeprom.DTMF_CODE_PERSIST_TIME = (Data[0] < 101) ? Data[0] * 10 : 100;
  177. gEeprom.DTMF_CODE_INTERVAL_TIME = (Data[1] < 101) ? Data[1] * 10 : 100;
  178. #ifdef ENABLE_DTMF_CALLING
  179. gEeprom.PERMIT_REMOTE_KILL = (Data[2] < 2) ? Data[2] : true;
  180. // 0EE0..0EE7
  181. EEPROM_ReadBuffer(0x0EE0, Data, sizeof(gEeprom.ANI_DTMF_ID));
  182. if (DTMF_ValidateCodes((char *)Data, sizeof(gEeprom.ANI_DTMF_ID))) {
  183. memcpy(gEeprom.ANI_DTMF_ID, Data, sizeof(gEeprom.ANI_DTMF_ID));
  184. } else {
  185. strcpy(gEeprom.ANI_DTMF_ID, "123");
  186. }
  187. // 0EE8..0EEF
  188. EEPROM_ReadBuffer(0x0EE8, Data, sizeof(gEeprom.KILL_CODE));
  189. if (DTMF_ValidateCodes((char *)Data, sizeof(gEeprom.KILL_CODE))) {
  190. memcpy(gEeprom.KILL_CODE, Data, sizeof(gEeprom.KILL_CODE));
  191. } else {
  192. strcpy(gEeprom.KILL_CODE, "DEADB");
  193. }
  194. // 0EF0..0EF7
  195. EEPROM_ReadBuffer(0x0EF0, Data, sizeof(gEeprom.REVIVE_CODE));
  196. if (DTMF_ValidateCodes((char *)Data, sizeof(gEeprom.REVIVE_CODE))) {
  197. memcpy(gEeprom.REVIVE_CODE, Data, sizeof(gEeprom.REVIVE_CODE));
  198. } else {
  199. strcpy(gEeprom.REVIVE_CODE, "9DCBA");
  200. }
  201. #endif
  202. // 0EF8..0F07
  203. EEPROM_ReadBuffer(0x0EF8, Data, sizeof(gEeprom.DTMF_UP_CODE));
  204. if (DTMF_ValidateCodes((char *)Data, sizeof(gEeprom.DTMF_UP_CODE))) {
  205. memcpy(gEeprom.DTMF_UP_CODE, Data, sizeof(gEeprom.DTMF_UP_CODE));
  206. } else {
  207. strcpy(gEeprom.DTMF_UP_CODE, "12345");
  208. }
  209. // 0F08..0F17
  210. EEPROM_ReadBuffer(0x0F08, Data, sizeof(gEeprom.DTMF_DOWN_CODE));
  211. if (DTMF_ValidateCodes((char *)Data, sizeof(gEeprom.DTMF_DOWN_CODE))) {
  212. memcpy(gEeprom.DTMF_DOWN_CODE, Data, sizeof(gEeprom.DTMF_DOWN_CODE));
  213. } else {
  214. strcpy(gEeprom.DTMF_DOWN_CODE, "54321");
  215. }
  216. // 0F18..0F1F
  217. EEPROM_ReadBuffer(0x0F18, Data, 8);
  218. gEeprom.SCAN_LIST_DEFAULT = (Data[0] < 3) ? Data[0] : 0; // we now have 'all' channel scan option
  219. for (unsigned int i = 0; i < 2; i++)
  220. {
  221. const unsigned int j = 1 + (i * 3);
  222. gEeprom.SCAN_LIST_ENABLED[i] = (Data[j + 0] < 2) ? Data[j] : false;
  223. gEeprom.SCANLIST_PRIORITY_CH1[i] = Data[j + 1];
  224. gEeprom.SCANLIST_PRIORITY_CH2[i] = Data[j + 2];
  225. }
  226. // 0F40..0F47
  227. EEPROM_ReadBuffer(0x0F40, Data, 8);
  228. gSetting_F_LOCK = (Data[0] < F_LOCK_LEN) ? Data[0] : F_LOCK_DEF;
  229. gSetting_350TX = (Data[1] < 2) ? Data[1] : false; // was true
  230. #ifdef ENABLE_DTMF_CALLING
  231. gSetting_KILLED = (Data[2] < 2) ? Data[2] : false;
  232. #endif
  233. gSetting_200TX = (Data[3] < 2) ? Data[3] : false;
  234. gSetting_500TX = (Data[4] < 2) ? Data[4] : false;
  235. gSetting_350EN = (Data[5] < 2) ? Data[5] : true;
  236. gSetting_ScrambleEnable = (Data[6] < 2) ? Data[6] : true;
  237. //gSetting_TX_EN = (Data[7] & (1u << 0)) ? true : false;
  238. gSetting_live_DTMF_decoder = !!(Data[7] & (1u << 1));
  239. gSetting_battery_text = (((Data[7] >> 2) & 3u) <= 2) ? (Data[7] >> 2) & 3 : 2;
  240. #ifdef ENABLE_AUDIO_BAR
  241. gSetting_mic_bar = !!(Data[7] & (1u << 4));
  242. #endif
  243. //gSetting_AM_fix = !!(Data[7] & (1u << 5));
  244. gSetting_backlight_on_tx_rx = (Data[7] >> 6) & 3u;
  245. if (!gEeprom.VFO_OPEN)
  246. {
  247. gEeprom.ScreenChannel[0] = gEeprom.MrChannel[0];
  248. gEeprom.ScreenChannel[1] = gEeprom.MrChannel[1];
  249. }
  250. // 0D60..0E27
  251. EEPROM_ReadBuffer(0x0D60, gMR_ChannelAttributes, sizeof(gMR_ChannelAttributes));
  252. for(uint16_t i = 0; i < sizeof(gMR_ChannelAttributes); i++) {
  253. ChannelAttributes_t *att = &gMR_ChannelAttributes[i];
  254. if(att->__val == 0xff){
  255. att->__val = 0;
  256. att->band = 0xf;
  257. }
  258. }
  259. // 0F30..0F3F
  260. /*EEPROM_ReadBuffer(0x0F30, gCustomAesKey, sizeof(gCustomAesKey));
  261. bHasCustomAesKey = false;
  262. for (unsigned int i = 0; i < ARRAY_SIZE(gCustomAesKey); i++)
  263. {
  264. if (gCustomAesKey[i] != 0xFFFFFFFFu)
  265. {
  266. bHasCustomAesKey = true;
  267. return;
  268. }
  269. }*/
  270. }
  271. void SETTINGS_LoadCalibration(void)
  272. {
  273. // uint8_t Mic;
  274. EEPROM_ReadBuffer(0x1EC0, gEEPROM_RSSI_CALIB[3], 8);
  275. memcpy(gEEPROM_RSSI_CALIB[4], gEEPROM_RSSI_CALIB[3], 8);
  276. memcpy(gEEPROM_RSSI_CALIB[5], gEEPROM_RSSI_CALIB[3], 8);
  277. memcpy(gEEPROM_RSSI_CALIB[6], gEEPROM_RSSI_CALIB[3], 8);
  278. EEPROM_ReadBuffer(0x1EC8, gEEPROM_RSSI_CALIB[0], 8);
  279. memcpy(gEEPROM_RSSI_CALIB[1], gEEPROM_RSSI_CALIB[0], 8);
  280. memcpy(gEEPROM_RSSI_CALIB[2], gEEPROM_RSSI_CALIB[0], 8);
  281. EEPROM_ReadBuffer(0x1F40, gBatteryCalibration, 12);
  282. if (gBatteryCalibration[0] >= 5000)
  283. {
  284. gBatteryCalibration[0] = 1900;
  285. gBatteryCalibration[1] = 2000;
  286. }
  287. gBatteryCalibration[5] = 2300;
  288. #ifdef ENABLE_VOX
  289. EEPROM_ReadBuffer(0x1F50 + (gEeprom.VOX_LEVEL * 2), &gEeprom.VOX1_THRESHOLD, 2);
  290. EEPROM_ReadBuffer(0x1F68 + (gEeprom.VOX_LEVEL * 2), &gEeprom.VOX0_THRESHOLD, 2);
  291. #endif
  292. //EEPROM_ReadBuffer(0x1F80 + gEeprom.MIC_SENSITIVITY, &Mic, 1);
  293. //gEeprom.MIC_SENSITIVITY_TUNING = (Mic < 32) ? Mic : 15;
  294. gEeprom.MIC_SENSITIVITY_TUNING = gMicGain_dB2[gEeprom.MIC_SENSITIVITY];
  295. {
  296. struct
  297. {
  298. int16_t BK4819_XtalFreqLow;
  299. uint16_t EEPROM_1F8A;
  300. uint16_t EEPROM_1F8C;
  301. uint8_t VOLUME_GAIN;
  302. uint8_t DAC_GAIN;
  303. } __attribute__((packed)) Misc;
  304. // radio 1 .. 04 00 46 00 50 00 2C 0E
  305. // radio 2 .. 05 00 46 00 50 00 2C 0E
  306. EEPROM_ReadBuffer(0x1F88, &Misc, 8);
  307. gEeprom.BK4819_XTAL_FREQ_LOW = (Misc.BK4819_XtalFreqLow >= -1000 && Misc.BK4819_XtalFreqLow <= 1000) ? Misc.BK4819_XtalFreqLow : 0;
  308. gEEPROM_1F8A = Misc.EEPROM_1F8A & 0x01FF;
  309. gEEPROM_1F8C = Misc.EEPROM_1F8C & 0x01FF;
  310. gEeprom.VOLUME_GAIN = (Misc.VOLUME_GAIN < 64) ? Misc.VOLUME_GAIN : 58;
  311. gEeprom.DAC_GAIN = (Misc.DAC_GAIN < 16) ? Misc.DAC_GAIN : 8;
  312. BK4819_WriteRegister(BK4819_REG_3B, 22656 + gEeprom.BK4819_XTAL_FREQ_LOW);
  313. // BK4819_WriteRegister(BK4819_REG_3C, gEeprom.BK4819_XTAL_FREQ_HIGH);
  314. }
  315. }
  316. uint32_t SETTINGS_FetchChannelFrequency(const int channel)
  317. {
  318. struct
  319. {
  320. uint32_t frequency;
  321. uint32_t offset;
  322. } __attribute__((packed)) info;
  323. EEPROM_ReadBuffer(channel * 16, &info, sizeof(info));
  324. return info.frequency;
  325. }
  326. void SETTINGS_FetchChannelName(char *s, const int channel)
  327. {
  328. if (s == NULL)
  329. return;
  330. s[0] = 0;
  331. if (channel < 0)
  332. return;
  333. if (!RADIO_CheckValidChannel(channel, false, 0))
  334. return;
  335. EEPROM_ReadBuffer(0x0F50 + (channel * 16), s, 10);
  336. int i;
  337. for (i = 0; i < 10; i++)
  338. if (s[i] < 32 || s[i] > 127)
  339. break; // invalid char
  340. s[i--] = 0; // null term
  341. while (i >= 0 && s[i] == 32) // trim trailing spaces
  342. s[i--] = 0; // null term
  343. }
  344. void SETTINGS_FactoryReset(bool bIsAll)
  345. {
  346. uint16_t i;
  347. uint8_t Template[8];
  348. memset(Template, 0xFF, sizeof(Template));
  349. for (i = 0x0C80; i < 0x1E00; i += 8)
  350. {
  351. if (
  352. !(i >= 0x0EE0 && i < 0x0F18) && // ANI ID + DTMF codes
  353. !(i >= 0x0F30 && i < 0x0F50) && // AES KEY + F LOCK + Scramble Enable
  354. !(i >= 0x1C00 && i < 0x1E00) && // DTMF contacts
  355. !(i >= 0x0EB0 && i < 0x0ED0) && // Welcome strings
  356. !(i >= 0x0EA0 && i < 0x0EA8) && // Voice Prompt
  357. (bIsAll ||
  358. (
  359. !(i >= 0x0D60 && i < 0x0E28) && // MR Channel Attributes
  360. !(i >= 0x0F18 && i < 0x0F30) && // Scan List
  361. !(i >= 0x0F50 && i < 0x1C00) && // MR Channel Names
  362. !(i >= 0x0E40 && i < 0x0E70) && // FM Channels
  363. !(i >= 0x0E88 && i < 0x0E90) // FM settings
  364. ))
  365. )
  366. {
  367. EEPROM_WriteBuffer(i, Template);
  368. }
  369. }
  370. if (bIsAll)
  371. {
  372. RADIO_InitInfo(gRxVfo, FREQ_CHANNEL_FIRST + BAND6_400MHz, 43350000);
  373. // set the first few memory channels
  374. for (i = 0; i < ARRAY_SIZE(gDefaultFrequencyTable); i++)
  375. {
  376. const uint32_t Frequency = gDefaultFrequencyTable[i];
  377. gRxVfo->freq_config_RX.Frequency = Frequency;
  378. gRxVfo->freq_config_TX.Frequency = Frequency;
  379. gRxVfo->Band = FREQUENCY_GetBand(Frequency);
  380. SETTINGS_SaveChannel(MR_CHANNEL_FIRST + i, 0, gRxVfo, 2);
  381. }
  382. }
  383. }
  384. #ifdef ENABLE_FMRADIO
  385. void SETTINGS_SaveFM(void)
  386. {
  387. union {
  388. struct {
  389. uint16_t selFreq;
  390. uint8_t selChn;
  391. uint8_t isMrMode:1;
  392. uint8_t band:2;
  393. uint8_t space:2;
  394. uint8_t preemp:1;
  395. };
  396. uint8_t __raw[8];
  397. } __attribute__((packed)) fmCfg;
  398. memset(fmCfg.__raw, 0xFF, sizeof(fmCfg.__raw));
  399. fmCfg.selChn = gEeprom.FM_SelectedChannel;
  400. fmCfg.selFreq = gEeprom.FM_SelectedFrequency;
  401. fmCfg.isMrMode = gEeprom.FM_IsMrMode;
  402. fmCfg.band = gEeprom.FM_Band;
  403. fmCfg.space = gEeprom.FM_Space;
  404. fmCfg.preemp = gEeprom.FM_Preemp;
  405. EEPROM_WriteBuffer(0x0E88, fmCfg.__raw);
  406. for (unsigned i = 0; i < 5; i++)
  407. EEPROM_WriteBuffer(0x0E40 + (i * 8), &gFM_Channels[i * 4]);
  408. }
  409. #endif
  410. void SETTINGS_SaveVfoIndices(void)
  411. {
  412. uint8_t State[8];
  413. #ifndef ENABLE_NOAA
  414. EEPROM_ReadBuffer(0x0E80, State, sizeof(State));
  415. #endif
  416. State[0] = gEeprom.ScreenChannel[0];
  417. State[1] = gEeprom.MrChannel[0];
  418. State[2] = gEeprom.FreqChannel[0];
  419. State[3] = gEeprom.ScreenChannel[1];
  420. State[4] = gEeprom.MrChannel[1];
  421. State[5] = gEeprom.FreqChannel[1];
  422. #ifdef ENABLE_NOAA
  423. State[6] = gEeprom.NoaaChannel[0];
  424. State[7] = gEeprom.NoaaChannel[1];
  425. #endif
  426. EEPROM_WriteBuffer(0x0E80, State);
  427. }
  428. void SETTINGS_SaveSettings(void)
  429. {
  430. #ifdef ENABLE_PMR_MODE
  431. if ( gPMR_Mode_Active ) {
  432. return;
  433. }
  434. #endif
  435. uint8_t State[8];
  436. uint32_t Password[2];
  437. State[0] = gEeprom.CHAN_1_CALL;
  438. State[1] = gEeprom.SQUELCH_LEVEL;
  439. State[2] = gEeprom.TX_TIMEOUT_TIMER;
  440. #ifdef ENABLE_NOAA
  441. State[3] = gEeprom.NOAA_AUTO_SCAN;
  442. #else
  443. #ifdef ENABLE_PMR_MODE
  444. State[3] = (gPMR_Mode_Active << 7);
  445. #else
  446. State[3] = false;
  447. #endif
  448. #endif
  449. State[4] = gEeprom.KEY_LOCK;
  450. #ifdef ENABLE_VOX
  451. State[5] = gEeprom.VOX_SWITCH;
  452. State[6] = gEeprom.VOX_LEVEL;
  453. #else
  454. State[5] = false;
  455. State[6] = 0;
  456. #endif
  457. State[7] = gEeprom.MIC_SENSITIVITY;
  458. EEPROM_WriteBuffer(0x0E70, State);
  459. State[0] = (gEeprom.BACKLIGHT_MIN << 4) + gEeprom.BACKLIGHT_MAX;
  460. State[1] = gEeprom.CHANNEL_DISPLAY_MODE;
  461. State[2] = gEeprom.CROSS_BAND_RX_TX;
  462. State[3] = gEeprom.BATTERY_SAVE;
  463. State[4] = gEeprom.DUAL_WATCH;
  464. State[5] = gEeprom.BACKLIGHT_TIME;
  465. //State[6] = gEeprom.TAIL_TONE_ELIMINATION;
  466. State[7] = gEeprom.VFO_OPEN;
  467. #ifdef ENABLE_CONTRAST
  468. State[6] = (gEeprom.LCD_CONTRAST & 0x7F) | (gEeprom.TAIL_TONE_ELIMINATION << 7);
  469. #else
  470. State[6] = (gEeprom.TAIL_TONE_ELIMINATION << 7);
  471. #endif
  472. EEPROM_WriteBuffer(0x0E78, State);
  473. State[0] = gEeprom.BEEP_CONTROL;
  474. State[0] |= gEeprom.KEY_M_LONG_PRESS_ACTION << 1;
  475. State[1] = gEeprom.KEY_1_SHORT_PRESS_ACTION;
  476. State[2] = gEeprom.KEY_1_LONG_PRESS_ACTION;
  477. State[3] = gEeprom.KEY_2_SHORT_PRESS_ACTION;
  478. State[4] = gEeprom.KEY_2_LONG_PRESS_ACTION;
  479. State[5] = gEeprom.SCAN_RESUME_MODE;
  480. State[6] = gEeprom.AUTO_KEYPAD_LOCK;
  481. State[7] = gEeprom.POWER_ON_DISPLAY_MODE;
  482. EEPROM_WriteBuffer(0x0E90, State);
  483. memset(Password, 0xFF, sizeof(Password));
  484. #ifdef ENABLE_PWRON_PASSWORD
  485. Password[0] = gEeprom.POWER_ON_PASSWORD;
  486. #endif
  487. EEPROM_WriteBuffer(0x0E98, Password);
  488. memset(State, 0xFF, sizeof(State));
  489. #ifdef ENABLE_VOICE
  490. State[0] = gEeprom.VOICE_PROMPT;
  491. #endif
  492. #ifdef ENABLE_RSSI_BAR
  493. State[1] = gEeprom.S0_LEVEL;
  494. State[2] = gEeprom.S9_LEVEL;
  495. #endif
  496. EEPROM_WriteBuffer(0x0EA0, State);
  497. #if defined(ENABLE_ALARM) || defined(ENABLE_TX1750)
  498. State[0] = gEeprom.ALARM_MODE;
  499. #else
  500. State[0] = false;
  501. #endif
  502. State[1] = gEeprom.ROGER;
  503. State[2] = gEeprom.REPEATER_TAIL_TONE_ELIMINATION;
  504. State[3] = gEeprom.TX_VFO;
  505. State[4] = gEeprom.BATTERY_TYPE;
  506. EEPROM_WriteBuffer(0x0EA8, State);
  507. State[0] = gEeprom.DTMF_SIDE_TONE;
  508. #ifdef ENABLE_DTMF_CALLING
  509. State[1] = gEeprom.DTMF_SEPARATE_CODE;
  510. State[2] = gEeprom.DTMF_GROUP_CALL_CODE;
  511. State[3] = gEeprom.DTMF_DECODE_RESPONSE;
  512. State[4] = gEeprom.DTMF_auto_reset_time;
  513. #endif
  514. State[5] = gEeprom.DTMF_PRELOAD_TIME / 10U;
  515. State[6] = gEeprom.DTMF_FIRST_CODE_PERSIST_TIME / 10U;
  516. State[7] = gEeprom.DTMF_HASH_CODE_PERSIST_TIME / 10U;
  517. EEPROM_WriteBuffer(0x0ED0, State);
  518. memset(State, 0xFF, sizeof(State));
  519. State[0] = gEeprom.DTMF_CODE_PERSIST_TIME / 10U;
  520. State[1] = gEeprom.DTMF_CODE_INTERVAL_TIME / 10U;
  521. #ifdef ENABLE_DTMF_CALLING
  522. State[2] = gEeprom.PERMIT_REMOTE_KILL;
  523. #endif
  524. EEPROM_WriteBuffer(0x0ED8, State);
  525. State[0] = gEeprom.SCAN_LIST_DEFAULT;
  526. State[1] = gEeprom.SCAN_LIST_ENABLED[0];
  527. State[2] = gEeprom.SCANLIST_PRIORITY_CH1[0];
  528. State[3] = gEeprom.SCANLIST_PRIORITY_CH2[0];
  529. State[4] = gEeprom.SCAN_LIST_ENABLED[1];
  530. State[5] = gEeprom.SCANLIST_PRIORITY_CH1[1];
  531. State[6] = gEeprom.SCANLIST_PRIORITY_CH2[1];
  532. State[7] = 0xFF;
  533. EEPROM_WriteBuffer(0x0F18, State);
  534. memset(State, 0xFF, sizeof(State));
  535. State[0] = gSetting_F_LOCK;
  536. State[1] = gSetting_350TX;
  537. #ifdef ENABLE_DTMF_CALLING
  538. State[2] = gSetting_KILLED;
  539. #endif
  540. State[3] = gSetting_200TX;
  541. State[4] = gSetting_500TX;
  542. State[5] = gSetting_350EN;
  543. State[6] = gSetting_ScrambleEnable;
  544. //if (!gSetting_TX_EN) State[7] &= ~(1u << 0);
  545. if (!gSetting_live_DTMF_decoder) State[7] &= ~(1u << 1);
  546. State[7] = (State[7] & ~(3u << 2)) | ((gSetting_battery_text & 3u) << 2);
  547. #ifdef ENABLE_AUDIO_BAR
  548. if (!gSetting_mic_bar) State[7] &= ~(1u << 4);
  549. #endif
  550. //if (!gSetting_AM_fix) State[7] &= ~(1u << 5);
  551. State[7] = (State[7] & ~(3u << 6)) | ((gSetting_backlight_on_tx_rx & 3u) << 6);
  552. EEPROM_WriteBuffer(0x0F40, State);
  553. }
  554. void SETTINGS_SaveChannel(uint8_t Channel, uint8_t VFO, const VFO_Info_t *pVFO, uint8_t Mode)
  555. {
  556. #ifdef ENABLE_NOAA
  557. if (IS_NOAA_CHANNEL(Channel))
  558. return;
  559. #endif
  560. uint16_t OffsetVFO = Channel * 16;
  561. if (IS_FREQ_CHANNEL(Channel)) { // it's a VFO, not a channel
  562. OffsetVFO = (VFO == 0) ? 0x0C80 : 0x0C90;
  563. OffsetVFO += (Channel - FREQ_CHANNEL_FIRST) * 32;
  564. }
  565. if (Mode >= 2 || IS_FREQ_CHANNEL(Channel)) { // copy VFO to a channel
  566. union {
  567. uint8_t _8[8];
  568. uint32_t _32[2];
  569. } State;
  570. State._32[0] = pVFO->freq_config_RX.Frequency;
  571. State._32[1] = pVFO->TX_OFFSET_FREQUENCY;
  572. EEPROM_WriteBuffer(OffsetVFO + 0, State._32);
  573. State._8[0] = pVFO->freq_config_RX.Code;
  574. State._8[1] = pVFO->freq_config_TX.Code;
  575. State._8[2] = (pVFO->freq_config_TX.CodeType << 4) | pVFO->freq_config_RX.CodeType;
  576. State._8[3] = (pVFO->Modulation << 4) | pVFO->TX_OFFSET_FREQUENCY_DIRECTION;
  577. State._8[4] = 0
  578. | (pVFO->BUSY_CHANNEL_LOCK << 4)
  579. | (pVFO->OUTPUT_POWER << 2)
  580. | (pVFO->CHANNEL_BANDWIDTH << 1)
  581. | (pVFO->FrequencyReverse << 0);
  582. State._8[5] = ((pVFO->DTMF_PTT_ID_TX_MODE & 7u) << 1)
  583. #ifdef ENABLE_DTMF_CALLING
  584. | ((pVFO->DTMF_DECODING_ENABLE & 1u) << 0)
  585. #endif
  586. ;
  587. State._8[6] = pVFO->STEP_SETTING;
  588. State._8[7] = pVFO->SCRAMBLING_TYPE;
  589. EEPROM_WriteBuffer(OffsetVFO + 8, State._8);
  590. SETTINGS_UpdateChannel(Channel, pVFO, true);
  591. if (IS_MR_CHANNEL(Channel)) {
  592. #ifndef ENABLE_KEEP_MEM_NAME
  593. // clear/reset the channel name
  594. SETTINGS_SaveChannelName(Channel, "");
  595. #else
  596. if (Mode >= 3) {
  597. SETTINGS_SaveChannelName(Channel, pVFO->Name);
  598. }
  599. #endif
  600. }
  601. }
  602. }
  603. void SETTINGS_SaveBatteryCalibration(const uint16_t * batteryCalibration)
  604. {
  605. uint16_t buf[4];
  606. EEPROM_WriteBuffer(0x1F40, batteryCalibration);
  607. EEPROM_ReadBuffer( 0x1F48, buf, sizeof(buf));
  608. buf[0] = batteryCalibration[4];
  609. buf[1] = batteryCalibration[5];
  610. EEPROM_WriteBuffer(0x1F48, buf);
  611. }
  612. void SETTINGS_SaveChannelName(uint8_t channel, const char * name)
  613. {
  614. uint16_t offset = channel * 16;
  615. uint8_t buf[16] = {0};
  616. memcpy(buf, name, MIN(strlen(name), 10u));
  617. EEPROM_WriteBuffer(0x0F50 + offset, buf);
  618. EEPROM_WriteBuffer(0x0F58 + offset, buf + 8);
  619. }
  620. void SETTINGS_UpdateChannel(uint8_t channel, const VFO_Info_t *pVFO, bool keep)
  621. {
  622. #ifdef ENABLE_NOAA
  623. if (!IS_NOAA_CHANNEL(channel))
  624. #endif
  625. {
  626. uint8_t state[8];
  627. ChannelAttributes_t att = {
  628. .band = 0xf,
  629. .compander = 0,
  630. .scanlist1 = 0,
  631. .scanlist2 = 0,
  632. }; // default attributes
  633. uint16_t offset = 0x0D60 + (channel & ~7u);
  634. EEPROM_ReadBuffer(offset, state, sizeof(state));
  635. if (keep) {
  636. att.band = pVFO->Band;
  637. att.scanlist1 = pVFO->SCANLIST1_PARTICIPATION;
  638. att.scanlist2 = pVFO->SCANLIST2_PARTICIPATION;
  639. att.compander = pVFO->Compander;
  640. if (state[channel & 7u] == att.__val)
  641. return; // no change in the attributes
  642. }
  643. state[channel & 7u] = att.__val;
  644. EEPROM_WriteBuffer(offset, state);
  645. gMR_ChannelAttributes[channel] = att;
  646. if (IS_MR_CHANNEL(channel)) { // it's a memory channel
  647. if (!keep) {
  648. // clear/reset the channel name
  649. SETTINGS_SaveChannelName(channel, "");
  650. }
  651. }
  652. }
  653. }
  654. void SETTINGS_WriteBuildOptions(void)
  655. {
  656. uint8_t buf[8] = {0};
  657. buf[0] = 0
  658. #ifdef ENABLE_FMRADIO
  659. | (1 << 0)
  660. #endif
  661. #ifdef ENABLE_NOAA
  662. | (1 << 1)
  663. #endif
  664. #ifdef ENABLE_VOICE
  665. | (1 << 2)
  666. #endif
  667. #ifdef ENABLE_VOX
  668. | (1 << 3)
  669. #endif
  670. #ifdef ENABLE_ALARM
  671. | (1 << 4)
  672. #endif
  673. #ifdef ENABLE_TX1750
  674. | (1 << 5)
  675. #endif
  676. #ifdef ENABLE_PWRON_PASSWORD
  677. | (1 << 6)
  678. #endif
  679. #ifdef ENABLE_DTMF_CALLING
  680. | (1 << 7)
  681. #endif
  682. ;
  683. buf[1] = 0
  684. #ifdef ENABLE_FLASHLIGHT
  685. | (1 << 0)
  686. #endif
  687. #ifdef ENABLE_WIDE_RX
  688. | (1 << 1)
  689. #endif
  690. #ifdef ENABLE_BYP_RAW_DEMODULATORS
  691. | (1 << 2)
  692. #endif
  693. #ifdef ENABLE_BLMIN_TMP_OFF
  694. | (1 << 3)
  695. #endif
  696. #ifdef ENABLE_SPECTRUM
  697. | (1 << 4)
  698. #endif
  699. ;
  700. EEPROM_WriteBuffer(0x1FF0, buf);
  701. }