mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-03 07:21:14 -05:00
364 lines
9.2 KiB
C++
364 lines
9.2 KiB
C++
///////////////////////////////////////////////////////////////////////////////////
|
|
// Copyright (C) 2017 F4EXB //
|
|
// //
|
|
// This program is free software; you can redistribute it and/or modify //
|
|
// it under the terms of the GNU General Public License as published by //
|
|
// the Free Software Foundation as version 3 of the License, or //
|
|
// (at your option) any later version. //
|
|
// //
|
|
// This program is distributed in the hope that it will be useful, //
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of //
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
|
|
// GNU General Public License V3 for more details. //
|
|
// //
|
|
// You should have received a copy of the GNU General Public License //
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include <QColor>
|
|
|
|
#include "dsp/dspengine.h"
|
|
#include "util/simpleserializer.h"
|
|
#include "settings/serializable.h"
|
|
#include "atvdemodsettings.h"
|
|
|
|
ATVDemodSettings::ATVDemodSettings() :
|
|
m_channelMarker(0)
|
|
{
|
|
resetToDefaults();
|
|
}
|
|
|
|
void ATVDemodSettings::resetToDefaults()
|
|
{
|
|
m_inputFrequencyOffset = 0;
|
|
m_forceDecimator = false;
|
|
m_bfoFrequency = 0.0f;
|
|
m_atvModulation = ATV_FM1;
|
|
m_fmDeviation = 0.5f;
|
|
m_amScalingFactor = 100;
|
|
m_amOffsetFactor = 0;
|
|
m_fftFiltering = false;
|
|
m_fftOppBandwidth = 0;
|
|
m_fftBandwidth = 6000;
|
|
m_nbLines = 625;
|
|
m_fps = 25;
|
|
m_atvStd = ATVStdPAL625;
|
|
m_hSync = false;
|
|
m_vSync = false;
|
|
m_invertVideo = false;
|
|
m_halfFrames = false; // m_fltRatioOfRowsToDisplay = 1.0
|
|
m_levelSynchroTop = 0.1f;
|
|
m_levelBlack = 0.3f;
|
|
m_lineTimeFactor = 0;
|
|
m_topTimeFactor = 25;
|
|
m_rgbColor = QColor(255, 255, 255).rgb();
|
|
m_title = "ATV Demodulator";
|
|
m_udpAddress = "127.0.0.1";
|
|
m_udpPort = 9999;
|
|
m_streamIndex = 0;
|
|
}
|
|
|
|
QByteArray ATVDemodSettings::serialize() const
|
|
{
|
|
SimpleSerializer s(1);
|
|
|
|
s.writeS64(1, m_inputFrequencyOffset);
|
|
s.writeU32(2, m_rgbColor);
|
|
s.writeS32(3, roundf(m_levelSynchroTop*1000.0)); // mV
|
|
s.writeS32(4, roundf(m_levelBlack*1000.0)); // mV
|
|
s.writeS32(5, m_lineTimeFactor);
|
|
s.writeS32(6, m_topTimeFactor);
|
|
s.writeS32(7, m_atvModulation);
|
|
s.writeS32(8, m_fps);
|
|
s.writeBool(9, m_hSync);
|
|
s.writeBool(10,m_vSync);
|
|
s.writeBool(11, m_halfFrames);
|
|
s.writeU32(12, m_fftBandwidth);
|
|
s.writeU32(13, m_fftOppBandwidth);
|
|
s.writeS32(14, m_bfoFrequency);
|
|
s.writeBool(15, m_invertVideo);
|
|
s.writeS32(16, m_nbLines);
|
|
s.writeS32(17, roundf(m_fmDeviation * 500.0));
|
|
s.writeS32(18, m_atvStd);
|
|
|
|
if (m_channelMarker) {
|
|
s.writeBlob(19, m_channelMarker->serialize());
|
|
}
|
|
|
|
s.writeString(20, m_title);
|
|
s.writeS32(21, m_streamIndex);
|
|
s.writeS32(22, m_amScalingFactor);
|
|
s.writeS32(23, m_amOffsetFactor);
|
|
s.writeBool(24, m_fftFiltering);
|
|
s.writeBool(25, m_forceDecimator);
|
|
|
|
return s.final();
|
|
}
|
|
|
|
bool ATVDemodSettings::deserialize(const QByteArray& arrData)
|
|
{
|
|
SimpleDeserializer d(arrData);
|
|
|
|
if (!d.isValid())
|
|
{
|
|
resetToDefaults();
|
|
return false;
|
|
}
|
|
|
|
if (d.getVersion() == 1)
|
|
{
|
|
QByteArray bytetmp;
|
|
int tmp;
|
|
|
|
d.readS64(1, &m_inputFrequencyOffset, 0);
|
|
// TODO: rgb color
|
|
d.readS32(3, &tmp, 100);
|
|
m_levelSynchroTop = tmp / 1000.0f;
|
|
d.readS32(4, &tmp, 310);
|
|
m_levelBlack = tmp / 1000.0f;
|
|
d.readS32(5, &m_lineTimeFactor, 0);
|
|
d.readS32(6, &m_topTimeFactor, 25);
|
|
d.readS32(7, &tmp, 0);
|
|
m_atvModulation = static_cast<ATVModulation>(tmp);
|
|
d.readS32(8, &tmp, 25);
|
|
int fpsIndex = getFpsIndex(tmp);
|
|
m_fps = getFps(fpsIndex);
|
|
d.readBool(9, &m_hSync, false);
|
|
d.readBool(10, &m_vSync, false);
|
|
d.readBool(11, &m_halfFrames, false);
|
|
d.readU32(12, &m_fftBandwidth, 6000);
|
|
d.readU32(13, &m_fftOppBandwidth, 0);
|
|
d.readS32(14, &m_bfoFrequency, 0);
|
|
d.readBool(15, &m_invertVideo, false);
|
|
d.readS32(16, &tmp, 625);
|
|
int nbLinesIndex = getNumberOfLinesIndex(tmp);
|
|
m_nbLines = getNumberOfLines(nbLinesIndex);
|
|
d.readS32(17, &tmp, 250);
|
|
m_fmDeviation = tmp / 500.0f;
|
|
d.readS32(18, &tmp, 1);
|
|
m_atvStd = static_cast<ATVStd>(tmp);
|
|
d.readS32(21, &m_streamIndex, 0);
|
|
d.readS32(22, &m_amScalingFactor, 100);
|
|
d.readS32(23, &m_amOffsetFactor, 0);
|
|
d.readBool(24, &m_fftFiltering, false);
|
|
d.readBool(25, &m_forceDecimator, false);
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
resetToDefaults();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
int ATVDemodSettings::getFps(int fpsIndex)
|
|
{
|
|
switch(fpsIndex)
|
|
{
|
|
case 0:
|
|
return 30;
|
|
break;
|
|
case 2:
|
|
return 20;
|
|
break;
|
|
case 3:
|
|
return 16;
|
|
break;
|
|
case 4:
|
|
return 12;
|
|
break;
|
|
case 5:
|
|
return 10;
|
|
break;
|
|
case 6:
|
|
return 8;
|
|
break;
|
|
case 7:
|
|
return 5;
|
|
break;
|
|
case 8:
|
|
return 2;
|
|
break;
|
|
case 9:
|
|
return 1;
|
|
break;
|
|
case 1:
|
|
default:
|
|
return 25;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int ATVDemodSettings::getFpsIndex(int fps)
|
|
{
|
|
if (fps <= 1) {
|
|
return 9;
|
|
} else if (fps <= 2) {
|
|
return 8;
|
|
} else if (fps <= 5) {
|
|
return 7;
|
|
} else if (fps <= 8) {
|
|
return 6;
|
|
} else if (fps <= 10) {
|
|
return 5;
|
|
} else if (fps <= 12) {
|
|
return 4;
|
|
} else if (fps <= 16) {
|
|
return 3;
|
|
} else if (fps <= 20) {
|
|
return 2;
|
|
} else if (fps <= 25) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int ATVDemodSettings::getNumberOfLines(int nbLinesIndex)
|
|
{
|
|
switch(nbLinesIndex)
|
|
{
|
|
case 0:
|
|
return 640;
|
|
break;
|
|
case 2:
|
|
return 525;
|
|
break;
|
|
case 3:
|
|
return 480;
|
|
break;
|
|
case 4:
|
|
return 405;
|
|
break;
|
|
case 5:
|
|
return 360;
|
|
break;
|
|
case 6:
|
|
return 343;
|
|
break;
|
|
case 7:
|
|
return 240;
|
|
break;
|
|
case 8:
|
|
return 180;
|
|
break;
|
|
case 9:
|
|
return 120;
|
|
break;
|
|
case 10:
|
|
return 90;
|
|
break;
|
|
case 11:
|
|
return 60;
|
|
break;
|
|
case 12:
|
|
return 32;
|
|
break;
|
|
case 1:
|
|
default:
|
|
return 625;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int ATVDemodSettings::getNumberOfLinesIndex(int nbLines)
|
|
{
|
|
if (nbLines <= 32) {
|
|
return 12;
|
|
} else if (nbLines <= 60) {
|
|
return 11;
|
|
} else if (nbLines <= 90) {
|
|
return 10;
|
|
} else if (nbLines <= 120) {
|
|
return 9;
|
|
} else if (nbLines <= 180) {
|
|
return 8;
|
|
} else if (nbLines <= 240) {
|
|
return 7;
|
|
} else if (nbLines <= 343) {
|
|
return 6;
|
|
} else if (nbLines <= 360) {
|
|
return 5;
|
|
} else if (nbLines <= 405) {
|
|
return 4;
|
|
} else if (nbLines <= 480) {
|
|
return 3;
|
|
} else if (nbLines <= 525) {
|
|
return 2;
|
|
} else if (nbLines <= 625) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
float ATVDemodSettings::getNominalLineTime(int nbLines, int fps)
|
|
{
|
|
return 1.0f / ((float) nbLines * (float) fps);
|
|
}
|
|
|
|
/**
|
|
* calculates m_fltLineTimeMultiplier
|
|
*/
|
|
void ATVDemodSettings::lineTimeUpdate(unsigned int sampleRate)
|
|
{
|
|
float nominalLineTime = getNominalLineTime(m_nbLines, m_fps);
|
|
int lineTimeScaleFactor = (int) std::log10(nominalLineTime);
|
|
|
|
if (sampleRate == 0) {
|
|
m_fltLineTimeMultiplier = std::pow(10.0, lineTimeScaleFactor-3);
|
|
} else {
|
|
m_fltLineTimeMultiplier = 1.0f / sampleRate;
|
|
}
|
|
}
|
|
|
|
float ATVDemodSettings::getLineTime(unsigned int sampleRate)
|
|
{
|
|
lineTimeUpdate(sampleRate);
|
|
float nominalLineTime = 1.0f / ((float) m_nbLines * (float) m_fps);
|
|
return nominalLineTime + m_fltLineTimeMultiplier * m_lineTimeFactor;
|
|
}
|
|
|
|
float ATVDemodSettings::getTopTime(unsigned int sampleRate)
|
|
{
|
|
return getNominalLineTime(m_nbLines, m_fps) * (4.7f / 64.0f) * (m_topTimeFactor / 100.0f);
|
|
}
|
|
|
|
int ATVDemodSettings::getRFSliderDivisor(unsigned int sampleRate)
|
|
{
|
|
int scaleFactor = (int) std::log10(sampleRate/2);
|
|
return std::pow(10.0, scaleFactor-1);
|
|
}
|
|
|
|
float ATVDemodSettings::getRFBandwidthDivisor(ATVModulation modulation)
|
|
{
|
|
switch(modulation)
|
|
{
|
|
case ATV_USB:
|
|
case ATV_LSB:
|
|
return 1.05f;
|
|
break;
|
|
case ATV_FM1:
|
|
case ATV_FM2:
|
|
case ATV_AM:
|
|
default:
|
|
return 2.2f;
|
|
}
|
|
}
|
|
|
|
void ATVDemodSettings::getBaseValues(int sampleRate, int linesPerSecond, int& tvSampleRate, uint32_t& nbPointsPerLine)
|
|
{
|
|
int maxPoints = sampleRate / linesPerSecond;
|
|
int i = maxPoints;
|
|
|
|
for (; i > 0; i--)
|
|
{
|
|
if ((i * linesPerSecond) % 10 == 0) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
nbPointsPerLine = i == 0 ? maxPoints : i;
|
|
tvSampleRate = nbPointsPerLine * linesPerSecond;
|
|
} |