add rds program type name decoding

This commit is contained in:
AlexandreRouma 2024-01-29 21:00:23 +01:00
parent 05ab17add3
commit 120745de19
2 changed files with 65 additions and 1 deletions

View File

@ -156,7 +156,7 @@ namespace rds {
// Acquire lock // Acquire lock
std::lock_guard<std::mutex> lck(blockBMtx); std::lock_guard<std::mutex> lck(blockBMtx);
// If it didn't decode properly return // If it didn't decode properly return (TODO: Make sure this is not needed)
if (!blockAvail[BLOCK_TYPE_B]) { return; } if (!blockAvail[BLOCK_TYPE_B]) { return; }
// Decode group type and version // Decode group type and version
@ -240,6 +240,48 @@ namespace rds {
group2LastUpdate = std::chrono::high_resolution_clock::now(); group2LastUpdate = std::chrono::high_resolution_clock::now();
} }
void Decoder::decodeGroup10() {
// Acquire lock
std::lock_guard<std::mutex> lck(group10Mtx);
// Check if the text needs to be cleared
bool ab = (blocks[BLOCK_TYPE_B] >> 14) & 1;
if (ab != ptnAB) {
programTypeName = " ";
}
ptnAB = ab;
// Decode segment address
bool addr = (blocks[BLOCK_TYPE_B] >> 10) & 1;
// Save text depending on address
if (addr) {
if (blockAvail[BLOCK_TYPE_C]) {
programTypeName[4] = (blocks[BLOCK_TYPE_C] >> 18) & 0xFF;
programTypeName[5] = (blocks[BLOCK_TYPE_C] >> 10) & 0xFF;
}
if (blockAvail[BLOCK_TYPE_D]) {
programTypeName[6] = (blocks[BLOCK_TYPE_D] >> 18) & 0xFF;
programTypeName[7] = (blocks[BLOCK_TYPE_D] >> 10) & 0xFF;
}
}
else {
if (blockAvail[BLOCK_TYPE_C]) {
programTypeName[0] = (blocks[BLOCK_TYPE_C] >> 18) & 0xFF;
programTypeName[1] = (blocks[BLOCK_TYPE_C] >> 10) & 0xFF;
}
if (blockAvail[BLOCK_TYPE_D]) {
programTypeName[2] = (blocks[BLOCK_TYPE_D] >> 18) & 0xFF;
programTypeName[3] = (blocks[BLOCK_TYPE_D] >> 10) & 0xFF;
}
}
flog::debug("PTN: '{}'", programTypeName);
// Update timeout
group10LastUpdate = std::chrono::high_resolution_clock::now();
}
void Decoder::decodeGroup() { void Decoder::decodeGroup() {
// Make sure blocks B is available // Make sure blocks B is available
if (!blockAvail[BLOCK_TYPE_B]) { return; } if (!blockAvail[BLOCK_TYPE_B]) { return; }
@ -247,6 +289,8 @@ namespace rds {
// Decode block B // Decode block B
decodeBlockB(); decodeBlockB();
//flog::debug("RDS Group {}{}", groupType, (groupVer == GROUP_VER_A) ? 'A':'B');
// Decode depending on group type // Decode depending on group type
switch (groupType) { switch (groupType) {
case 0: case 0:
@ -255,6 +299,9 @@ namespace rds {
case 2: case 2:
decodeGroup2(); decodeGroup2();
break; break;
case 10:
decodeGroup10();
break;
default: default:
break; break;
} }
@ -298,4 +345,9 @@ namespace rds {
auto now = std::chrono::high_resolution_clock::now(); auto now = std::chrono::high_resolution_clock::now();
return (std::chrono::duration_cast<std::chrono::milliseconds>(now - group2LastUpdate)).count() < RDS_GROUP_2_TIMEOUT_MS; return (std::chrono::duration_cast<std::chrono::milliseconds>(now - group2LastUpdate)).count() < RDS_GROUP_2_TIMEOUT_MS;
} }
bool Decoder::group10Valid() {
auto now = std::chrono::high_resolution_clock::now();
return (std::chrono::duration_cast<std::chrono::milliseconds>(now - group10LastUpdate)).count() < RDS_GROUP_10_TIMEOUT_MS;
}
} }

View File

@ -8,6 +8,7 @@
#define RDS_BLOCK_B_TIMEOUT_MS 5000.0 #define RDS_BLOCK_B_TIMEOUT_MS 5000.0
#define RDS_GROUP_0_TIMEOUT_MS 5000.0 #define RDS_GROUP_0_TIMEOUT_MS 5000.0
#define RDS_GROUP_2_TIMEOUT_MS 5000.0 #define RDS_GROUP_2_TIMEOUT_MS 5000.0
#define RDS_GROUP_10_TIMEOUT_MS 5000.0
namespace rds { namespace rds {
enum BlockType { enum BlockType {
@ -232,6 +233,9 @@ namespace rds {
bool radioTextValid() { std::lock_guard<std::mutex> lck(group2Mtx); return group2Valid(); } bool radioTextValid() { std::lock_guard<std::mutex> lck(group2Mtx); return group2Valid(); }
std::string getRadioText() { std::lock_guard<std::mutex> lck(group2Mtx); return radioText; } std::string getRadioText() { std::lock_guard<std::mutex> lck(group2Mtx); return radioText; }
bool programTypeNameValid() { std::lock_guard<std::mutex> lck(group10Mtx); return group10Valid(); }
std::string getProgramTypeName() { std::lock_guard<std::mutex> lck(group10Mtx); return programTypeName; }
private: private:
static uint16_t calcSyndrome(uint32_t block); static uint16_t calcSyndrome(uint32_t block);
static uint32_t correctErrors(uint32_t block, BlockType type, bool& recovered); static uint32_t correctErrors(uint32_t block, BlockType type, bool& recovered);
@ -239,6 +243,7 @@ namespace rds {
void decodeBlockB(); void decodeBlockB();
void decodeGroup0(); void decodeGroup0();
void decodeGroup2(); void decodeGroup2();
void decodeGroup10();
void decodeGroup(); void decodeGroup();
void decodeCallsign(); void decodeCallsign();
@ -247,6 +252,7 @@ namespace rds {
bool blockBValid(); bool blockBValid();
bool group0Valid(); bool group0Valid();
bool group2Valid(); bool group2Valid();
bool group10Valid();
// State machine // State machine
uint32_t shiftReg = 0; uint32_t shiftReg = 0;
@ -289,5 +295,11 @@ namespace rds {
bool rtAB = false; bool rtAB = false;
std::string radioText = " "; std::string radioText = " ";
// Group type 10
std::mutex group10Mtx;
std::chrono::time_point<std::chrono::high_resolution_clock> group10LastUpdate{}; // 1970-01-01
bool ptnAB = false;
std::string programTypeName = " ";
}; };
} }