0
mirror of https://github.com/OneOfEleven/uv-k5-firmware-custom.git synced 2025-04-28 14:21:25 +03:00

291 lines
6.7 KiB
C
Raw Normal View History

2023-09-09 08:03:56 +01: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.
*/
#include "app/aircopy.h"
#include "audio.h"
#include "driver/bk4819.h"
#include "driver/crc.h"
#include "driver/eeprom.h"
#include "frequencies.h"
#include "misc.h"
#include "radio.h"
2023-10-12 12:34:01 +01:00
#include "settings.h"
2023-09-09 08:03:56 +01:00
#include "ui/helper.h"
#include "ui/inputbox.h"
#include "ui/ui.h"
2023-10-12 14:55:10 +01:00
#define AIRCOPY_MAGIC_START 0xABCD
#define AIRCOPY_MAGIC_END 0xDCBA
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
#define AIRCOPY_LAST_EEPROM_ADDR 0x1E00
static const uint16_t Obfuscation[] = {
0x6C16, 0xE614, 0x912E, 0x400D, 0x3521, 0x40D5, 0x0313, 0x80E9
};
const uint8_t g_aircopy_block_max = 120;
uint8_t g_aircopy_block_number;
uint8_t g_aircopy_rx_errors;
2023-10-12 13:24:02 +01:00
aircopy_state_t g_aircopy_state;
2023-10-12 14:55:10 +01:00
uint16_t g_aircopy_fsk_buffer[36];
uint8_t g_aircopy_send_count_down_10ms;
unsigned int g_aircopy_fsk_write_index;
2023-09-09 08:03:56 +01:00
void AIRCOPY_SendMessage(const uint8_t request_packet)
2023-09-09 08:03:56 +01:00
{
2023-10-12 14:55:10 +01:00
unsigned int i;
const uint16_t eeprom_addr = (uint16_t)g_aircopy_block_number * 64;
// will be used to ask the TX/ing radio to resend a missing/corrupted packet
(void)request_packet;
2023-10-12 14:55:10 +01:00
// *********
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
// packet start
g_aircopy_fsk_buffer[0] = AIRCOPY_MAGIC_START;
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
// eeprom address
g_aircopy_fsk_buffer[1] = eeprom_addr;
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
// data
EEPROM_ReadBuffer(eeprom_addr, &g_aircopy_fsk_buffer[2], 64);
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
// data CRC
g_aircopy_fsk_buffer[34] = CRC_Calculate(&g_aircopy_fsk_buffer[1], 2 + 64);
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
// packet end
g_aircopy_fsk_buffer[35] = AIRCOPY_MAGIC_END;
// *********
// scramble the packet
for (i = 0; i < 34; i++)
g_aircopy_fsk_buffer[1 + i] ^= Obfuscation[i % ARRAY_SIZE(Obfuscation)];
2023-09-09 08:03:56 +01:00
// TX the packet
2023-09-09 08:03:56 +01:00
RADIO_SetTxParameters();
BK4819_SetupPowerAmplifier(0, g_current_vfo->pTX->frequency); // VERY low TX power
2023-10-12 14:55:10 +01:00
BK4819_SendFSKData(g_aircopy_fsk_buffer);
2023-09-09 08:03:56 +01:00
BK4819_SetupPowerAmplifier(0, 0);
BK4819_set_GPIO_pin(BK4819_GPIO5_PIN1, false);
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
if (++g_aircopy_block_number >= g_aircopy_block_max)
{
g_aircopy_state = AIRCOPY_TX_COMPLETE;
g_update_display = true;
}
// TX pause/gap time
#if 0
g_aircopy_send_count_down_10ms = 300 / 10; // 300ms
#else
g_aircopy_send_count_down_10ms = 30 / 10; // 30ms
#endif
2023-09-09 08:03:56 +01:00
}
void AIRCOPY_StorePacket(void)
{
uint16_t Status;
2023-10-12 14:55:10 +01:00
if (g_aircopy_fsk_write_index < ARRAY_SIZE(g_aircopy_fsk_buffer))
2023-09-09 08:03:56 +01:00
return;
2023-10-12 14:55:10 +01:00
g_aircopy_fsk_write_index = 0;
g_update_display = true;
Status = BK4819_ReadRegister(BK4819_REG_0B);
2023-09-09 08:03:56 +01:00
BK4819_PrepareFSKReceive();
2023-10-12 14:55:10 +01:00
// Doc says bit 4 should be 1 = CRC OK, 0 = CRC FAIL, but original firmware checks for FAIL
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
if ((Status & (1u << 4)) == 0 && g_aircopy_fsk_buffer[0] == AIRCOPY_MAGIC_START && g_aircopy_fsk_buffer[35] == AIRCOPY_MAGIC_END)
2023-09-09 08:03:56 +01:00
{
uint16_t CRC;
unsigned int i;
for (i = 0; i < 34; i++)
2023-10-12 14:55:10 +01:00
g_aircopy_fsk_buffer[1 + i] ^= Obfuscation[i % ARRAY_SIZE(Obfuscation)];
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
CRC = CRC_Calculate(&g_aircopy_fsk_buffer[1], 2 + 64);
2023-10-12 14:55:10 +01:00
if (g_aircopy_fsk_buffer[34] == CRC)
2023-09-09 08:03:56 +01:00
{
2023-10-12 14:55:10 +01:00
uint16_t eeprom_addr = g_aircopy_fsk_buffer[1];
if (eeprom_addr < AIRCOPY_LAST_EEPROM_ADDR)
2023-09-09 08:03:56 +01:00
{
2023-10-12 14:55:10 +01:00
const uint16_t *pData = &g_aircopy_fsk_buffer[2];
2023-09-09 08:03:56 +01:00
for (i = 0; i < 8; i++)
{
2023-10-12 14:55:10 +01:00
EEPROM_WriteBuffer(eeprom_addr, pData);
pData += 4;
eeprom_addr += 8;
2023-09-09 08:03:56 +01:00
}
2023-10-08 20:23:37 +01:00
2023-10-12 14:55:10 +01:00
//g_aircopy_block_number++;
g_aircopy_block_number = eeprom_addr / 64;
if (eeprom_addr >= AIRCOPY_LAST_EEPROM_ADDR)
{ // reached end of eeprom config area
2023-10-12 12:58:58 +01:00
g_aircopy_state = AIRCOPY_RX_COMPLETE;
2023-10-12 12:34:01 +01:00
g_update_display = true;
}
2023-09-09 08:03:56 +01:00
return;
}
}
}
2023-10-12 12:34:01 +01:00
2023-10-12 14:55:10 +01:00
g_aircopy_rx_errors++;
2023-09-09 08:03:56 +01:00
}
2023-10-08 20:23:37 +01:00
static void AIRCOPY_Key_DIGITS(key_code_t Key, bool key_pressed, bool key_held)
2023-09-09 08:03:56 +01:00
{
2023-10-08 20:23:37 +01:00
if (!key_held && key_pressed)
2023-09-09 08:03:56 +01:00
{
uint32_t Frequency;
unsigned int i;
INPUTBOX_Append(Key);
2023-10-08 20:23:37 +01:00
g_request_display_screen = DISPLAY_AIRCOPY;
2023-10-08 20:23:37 +01:00
if (g_input_box_index < 6)
2023-09-09 08:03:56 +01:00
{
2023-09-14 09:56:30 +01:00
#ifdef ENABLE_VOICE
2023-10-08 17:14:13 +01:00
g_another_voice_id = (voice_id_t)Key;
2023-09-09 08:03:56 +01:00
#endif
return;
}
2023-10-08 20:23:37 +01:00
g_input_box_index = 0;
2023-09-09 08:03:56 +01:00
2023-10-08 20:23:37 +01:00
NUMBER_Get(g_input_box, &Frequency);
2023-09-09 08:03:56 +01:00
2023-10-08 20:23:37 +01:00
for (i = 0; i < ARRAY_SIZE(FREQ_BAND_TABLE); i++)
2023-09-09 08:03:56 +01:00
{
2023-10-08 20:23:37 +01:00
if (Frequency >= FREQ_BAND_TABLE[i].lower && Frequency < FREQ_BAND_TABLE[i].upper)
2023-09-09 08:03:56 +01:00
{
2023-09-14 09:56:30 +01:00
#ifdef ENABLE_VOICE
2023-10-12 12:34:01 +01:00
g_another_voice_id = (voice_id_t)Key;
2023-09-09 08:03:56 +01:00
#endif
2023-10-12 12:34:01 +01:00
g_rx_vfo->band = i;
// round the frequency to nearest step size
Frequency = ((Frequency + (g_rx_vfo->step_freq / 2)) / g_rx_vfo->step_freq) * g_rx_vfo->step_freq;
2023-10-12 14:55:10 +01:00
g_aircopy_freq = Frequency;
2023-10-12 12:34:01 +01:00
#ifdef ENABLE_AIRCOPY_FREQ
SETTINGS_SaveSettings(); // remeber the frequency for the next time
#endif
2023-10-08 20:23:37 +01:00
g_rx_vfo->freq_config_rx.frequency = Frequency;
g_rx_vfo->freq_config_tx.frequency = Frequency;
RADIO_ConfigureSquelchAndOutputPower(g_rx_vfo);
2023-10-12 12:34:01 +01:00
g_current_vfo = g_rx_vfo;
2023-09-09 08:03:56 +01:00
RADIO_SetupRegisters(true);
BK4819_SetupAircopy();
BK4819_ResetFSK();
2023-10-12 12:34:01 +01:00
2023-09-09 08:03:56 +01:00
return;
}
}
2023-10-08 20:23:37 +01:00
g_request_display_screen = DISPLAY_AIRCOPY;
2023-09-09 08:03:56 +01:00
}
}
2023-10-08 20:23:37 +01:00
static void AIRCOPY_Key_EXIT(bool key_pressed, bool key_held)
2023-09-09 08:03:56 +01:00
{
2023-10-08 20:23:37 +01:00
if (!key_held && key_pressed)
2023-09-09 08:03:56 +01:00
{
2023-10-12 12:34:01 +01:00
if (g_input_box_index > 0)
{ // entering a new frequency to use
g_input_box[--g_input_box_index] = 10;
}
else
{ // enter RX mode
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
g_aircopy_state = AIRCOPY_RX;
g_update_display = true;
2023-10-12 12:34:01 +01:00
GUI_DisplayScreen();
2023-09-09 08:03:56 +01:00
2023-10-12 14:55:10 +01:00
g_aircopy_fsk_write_index = 0;
g_aircopy_block_number = 0;
g_aircopy_rx_errors = 0;
g_input_box_index = 0;
2023-10-12 12:34:01 +01:00
BK4819_PrepareFSKReceive();
2023-09-09 08:03:56 +01:00
}
2023-10-08 20:23:37 +01:00
g_request_display_screen = DISPLAY_AIRCOPY;
2023-09-09 08:03:56 +01:00
}
}
2023-10-08 20:23:37 +01:00
static void AIRCOPY_Key_MENU(bool key_pressed, bool key_held)
2023-09-09 08:03:56 +01:00
{
2023-10-08 20:23:37 +01:00
if (!key_held && key_pressed)
2023-10-12 12:34:01 +01:00
{ // enter TX mode
2023-10-12 14:55:10 +01:00
g_aircopy_state = AIRCOPY_TX;
g_update_display = true;
2023-10-12 12:34:01 +01:00
GUI_DisplayScreen();
2023-10-12 14:55:10 +01:00
g_input_box_index = 0;
g_aircopy_fsk_write_index = 0;
g_aircopy_block_number = 0;
2023-09-09 08:03:56 +01:00
// send initial packet
//AIRCOPY_SendMessage(0xff);
g_aircopy_send_count_down_10ms = 30 / 10; // 30ms
2023-09-09 08:03:56 +01:00
}
}
2023-10-08 20:23:37 +01:00
void AIRCOPY_ProcessKeys(key_code_t Key, bool key_pressed, bool key_held)
2023-09-09 08:03:56 +01:00
{
switch (Key)
{
case KEY_0:
case KEY_1:
case KEY_2:
case KEY_3:
case KEY_4:
case KEY_5:
case KEY_6:
case KEY_7:
case KEY_8:
case KEY_9:
2023-10-08 20:23:37 +01:00
AIRCOPY_Key_DIGITS(Key, key_pressed, key_held);
2023-09-09 08:03:56 +01:00
break;
case KEY_MENU:
2023-10-08 20:23:37 +01:00
AIRCOPY_Key_MENU(key_pressed, key_held);
2023-09-09 08:03:56 +01:00
break;
case KEY_EXIT:
2023-10-08 20:23:37 +01:00
AIRCOPY_Key_EXIT(key_pressed, key_held);
2023-09-09 08:03:56 +01:00
break;
default:
break;
}
}