/** * Digital Voice Modem - Host Software * GPLv2 Open Source. Use is subject to license terms. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * @package DVM / Host Software * */ // // Based on code from the MMDVMHost project. (https://github.com/g4klx/MMDVMHost) // Licensed under the GPLv2 License (https://opensource.org/licenses/GPL-2.0) // /* * Copyright (C) 2016 by Jonathan Naylor G4KLX * Copyright (C) 2017-2022 by Bryan Biedenkapp N2PLL * * 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; either version 2 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 for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "lookups/RadioIdLookup.h" #include "p25/P25Defines.h" #include "Log.h" #include "Timer.h" using namespace lookups; #include #include #include #include #include #include #include // --------------------------------------------------------------------------- // Public Class Members // --------------------------------------------------------------------------- /// /// Initializes a new instance of the RadioIdLookup class. /// /// Full-path to the radio ID table file. /// Interval of time to reload the radio ID table. /// Flag indicating whether radio ID access control is enabled. RadioIdLookup::RadioIdLookup(const std::string& filename, uint32_t reloadTime, bool ridAcl) : LookupTable(filename, reloadTime), m_acl(ridAcl) { /* stub */ } /// /// Finalizes a instance of the RadioIdLookup class. /// RadioIdLookup::~RadioIdLookup() { /* stub */ } /// /// Toggles the specified radio ID enabled or disabled. /// /// Unique ID to toggle. /// Flag indicating if radio ID is enabled or not. void RadioIdLookup::toggleEntry(uint32_t id, bool enabled) { RadioId rid = find(id); if (rid.radioEnabled() == false && rid.radioDefault() == true) { if (enabled) { LogMessage(LOG_HOST, "Added enabled RID %u to RID ACL table", id); } else { LogMessage(LOG_HOST, "Added disabled RID %u to RID ACL table", id); } } if (rid.radioEnabled() == false && rid.radioDefault() == false) { if (enabled) { LogMessage(LOG_HOST, "Enabled RID %u in RID ACL table", id); } else { LogMessage(LOG_HOST, "Disabled RID %u in RID ACL table", id); } } addEntry(id, enabled); } /// /// Adds a new entry to the lookup table by the specified unique ID. /// /// Unique ID to add. /// Flag indicating if radio ID is enabled or not. void RadioIdLookup::addEntry(uint32_t id, bool enabled) { if ((id == p25::P25_WUID_ALL) || (id == p25::P25_WUID_FNE)) { return; } RadioId entry = RadioId(enabled, false); m_mutex.lock(); { try { RadioId _entry = m_table.at(id); // if the enabled value doesn't match -- override with the intended if (_entry.radioEnabled() != enabled) { _entry = RadioId(enabled, false); m_table[id] = _entry; } } catch (...) { m_table[id] = entry; } } m_mutex.unlock(); } /// /// Finds a table entry in this lookup table. /// /// Unique identifier for table entry. /// Table entry. RadioId RadioIdLookup::find(uint32_t id) { RadioId entry; if ((id == p25::P25_WUID_ALL) || (id == p25::P25_WUID_FNE)) { return RadioId(true, false); } m_mutex.lock(); { try { entry = m_table.at(id); } catch (...) { entry = RadioId(false, true); } } m_mutex.unlock(); return entry; } /// /// Flag indicating whether radio ID access control is enabled or not. /// /// True, if radio ID access control is enabled, otherwise false. bool RadioIdLookup::getACL() { return m_acl; } // --------------------------------------------------------------------------- // Private Class Members // --------------------------------------------------------------------------- /// /// Loads the table from the passed lookup table file. /// /// True, if lookup table was loaded, otherwise false. bool RadioIdLookup::load() { if (m_filename.length() <= 0) { return false; } std::ifstream file (m_filename, std::ifstream::in); if (file.fail()) { LogError(LOG_HOST, "Cannot open the lookup file - %s", m_filename.c_str()); return false; } // clear table clear(); m_mutex.lock(); { // read lines from file std::string line; while (std::getline(file, line)) { if (line.length() > 0) { if (line.at(0) == '#') continue; // tokenize line std::string next; std::vector parsed; char delim = ','; for (char c : line) { if (c == delim) { if (!next.empty()) { parsed.push_back(next); next.clear(); } } else next += c; } if (!next.empty()) parsed.push_back(next); // parse tokenized line uint32_t id = ::atoi(parsed[0].c_str()); bool radioEnabled = ::atoi(parsed[1].c_str()) == 1; bool radioDefault = false; m_table[id] = RadioId(radioEnabled, radioDefault); } } } m_mutex.unlock(); file.close(); size_t size = m_table.size(); if (size == 0U) return false; LogInfoEx(LOG_HOST, "Loaded %u entries into lookup table", size); return true; }