AP_VideoTX: add lookup tables for VTX power settings

correct settings when power set is received
add support for capturing all supported power levels
learn power levels in SmartAudio 2.1
add better support for VTX power levels
don't set power to 0 if in pitmode
add option for iNav compatibility
support non-conforming SmartAudio implementations
re-enable pitmode on SmartAudio 2.0
add support for "blind" VTX setting
This commit is contained in:
Andy Piper 2021-12-12 16:35:10 +00:00 committed by Andrew Tridgell
parent 0742fec2e4
commit f98ce3dc9e
4 changed files with 348 additions and 222 deletions

View File

@ -51,7 +51,7 @@ bool AP_SmartAudio::init()
_port = AP::serialmanager().find_serial(AP_SerialManager::SerialProtocol_SmartAudio, 0); _port = AP::serialmanager().find_serial(AP_SerialManager::SerialProtocol_SmartAudio, 0);
if (_port!=nullptr) { if (_port!=nullptr) {
_port->configure_parity(0); _port->configure_parity(0);
_port->set_stop_bits(2); _port->set_stop_bits(AP::vtx().has_option(AP_VideoTX::VideoOptions::VTX_SA_ONE_STOP_BIT) ? 1 : 2);
_port->set_flow_control(AP_HAL::UARTDriver::FLOW_CONTROL_DISABLE); _port->set_flow_control(AP_HAL::UARTDriver::FLOW_CONTROL_DISABLE);
_port->set_options((_port->get_options() & ~AP_HAL::UARTDriver::OPTION_RXINV) _port->set_options((_port->get_options() & ~AP_HAL::UARTDriver::OPTION_RXINV)
| AP_HAL::UARTDriver::OPTION_HDPLEX | AP_HAL::UARTDriver::OPTION_PULLDOWN_TX | AP_HAL::UARTDriver::OPTION_PULLDOWN_RX); | AP_HAL::UARTDriver::OPTION_HDPLEX | AP_HAL::UARTDriver::OPTION_PULLDOWN_TX | AP_HAL::UARTDriver::OPTION_PULLDOWN_RX);
@ -123,9 +123,13 @@ void AP_SmartAudio::loop()
// and the initial response is 40ms long so we should wait at least 140ms before giving up // and the initial response is 40ms long so we should wait at least 140ms before giving up
if (now - _last_request_sent_ms < 200 && _is_waiting_response) { if (now - _last_request_sent_ms < 200 && _is_waiting_response) {
// setup sheduler delay to 50 ms again after response processes // setup scheduler delay to 50 ms again after response processes
if (!read_response(_response_buffer)) { if (!read_response(_response_buffer)) {
hal.scheduler->delay(10); hal.scheduler->delay(10);
} else {
// successful response, wait another 100ms to give the VTX a chance to recover
// before sending another command. This is needed on the Atlatl v1.
hal.scheduler->delay(100);
} }
} else if (_is_waiting_response) { // timeout } else if (_is_waiting_response) { // timeout
@ -134,13 +138,14 @@ void AP_SmartAudio::loop()
_port->discard_input(); _port->discard_input();
_inline_buffer_length = 0; _inline_buffer_length = 0;
_is_waiting_response = false; _is_waiting_response = false;
} else { debug("response timeout");
} else if (_initialised) {
if (AP::vtx().have_params_changed() ||_vtx_power_change_pending if (AP::vtx().have_params_changed() ||_vtx_power_change_pending
|| _vtx_freq_change_pending || _vtx_options_change_pending) { || _vtx_freq_change_pending || _vtx_options_change_pending) {
update_vtx_params(); update_vtx_params();
set_configuration_pending(true); set_configuration_pending(true);
vtx.set_configuration_finished(false); vtx.set_configuration_finished(false);
// we've tried to udpate something, re-request the settings so that they // we've tried to update something, re-request the settings so that they
// are reflected correctly // are reflected correctly
request_settings(); request_settings();
} else if (is_configuration_pending()) { } else if (is_configuration_pending()) {
@ -176,22 +181,20 @@ void AP_SmartAudio::update_vtx_params()
vtx.get_channel(), vtx.get_configured_channel(), vtx.get_channel(), vtx.get_configured_channel(),
vtx.get_band(), vtx.get_configured_band(), vtx.get_band(), vtx.get_configured_band(),
vtx.get_power_mw(), vtx.get_configured_power_mw(), vtx.get_power_mw(), vtx.get_configured_power_mw(),
vtx.get_options(), vtx.get_configured_options()); vtx.get_options() & 0xF, vtx.get_configured_options() & 0xF);
uint8_t opts = vtx.get_configured_options(); uint8_t opts = vtx.get_configured_options();
uint8_t pitModeRunning = (vtx.get_options() & uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE)); uint8_t pitMode = vtx.get_configured_pitmode();
uint8_t pitMode = opts & uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE);
uint8_t mode; uint8_t mode;
// check if we are turning pitmode on or off, but only on SA 2.1 as older versions // check if we are turning pitmode on or off, but only on SA 2.0+
// appear not to work properly if (pitMode != vtx.get_pitmode() && _protocol_version >= SMARTAUDIO_SPEC_PROTOCOL_v2) {
if (pitMode != pitModeRunning && _protocol_version >= SMARTAUDIO_SPEC_PROTOCOL_v21) { if (vtx.get_pitmode()) {
if (pitModeRunning) {
debug("Turning OFF pitmode"); debug("Turning OFF pitmode");
// turn it off // turn it off
mode = 0x04 | ((opts & uint8_t(AP_VideoTX::VideoOptions::VTX_UNLOCKED)) << 2); mode = 0x04 | ((opts & uint8_t(AP_VideoTX::VideoOptions::VTX_UNLOCKED)) << 2);
} else { } else {
debug("Turning ON pitmode"); debug("Turning ON pitmode");
// turn it on // turn it on (in range pitmode flag)
mode = 0x01 | ((opts & uint8_t(AP_VideoTX::VideoOptions::VTX_UNLOCKED)) << 2); mode = 0x01 | ((opts & uint8_t(AP_VideoTX::VideoOptions::VTX_UNLOCKED)) << 2);
} }
} else { } else {
@ -218,7 +221,7 @@ void AP_SmartAudio::update_vtx_params()
set_channel(vtx.get_configured_band() * VTX_MAX_CHANNELS + vtx.get_configured_channel()); set_channel(vtx.get_configured_band() * VTX_MAX_CHANNELS + vtx.get_configured_channel());
} }
} else if (_vtx_power_change_pending) { } else if (_vtx_power_change_pending) {
debug("update power"); debug("update power (ver %u)", _protocol_version);
switch (_protocol_version) { switch (_protocol_version) {
case SMARTAUDIO_SPEC_PROTOCOL_v21: case SMARTAUDIO_SPEC_PROTOCOL_v21:
set_power(vtx.get_configured_power_dbm() | 0x80); set_power(vtx.get_configured_power_dbm() | 0x80);
@ -264,7 +267,7 @@ void AP_SmartAudio::send_request(const Frame& requestFrame, uint8_t size)
_packets_sent++; _packets_sent++;
#ifdef SA_DEBUG #ifdef SA_DEBUG
print_bytes_to_hex_string("send_request():", request, size,0); print_bytes_to_hex_string("send_request():", request, size);
#endif #endif
} }
@ -316,6 +319,10 @@ bool AP_SmartAudio::read_response(uint8_t *response_buffer)
FrameHeader* header = (FrameHeader*)response_buffer; FrameHeader* header = (FrameHeader*)response_buffer;
incoming_bytes_count -= response_header_size; incoming_bytes_count -= response_header_size;
// implementations that ignore the CRC also appear to not account for it in the frame length
if (ignore_crc()) {
header->length++;
}
_packet_size = header->length; _packet_size = header->length;
} }
@ -339,12 +346,12 @@ bool AP_SmartAudio::read_response(uint8_t *response_buffer)
} }
#ifdef SA_DEBUG #ifdef SA_DEBUG
print_bytes_to_hex_string("read_response():", response_buffer, incoming_bytes_count,(_inline_buffer_length-incoming_bytes_count)>=0?(_inline_buffer_length-incoming_bytes_count):0); print_bytes_to_hex_string("read_response():", response_buffer, _inline_buffer_length);
#endif #endif
_is_waiting_response=false; _is_waiting_response = false;
bool correct_parse = parse_response_buffer(response_buffer); bool correct_parse = parse_response_buffer(response_buffer);
response_buffer=nullptr; response_buffer = nullptr;
_inline_buffer_length=0; _inline_buffer_length=0;
_packet_size = 0; _packet_size = 0;
_packets_rcvd++; _packets_rcvd++;
@ -357,6 +364,8 @@ bool AP_SmartAudio::read_response(uint8_t *response_buffer)
void AP_SmartAudio::push_command_only_frame(uint8_t cmd_id) void AP_SmartAudio::push_command_only_frame(uint8_t cmd_id)
{ {
Packet command; Packet command;
// according to the spec the length should include the CRC, but no implementation appears to
// do this
command.frame.header.init(cmd_id, 0); command.frame.header.init(cmd_id, 0);
command.frame_size = SMARTAUDIO_COMMAND_FRAME_SIZE; command.frame_size = SMARTAUDIO_COMMAND_FRAME_SIZE;
command.frame.payload[0] = crc8_dvb_s2_update(0, &command.frame, SMARTAUDIO_COMMAND_FRAME_SIZE - 1); command.frame.payload[0] = crc8_dvb_s2_update(0, &command.frame, SMARTAUDIO_COMMAND_FRAME_SIZE - 1);
@ -451,31 +460,32 @@ void AP_SmartAudio::unpack_frequency(AP_SmartAudio::Settings *settings, const ui
} }
} }
// SmartAudio v1/v2
void AP_SmartAudio::unpack_settings(Settings *settings, const SettingsResponseFrame *frame) void AP_SmartAudio::unpack_settings(Settings *settings, const SettingsResponseFrame *frame)
{ {
settings->channel = frame->channel % VTX_MAX_CHANNELS; settings->channel = frame->channel % VTX_MAX_CHANNELS;
settings->band = frame->channel / VTX_MAX_CHANNELS; settings->band = frame->channel / VTX_MAX_CHANNELS;
settings->power = frame->power; settings->power = frame->power;
settings->mode = frame->operationMode; settings->mode = frame->operationMode;
settings->num_power_levels = 0;
unpack_frequency(settings, be16toh(frame->frequency)); unpack_frequency(settings, be16toh(frame->frequency));
} }
// SmartAudio v2.1
void AP_SmartAudio::unpack_settings(Settings *settings, const SettingsExtendedResponseFrame *frame) void AP_SmartAudio::unpack_settings(Settings *settings, const SettingsExtendedResponseFrame *frame)
{ {
settings->channel = frame->channel % VTX_MAX_CHANNELS; unpack_settings(settings, &frame->settings);
settings->band = frame->channel / VTX_MAX_CHANNELS; settings->power_in_dbm = frame->power_dbm;
settings->power = frame->power; settings->num_power_levels = frame->num_power_levels + 1;
settings->power_in_dbm=frame->power_dbm; memcpy(settings->power_levels, frame->power_levels, frame->num_power_levels + 1);
settings->mode = frame->operationMode;
unpack_frequency(settings, be16toh(frame->frequency));
} }
#ifdef SA_DEBUG #ifdef SA_DEBUG
void AP_SmartAudio::print_bytes_to_hex_string(const char* msg, const uint8_t buf[], uint8_t len,uint8_t offset) void AP_SmartAudio::print_bytes_to_hex_string(const char* msg, const uint8_t buf[], uint8_t len)
{ {
hal.console->printf("SA: %s ", msg); hal.console->printf("SA: %s ", msg);
for (uint8_t i = 0; i < len; i++) { for (uint8_t i = 0; i < len; i++) {
hal.console->printf("0x%02X ", buf[i+offset]); hal.console->printf("0x%02X ", buf[i]);
} }
hal.console->printf("\n"); hal.console->printf("\n");
} }
@ -485,11 +495,11 @@ void AP_SmartAudio::print_settings(const Settings* settings)
{ {
debug("SETTINGS: VER: %u, MD: '%c%c%c%c%c', CH: %u, PWR: %u, DBM: %u FREQ: %u, BND: %u", debug("SETTINGS: VER: %u, MD: '%c%c%c%c%c', CH: %u, PWR: %u, DBM: %u FREQ: %u, BND: %u",
settings->version, settings->version,
(settings->mode & 0x10) ? 'U' : 'L', (settings->mode & 0x10) ? 'U' : 'L',// (L)ocked or (U)nlocked
(settings->mode & 0x8) ? 'O' : ' ', (settings->mode & 0x8) ? 'O' : ' ', // (O)ut-range pitmode
(settings->mode & 0x4) ? 'I' : ' ', (settings->mode & 0x4) ? 'I' : ' ', // (I)n-range pitmode
(settings->mode & 0x2) ? 'P' : ' ', (settings->mode & 0x2) ? 'P' : ' ', // (P)itmode running
(settings->mode & 0x1) ? 'F' : 'C', (settings->mode & 0x1) ? 'F' : 'C', // Set (F)requency or (C)hannel
settings->channel, settings->power, settings->power_in_dbm, settings->frequency, settings->band); settings->channel, settings->power, settings->power_in_dbm, settings->frequency, settings->band);
} }
@ -501,10 +511,13 @@ void AP_SmartAudio::update_vtx_settings(const Settings& settings)
vtx.set_frequency_mhz(settings.frequency); vtx.set_frequency_mhz(settings.frequency);
vtx.set_band(settings.band); vtx.set_band(settings.band);
vtx.set_channel(settings.channel); vtx.set_channel(settings.channel);
// SA21 sends us a complete packet with the supported power levels
if (settings.version == SMARTAUDIO_SPEC_PROTOCOL_v21) { if (settings.version == SMARTAUDIO_SPEC_PROTOCOL_v21) {
vtx.set_power_dbm(settings.power_in_dbm); vtx.set_power_dbm(settings.power_in_dbm);
// learn them all
vtx.update_all_power_dbm(settings.num_power_levels, settings.power_levels);
} else { } else {
vtx.set_power_level(settings.power); vtx.set_power_level(settings.power, AP_VideoTX::PowerActive::Active);
} }
// it seems like the spec is wrong, on a unify pro32 this setting is inverted // it seems like the spec is wrong, on a unify pro32 this setting is inverted
_vtx_use_set_freq = !(settings.mode & 1); _vtx_use_set_freq = !(settings.mode & 1);
@ -530,9 +543,10 @@ bool AP_SmartAudio::parse_response_buffer(const uint8_t *buffer)
const uint8_t *startPtr = buffer + 2; const uint8_t *startPtr = buffer + 2;
const uint8_t *endPtr = buffer + headerPayloadLength; const uint8_t *endPtr = buffer + headerPayloadLength;
if (crc8_dvb_s2_update(0x00, startPtr, headerPayloadLength-2)!=*(endPtr) if ((crc8_dvb_s2_update(0x00, startPtr, headerPayloadLength-2)!=*(endPtr) && !ignore_crc())
|| header->headerByte != SMARTAUDIO_HEADER_BYTE || header->headerByte != SMARTAUDIO_HEADER_BYTE
|| header->syncByte!=SMARTAUDIO_SYNC_BYTE) { || header->syncByte != SMARTAUDIO_SYNC_BYTE) {
debug("parse_response_buffer() failed - invalid CRC or header");
return false; return false;
} }
// SEND TO GCS A MESSAGE TO UNDERSTAND WHATS HAPPENING // SEND TO GCS A MESSAGE TO UNDERSTAND WHATS HAPPENING
@ -590,20 +604,24 @@ bool AP_SmartAudio::parse_response_buffer(const uint8_t *buffer)
const uint8_t power = resp->payload & 0xFF; const uint8_t power = resp->payload & 0xFF;
switch (_protocol_version) { switch (_protocol_version) {
case SMARTAUDIO_SPEC_PROTOCOL_v21: case SMARTAUDIO_SPEC_PROTOCOL_v21:
if (vtx.get_configured_power_dbm() != power) {
vtx.update_power_dbm(vtx.get_configured_power_dbm(), AP_VideoTX::PowerActive::Inactive);
}
vtx.set_power_dbm(power); vtx.set_power_dbm(power);
vtx.set_configured_power_mw(vtx.get_power_mw()); vtx.set_configured_power_mw(vtx.get_power_mw());
break; break;
case SMARTAUDIO_SPEC_PROTOCOL_v2: case SMARTAUDIO_SPEC_PROTOCOL_v2:
if (vtx.get_configured_power_level() != power) {
vtx.update_power_dbm(vtx.get_configured_power_dbm(), AP_VideoTX::PowerActive::Inactive);
}
vtx.set_power_level(power); vtx.set_power_level(power);
vtx.set_configured_power_mw(vtx.get_power_mw()); vtx.set_configured_power_mw(vtx.get_power_mw());
break; break;
default: default:
switch(power) { if (vtx.get_configured_power_dac() != power) {
case 16: vtx.set_power_level(1); break; // 200mw vtx.update_power_dbm(vtx.get_configured_power_dbm(), AP_VideoTX::PowerActive::Inactive);
case 25: vtx.set_power_level(2); break; // 500mw
case 40: vtx.set_power_level(3); break; // 800mw
default: vtx.set_power_level(0); break; // 25mw
} }
vtx.set_power_dac(power);
vtx.set_configured_power_mw(vtx.get_power_mw()); vtx.set_configured_power_mw(vtx.get_power_mw());
break; break;
} }

View File

@ -66,7 +66,7 @@
#define SMARTAUDIO_BANDCHAN_TO_INDEX(band, channel) (band * VTX_MAX_CHANNELS + (channel)) #define SMARTAUDIO_BANDCHAN_TO_INDEX(band, channel) (band * VTX_MAX_CHANNELS + (channel))
// #define SA_DEBUG //#define SA_DEBUG
class AP_SmartAudio class AP_SmartAudio
{ {
@ -85,7 +85,8 @@ public:
uint16_t frequency; uint16_t frequency;
uint8_t band; uint8_t band;
uint8_t* power_levels; uint8_t num_power_levels;
uint8_t power_levels[8];
uint8_t power_in_dbm; uint8_t power_in_dbm;
uint16_t pitmodeFrequency; uint16_t pitmodeFrequency;
@ -136,14 +137,10 @@ public:
} PACKED; } PACKED;
struct SettingsExtendedResponseFrame { struct SettingsExtendedResponseFrame {
FrameHeader header; SettingsResponseFrame settings;
uint8_t channel; uint8_t power_dbm; // current power
uint8_t power; uint8_t num_power_levels;
uint8_t operationMode; uint8_t power_levels[8]; // first in the list of dbm levels
uint16_t frequency;
uint8_t power_dbm;
uint8_t power_levels_len;
uint8_t power_dbm_levels; // first in the list of dbm levels
//uint8_t crc; //uint8_t crc;
} PACKED; } PACKED;
@ -217,13 +214,15 @@ private:
#ifdef SA_DEBUG #ifdef SA_DEBUG
// utility method for debugging. // utility method for debugging.
void print_bytes_to_hex_string(const char* msg, const uint8_t buf[], uint8_t x,uint8_t offset); void print_bytes_to_hex_string(const char* msg, const uint8_t buf[], uint8_t length);
#endif #endif
void print_settings(const Settings* settings); void print_settings(const Settings* settings);
void update_vtx_params(); void update_vtx_params();
void update_vtx_settings(const Settings& settings); void update_vtx_settings(const Settings& settings);
bool ignore_crc() const { return AP::vtx().has_option(AP_VideoTX::VideoOptions::VTX_SA_IGNORE_CRC); }
// looping over requests // looping over requests
void loop(); void loop();
// send a frame over the wire // send a frame over the wire

View File

@ -61,9 +61,9 @@ const AP_Param::GroupInfo AP_VideoTX::var_info[] = {
// @Param: OPTIONS // @Param: OPTIONS
// @DisplayName: Video Transmitter Options // @DisplayName: Video Transmitter Options
// @Description: Video Transmitter Options. Pitmode puts the VTX in a low power state. Unlocked enables certain restricted frequencies and power levels. Do not enable the Unlocked option unless you have appropriate permissions in your jurisdiction to transmit at high power levels. // @Description: Video Transmitter Options. Pitmode puts the VTX in a low power state. Unlocked enables certain restricted frequencies and power levels. Do not enable the Unlocked option unless you have appropriate permissions in your jurisdiction to transmit at high power levels. One stop-bit may be required for VTXs that erroneously mimic iNav behaviour.
// @User: Advanced // @User: Advanced
// @Bitmask: 0:Pitmode,1:Pitmode until armed,2:Pitmode when disarmed,3:Unlocked,4:Add leading zero byte to requests // @Bitmask: 0:Pitmode,1:Pitmode until armed,2:Pitmode when disarmed,3:Unlocked,4:Add leading zero byte to requests,5:Use 1 stop-bit in SmartAudio,6:Ignore CRC in SmartAudio,7:Ignore status updates in CRSF and blindly set VTX options
AP_GROUPINFO("OPTIONS", 6, AP_VideoTX, _options, 0), AP_GROUPINFO("OPTIONS", 6, AP_VideoTX, _options, 0),
// @Param: MAX_POWER // @Param: MAX_POWER
@ -75,6 +75,13 @@ const AP_Param::GroupInfo AP_VideoTX::var_info[] = {
AP_GROUPEND AP_GROUPEND
}; };
//#define VTX_DEBUG
#ifdef VTX_DEBUG
# define debug(fmt, args...) hal.console->printf("VTX: " fmt "\n", ##args)
#else
# define debug(fmt, args...) do {} while(0)
#endif
extern const AP_HAL::HAL& hal; extern const AP_HAL::HAL& hal;
const char * AP_VideoTX::band_names[] = {"A","B","E","F","R","L"}; const char * AP_VideoTX::band_names[] = {"A","B","E","F","R","L"};
@ -89,6 +96,23 @@ const uint16_t AP_VideoTX::VIDEO_CHANNELS[AP_VideoTX::MAX_BANDS][VTX_MAX_CHANNEL
{ 5621, 5584, 5547, 5510, 5473, 5436, 5399, 5362} /* LO Race */ { 5621, 5584, 5547, 5510, 5473, 5436, 5399, 5362} /* LO Race */
}; };
// mapping of power level to milliwatt to dbm
// valid power levels from SmartAudio spec, the adjacent levels might be the actual values
// so these are marked as level + 0x10 and will be switched if a dbm message proves it
AP_VideoTX::PowerLevel AP_VideoTX::_power_levels[VTX_MAX_POWER_LEVELS] = {
// level, mw, dbm, dac
{ 0xFF, 0, 0, 0 }, // only in SA 2.1
{ 0, 25, 14, 7 },
{ 0x11, 100, 20, 0xFF }, // only in SA 2.1
{ 1, 200, 23, 16 },
{ 0x12, 400, 26, 0xFF }, // only in SA 2.1
{ 2, 500, 27, 25 },
//{ 0x13, 600, 28, 0xFF },
{ 3, 800, 29, 40 },
{ 0x13, 1000, 30, 0xFF }, // only in SA 2.1
{ 0xFF, 0, 0, 0XFF, PowerActive::Inactive } // slot reserved for a custom power level
};
AP_VideoTX::AP_VideoTX() AP_VideoTX::AP_VideoTX()
{ {
if (singleton) { if (singleton) {
@ -111,7 +135,23 @@ bool AP_VideoTX::init(void)
return false; return false;
} }
_current_power = _power_mw; // PARAMETER_CONVERSION - Added: Sept-2022
_options.convert_parameter_width(AP_PARAM_INT16);
// find the index into the power table
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (_power_mw <= _power_levels[i].mw) {
if (_power_mw != _power_levels[i].mw) {
if (i > 0) {
_current_power = i - 1;
}
_power_mw.set_and_save(get_power_mw());
} else {
_current_power = i;
}
break;
}
}
_current_band = _band; _current_band = _band;
_current_channel = _channel; _current_channel = _channel;
_current_frequency = _frequency_mhz; _current_frequency = _frequency_mhz;
@ -137,91 +177,158 @@ bool AP_VideoTX::get_band_and_channel(uint16_t freq, VideoBand& band, uint8_t& c
} }
// set the current power // set the current power
void AP_VideoTX::set_configured_power_mw(uint16_t power) { void AP_VideoTX::set_configured_power_mw(uint16_t power)
{
_power_mw.set_and_save_ifchanged(power); _power_mw.set_and_save_ifchanged(power);
} }
uint8_t AP_VideoTX::find_current_power() const
{
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (_power_mw == _power_levels[i].mw) {
return i;
}
}
return 0;
}
// set the power in dbm, rounding appropriately // set the power in dbm, rounding appropriately
void AP_VideoTX::set_power_dbm(uint8_t power) { void AP_VideoTX::set_power_dbm(uint8_t power, PowerActive active)
switch (power) { {
case 14: if (power == _power_levels[_current_power].dbm
_current_power = 25; && _power_levels[_current_power].active == active) {
break; return;
case 20: }
_current_power = 100;
break; for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
case 23: if (power == _power_levels[i].dbm) {
_current_power = 200; _current_power = i;
break; _power_levels[i].active = active;
case 26: debug("learned power %ddbm", power);
_current_power = 400; // now unlearn the "other" power level since we have no other way of guessing
break; // the supported levels
case 27: if ((_power_levels[i].level & 0xF0) == 0x10) {
_current_power = 500; _power_levels[i].level = _power_levels[i].level & 0xF;
break; }
case 29: if (i > 0 && _power_levels[i-1].level == _power_levels[i].level) {
_current_power = 800; debug("invalidated power %dwm, level %d is now %dmw", _power_levels[i-1].mw, _power_levels[i].level, _power_levels[i].mw);
break; _power_levels[i-1].level = 0xFF;
default: _power_levels[i-1].active = PowerActive::Inactive;
_current_power = uint16_t(roundf(powf(10, power * 0.1f))); } else if (i < VTX_MAX_POWER_LEVELS-1 && _power_levels[i+1].level == _power_levels[i].level) {
break; debug("invalidated power %dwm, level %d is now %dmw", _power_levels[i+1].mw, _power_levels[i].level, _power_levels[i].mw);
_power_levels[i+1].level = 0xFF;
_power_levels[i+1].active = PowerActive::Inactive;
}
return;
}
}
// learn the non-standard power
_current_power = update_power_dbm(power, active);
}
// add an active power setting in dbm
uint8_t AP_VideoTX::update_power_dbm(uint8_t power, PowerActive active)
{
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (power == _power_levels[i].dbm) {
if (_power_levels[i].active != active) {
_power_levels[i].active = active;
debug("%s power %ddbm", active == PowerActive::Active ? "learned" : "invalidated", power);
}
return i;
}
}
// handed a non-standard value, use the last slot
_power_levels[VTX_MAX_POWER_LEVELS-1].dbm = power;
_power_levels[VTX_MAX_POWER_LEVELS-1].level = 255;
_power_levels[VTX_MAX_POWER_LEVELS-1].dac = 255;
_power_levels[VTX_MAX_POWER_LEVELS-1].mw = uint16_t(roundf(powf(10, power / 10.0f)));
_power_levels[VTX_MAX_POWER_LEVELS-1].active = active;
debug("non-standard power %ddbm -> %dmw", power, _power_levels[VTX_MAX_POWER_LEVELS-1].mw);
return VTX_MAX_POWER_LEVELS-1;
}
// add all active power setting in dbm
void AP_VideoTX::update_all_power_dbm(uint8_t nlevels, const uint8_t power[])
{
for (uint8_t i = 0; i < nlevels; i++) {
update_power_dbm(power[i], PowerActive::Active);
}
// invalidate the remaining ones
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (_power_levels[i].active == PowerActive::Unknown) {
_power_levels[i].active = PowerActive::Inactive;
}
} }
} }
// get the power in dbm, rounding appropriately // set the power in mw
uint8_t AP_VideoTX::get_configured_power_dbm() const { void AP_VideoTX::set_power_mw(uint16_t power)
switch (_power_mw.get()) { {
case 25: return 14; for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
case 100: return 20; if (power == _power_levels[i].mw) {
case 200: return 23; _current_power = i;
case 400: return 26; break;
case 500: return 27; }
case 800: return 29;
default:
return uint8_t(roundf(10.0f * log10f(_power_mw)));
}
}
// get the power "level"
uint8_t AP_VideoTX::get_configured_power_level() const {
if (_power_mw < 26) {
return 0;
} else if (_power_mw < 201) {
return 1;
} else if (_power_mw < 501) {
return 2;
} else { // 800
return 3;
} }
} }
// set the power "level" // set the power "level"
void AP_VideoTX::set_power_level(uint8_t level) { void AP_VideoTX::set_power_level(uint8_t level, PowerActive active)
switch (level) { {
case 1: if (level == _power_levels[_current_power].level
_current_power = 200; && _power_levels[_current_power].active == active) {
break; return;
case 2: }
_current_power = 500;
break; for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
case 3: if (level == _power_levels[i].level) {
_current_power = 800; _current_power = i;
break; _power_levels[i].active = active;
case 0: debug("learned power level %d: %dmw", level, get_power_mw());
default: break;
_current_power = 25; }
break; }
}
// set the power dac
void AP_VideoTX::set_power_dac(uint16_t power, PowerActive active)
{
if (power == _power_levels[_current_power].dac
&& _power_levels[_current_power].active == active) {
return;
}
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (power == _power_levels[i].dac) {
_current_power = i;
_power_levels[i].active = active;
debug("learned power %dmw", get_power_mw());
}
} }
} }
// set the current channel // set the current channel
void AP_VideoTX::set_enabled(bool enabled) { void AP_VideoTX::set_enabled(bool enabled)
{
_current_enabled = enabled; _current_enabled = enabled;
if (!_enabled.configured()) { if (!_enabled.configured()) {
_enabled.set_and_save(enabled); _enabled.set_and_save(enabled);
} }
} }
void AP_VideoTX::set_power_is_current()
{
set_power_dbm(get_configured_power_dbm());
}
void AP_VideoTX::set_freq_is_current()
{
_current_frequency = _frequency_mhz;
_current_band = _band;
_current_channel = _channel;
}
// periodic update // periodic update
void AP_VideoTX::update(void) void AP_VideoTX::update(void)
{ {
@ -241,6 +348,15 @@ void AP_VideoTX::update(void)
_options.set(_options | uint8_t(VideoOptions::VTX_PITMODE)); _options.set(_options | uint8_t(VideoOptions::VTX_PITMODE));
} }
} }
// check that the requested power is actually allowed
// reset if not
if (_power_mw != get_power_mw()) {
if (_power_levels[find_current_power()].active == PowerActive::Inactive) {
// reset to something we know works
debug("power reset to %dmw from %dmw", get_power_mw(), _power_mw.get());
_power_mw.set_and_save(get_power_mw());
}
}
} }
bool AP_VideoTX::update_options() const bool AP_VideoTX::update_options() const
@ -254,6 +370,12 @@ bool AP_VideoTX::update_options() const
return true; return true;
} }
#if HAL_CRSF_TELEM_ENABLED
// using CRSF so unlock is not an option
if (AP::crsf_telem() != nullptr) {
return false;
}
#endif
// check unlock only // check unlock only
if ((_options & uint8_t(VideoOptions::VTX_UNLOCKED)) != 0 if ((_options & uint8_t(VideoOptions::VTX_UNLOCKED)) != 0
&& (_current_options & uint8_t(VideoOptions::VTX_UNLOCKED)) == 0) { && (_current_options & uint8_t(VideoOptions::VTX_UNLOCKED)) == 0) {
@ -264,6 +386,21 @@ bool AP_VideoTX::update_options() const
return false; return false;
} }
bool AP_VideoTX::update_power() const {
if (!_defaults_set || _power_mw == get_power_mw() || get_pitmode()) {
return false;
}
// check that the requested power is actually allowed
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (_power_mw == _power_levels[i].mw
&& _power_levels[i].active != PowerActive::Inactive) {
return true;
}
}
// asked for something unsupported - only SA2.1 allows this and will have already provided a list
return false;
}
bool AP_VideoTX::have_params_changed() const bool AP_VideoTX::have_params_changed() const
{ {
return _enabled return _enabled
@ -301,7 +438,7 @@ bool AP_VideoTX::set_defaults()
return false; return false;
} }
// check that our current view of freqency matches band/channel // check that our current view of frequency matches band/channel
// if not then force one to be correct // if not then force one to be correct
uint16_t calced_freq = get_frequency_mhz(_current_band, _current_channel); uint16_t calced_freq = get_frequency_mhz(_current_band, _current_channel);
if (_current_frequency != calced_freq) { if (_current_frequency != calced_freq) {
@ -326,10 +463,10 @@ bool AP_VideoTX::set_defaults()
_channel.set_and_save(_current_channel); _channel.set_and_save(_current_channel);
} }
if (!_band.configured()) { if (!_band.configured()) {
_frequency_mhz.set_and_save(_current_band); _band.set_and_save(_current_band);
} }
if (!_power_mw.configured()) { if (!_power_mw.configured()) {
_power_mw.set_and_save(_current_power); _power_mw.set_and_save(get_power_mw());
} }
if (!_frequency_mhz.configured()) { if (!_frequency_mhz.configured()) {
_frequency_mhz.set_and_save(_current_frequency); _frequency_mhz.set_and_save(_current_frequency);
@ -366,96 +503,25 @@ void AP_VideoTX::change_power(int8_t position)
if (position < 0 || position > 5) { if (position < 0 || position > 5) {
return; return;
} }
// first find out how many possible levels there are
uint8_t num_active_levels = 0;
for (uint8_t i = 0; i < VTX_MAX_POWER_LEVELS; i++) {
if (_power_levels[i].active != PowerActive::Inactive) {
num_active_levels++;
}
}
// iterate through to find the level
uint16_t level = constrain_int16(roundf((num_active_levels * (position + 1)/ 6.0f) - 1), 0, num_active_levels - 1);
debug("looking for pos %d power level %d from %d", position, level, num_active_levels);
uint16_t power = 0; uint16_t power = 0;
// 0 or 25 for (uint8_t i = 0, j = 0; i < num_active_levels; i++, j++) {
if (_max_power_mw < 100) { while (_power_levels[j].active == PowerActive::Inactive && j < VTX_MAX_POWER_LEVELS-1) {
switch (position) { j++;
case 3:
case 4:
case 5:
power = 25;
break;
default:
power = 0;
break;
} }
} if (i == level) {
// 0, 25 or 100 power = _power_levels[j].mw;
else if (_max_power_mw < 200) { debug("selected power %dmw", power);
switch (position) { break;
case 0:
power = 0;
break;
case 5:
power = 100;
break;
default:
power = 25;
break;
}
}
// 0, 25, 100 or 200
else if (_max_power_mw < 500) {
switch (position) {
case 1:
case 2:
power = 25;
break;
case 3:
case 4:
power = 100;
break;
case 5:
power = 200;
break;
default:
power = 0;
break;
}
}
// 0, 25, 100, 200 or 500
else if (_max_power_mw < 800) {
switch (position) {
case 1:
case 2:
power = 25;
break;
case 3:
power = 100;
break;
case 4:
power = 200;
break;
case 5:
power = 500;
break;
default:
power = 0;
break;
}
}
// full range
else {
switch (position) {
case 1:
power = 25;
break;
case 2:
power = 100;
break;
case 3:
power = 200;
break;
case 4:
power = 500;
break;
case 5:
power = _max_power_mw; // some VTX's support 1000mw
break;
default:
power = 0;
break;
} }
} }

View File

@ -17,6 +17,7 @@
#include <AP_Param/AP_Param.h> #include <AP_Param/AP_Param.h>
#define VTX_MAX_CHANNELS 8 #define VTX_MAX_CHANNELS 8
#define VTX_MAX_POWER_LEVELS 9
class AP_VideoTX { class AP_VideoTX {
public: public:
@ -44,6 +45,9 @@ public:
VTX_PITMODE_ON_DISARM = (1 << 2), VTX_PITMODE_ON_DISARM = (1 << 2),
VTX_UNLOCKED = (1 << 3), VTX_UNLOCKED = (1 << 3),
VTX_PULLDOWN = (1 << 4), VTX_PULLDOWN = (1 << 4),
VTX_SA_ONE_STOP_BIT = (1 << 5),
VTX_SA_IGNORE_CRC = (1 << 6),
VTX_CRSF_IGNORE_STAT = (1 << 7),
}; };
static const char *band_names[]; static const char *band_names[];
@ -58,6 +62,22 @@ public:
MAX_BANDS MAX_BANDS
}; };
enum class PowerActive {
Unknown,
Active,
Inactive
};
struct PowerLevel {
uint8_t level;
uint16_t mw;
uint8_t dbm;
uint8_t dac; // SmartAudio v1 dac value
PowerActive active;
};
static PowerLevel _power_levels[VTX_MAX_POWER_LEVELS];
static const uint16_t VIDEO_CHANNELS[MAX_BANDS][VTX_MAX_CHANNELS]; static const uint16_t VIDEO_CHANNELS[MAX_BANDS][VTX_MAX_CHANNELS];
static uint16_t get_frequency_mhz(uint8_t band, uint8_t channel) { return VIDEO_CHANNELS[band][channel]; } static uint16_t get_frequency_mhz(uint8_t band, uint8_t channel) { return VIDEO_CHANNELS[band][channel]; }
@ -70,15 +90,31 @@ public:
bool update_frequency() const { return _defaults_set && _frequency_mhz != _current_frequency; } bool update_frequency() const { return _defaults_set && _frequency_mhz != _current_frequency; }
void update_configured_frequency(); void update_configured_frequency();
// get / set power level // get / set power level
void set_power_mw(uint16_t power) { _current_power = power; } void set_power_mw(uint16_t power);
void set_power_level(uint8_t level); void set_power_level(uint8_t level, PowerActive active=PowerActive::Active);
void set_power_dbm(uint8_t power); void set_power_dbm(uint8_t power, PowerActive active=PowerActive::Active);
void set_power_dac(uint16_t power, PowerActive active=PowerActive::Active);
// add a new dbm setting to those supported
uint8_t update_power_dbm(uint8_t power, PowerActive active=PowerActive::Active);
void update_all_power_dbm(uint8_t nlevels, const uint8_t levels[]);
void set_configured_power_mw(uint16_t power); void set_configured_power_mw(uint16_t power);
uint16_t get_configured_power_mw() const { return _power_mw; } uint16_t get_configured_power_mw() const { return _power_mw; }
uint16_t get_power_mw() const { return _current_power; } uint16_t get_power_mw() const { return _power_levels[_current_power].mw; }
uint8_t get_configured_power_dbm() const;
uint8_t get_configured_power_level() const; // get the power in dbm, rounding appropriately
bool update_power() const { return _defaults_set && _power_mw != _current_power; } uint8_t get_configured_power_dbm() const {
return _power_levels[find_current_power()].dbm;
}
// get the power "level"
uint8_t get_configured_power_level() const {
return _power_levels[find_current_power()].level & 0xF;
}
// get the power "dac"
uint8_t get_configured_power_dac() const {
return _power_levels[find_current_power()].dac;
}
bool update_power() const;
// change the video power based on switch input // change the video power based on switch input
void change_power(int8_t position); void change_power(int8_t position);
// get / set the frequency band // get / set the frequency band
@ -95,11 +131,13 @@ public:
bool update_channel() const { return _defaults_set && _channel != _current_channel; } bool update_channel() const { return _defaults_set && _channel != _current_channel; }
void update_configured_channel_and_band(); void update_configured_channel_and_band();
// get / set vtx option // get / set vtx option
void set_options(uint8_t options) { _current_options = options; } void set_options(uint16_t options) { _current_options = options; }
void set_configured_options(uint8_t options) { _options.set_and_save_ifchanged(options); } void set_configured_options(uint16_t options) { _options.set_and_save_ifchanged(options); }
uint8_t get_configured_options() const { return _options; } uint16_t get_configured_options() const { return _options; }
uint8_t get_options() const { return _current_options; } uint16_t get_options() const { return _current_options; }
bool has_option(VideoOptions option) const { return _options.get() & uint8_t(option); } bool has_option(VideoOptions option) const { return _options.get() & uint16_t(option); }
bool get_configured_pitmode() const { return _options.get() & uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE); }
bool get_pitmode() const { return _current_options & uint8_t(AP_VideoTX::VideoOptions::VTX_PITMODE); }
bool update_options() const; bool update_options() const;
// get / set whether the vtx is enabled // get / set whether the vtx is enabled
void set_enabled(bool enabled); void set_enabled(bool enabled);
@ -112,6 +150,10 @@ public:
bool set_defaults(); bool set_defaults();
// display the current VTX settings in the GCS // display the current VTX settings in the GCS
void announce_vtx_settings() const; void announce_vtx_settings() const;
// force the current values to reflect the configured values
void set_power_is_current();
void set_freq_is_current();
void set_options_are_current() { _current_options = _options; }
void set_configuration_finished(bool configuration_finished) { _configuration_finished = configuration_finished; } void set_configuration_finished(bool configuration_finished) { _configuration_finished = configuration_finished; }
bool is_configuration_finished() { return _configuration_finished; } bool is_configuration_finished() { return _configuration_finished; }
@ -119,6 +161,7 @@ public:
static AP_VideoTX *singleton; static AP_VideoTX *singleton;
private: private:
uint8_t find_current_power() const;
// channel frequency // channel frequency
AP_Int16 _frequency_mhz; AP_Int16 _frequency_mhz;
uint16_t _current_frequency; uint16_t _current_frequency;
@ -137,8 +180,8 @@ private:
uint8_t _current_channel; uint8_t _current_channel;
// vtx options // vtx options
AP_Int8 _options; AP_Int16 _options;
uint8_t _current_options; uint16_t _current_options;
AP_Int8 _enabled; AP_Int8 _enabled;
bool _current_enabled; bool _current_enabled;
@ -146,7 +189,7 @@ private:
bool _initialized; bool _initialized;
// when defaults have been configured // when defaults have been configured
bool _defaults_set; bool _defaults_set;
// true when configuration have been applied succesfully to the VTX // true when configuration have been applied successfully to the VTX
bool _configuration_finished; bool _configuration_finished;
}; };