mirror of https://github.com/pvnis/srsRAN_4G.git
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
555 lines
15 KiB
C++
555 lines
15 KiB
C++
/*
|
|
* Copyright 2013-2019 Software Radio Systems Limited
|
|
*
|
|
* This file is part of srsLTE.
|
|
*
|
|
* srsLTE is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* srsLTE 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 Affero General Public License for more details.
|
|
*
|
|
* A copy of the GNU Affero General Public License can be found in
|
|
* the LICENSE file in the top-level directory of this distribution
|
|
* and at http://www.gnu.org/licenses/.
|
|
*
|
|
*/
|
|
|
|
#ifndef ENB_CFG_PARSER_SIB1_H
|
|
#define ENB_CFG_PARSER_SIB1_H
|
|
|
|
#include <string>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <libconfig.h++>
|
|
#include <string.h>
|
|
#include <iostream>
|
|
#include "srsenb/hdr/parser.h"
|
|
|
|
#include "srsenb/hdr/stack/rrc/rrc.h"
|
|
#include "srslte/asn1/asn1_utils.h"
|
|
|
|
namespace srsenb {
|
|
|
|
using namespace libconfig;
|
|
|
|
class field_sched_info : public parser::field_itf
|
|
{
|
|
public:
|
|
field_sched_info(asn1::rrc::sib_type1_s* data_) { data = data_; }
|
|
~field_sched_info() {}
|
|
int parse(Setting &root);
|
|
const char* get_name() {
|
|
return "sched_info";
|
|
}
|
|
|
|
private:
|
|
asn1::rrc::sib_type1_s* data;
|
|
};
|
|
|
|
class field_intra_neigh_cell_list : public parser::field_itf
|
|
{
|
|
public:
|
|
field_intra_neigh_cell_list(asn1::rrc::sib_type4_s* data_) { data = data_; }
|
|
~field_intra_neigh_cell_list(){}
|
|
int parse(Setting& root);
|
|
const char* get_name() {
|
|
return "intra_neigh_cell_list";
|
|
}
|
|
|
|
private:
|
|
asn1::rrc::sib_type4_s* data;
|
|
};
|
|
|
|
class field_intra_black_cell_list : public parser::field_itf
|
|
{
|
|
public:
|
|
field_intra_black_cell_list(asn1::rrc::sib_type4_s* data_) { data = data_; }
|
|
~field_intra_black_cell_list(){}
|
|
int parse(Setting &root);
|
|
const char* get_name() {
|
|
return "intra_black_cell_list";
|
|
}
|
|
|
|
private:
|
|
asn1::rrc::sib_type4_s* data;
|
|
};
|
|
|
|
class field_carrier_freqs_info_list : public parser::field_itf
|
|
{
|
|
public:
|
|
field_carrier_freqs_info_list(asn1::rrc::sib_type7_s* data_) { data = data_; }
|
|
~field_carrier_freqs_info_list(){}
|
|
int parse(Setting &root);
|
|
const char* get_name() {
|
|
return "carrier_freqs_info_list";
|
|
}
|
|
|
|
private:
|
|
asn1::rrc::sib_type7_s* data;
|
|
};
|
|
|
|
class field_sf_mapping : public parser::field_itf
|
|
{
|
|
public:
|
|
field_sf_mapping(uint32_t *sf_mapping_, uint32_t *nof_subframes_) { sf_mapping = sf_mapping_; nof_subframes = nof_subframes_; }
|
|
~field_sf_mapping(){}
|
|
int parse(Setting &root);
|
|
const char* get_name() {
|
|
return "sf_mapping";
|
|
}
|
|
|
|
private:
|
|
uint32_t *sf_mapping;
|
|
uint32_t *nof_subframes;
|
|
};
|
|
|
|
class field_qci : public parser::field_itf
|
|
{
|
|
public:
|
|
field_qci(rrc_cfg_qci_t *cfg_) { cfg = cfg_; }
|
|
~field_qci(){}
|
|
const char* get_name() {
|
|
return "field_cqi";
|
|
}
|
|
|
|
int parse(Setting &root);
|
|
private:
|
|
rrc_cfg_qci_t *cfg;
|
|
};
|
|
|
|
// ASN1 parsers
|
|
|
|
class field_asn1 : public parser::field_itf
|
|
{
|
|
public:
|
|
field_asn1(const char* name_, bool* enabled_value_ = NULL)
|
|
{
|
|
name = name_;
|
|
enabled_value = enabled_value_;
|
|
}
|
|
|
|
const char* get_name() { return name; }
|
|
|
|
int parse(Setting& root)
|
|
{
|
|
if (root.exists(name)) {
|
|
|
|
if (enabled_value) {
|
|
*enabled_value = true;
|
|
}
|
|
return parse_value(root);
|
|
} else {
|
|
if (enabled_value) {
|
|
*enabled_value = false;
|
|
return 0;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual int parse_value(Setting& root) = 0;
|
|
|
|
protected:
|
|
const char* name;
|
|
bool* enabled_value;
|
|
};
|
|
|
|
template <class ListType>
|
|
class field_asn1_seqof_size : public field_asn1
|
|
{
|
|
ListType* store_ptr;
|
|
|
|
public:
|
|
field_asn1_seqof_size(const char* name_, ListType* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
uint32_t size_val;
|
|
if (root.lookupValue(name, size_val)) {
|
|
store_ptr->resize(size_val);
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
};
|
|
|
|
template <class ListType>
|
|
field_asn1_seqof_size<ListType>* make_asn1_seqof_size_parser(const char* name, ListType* store_ptr,
|
|
bool* enabled = NULL)
|
|
{
|
|
return new field_asn1_seqof_size<ListType>(name, store_ptr, enabled);
|
|
}
|
|
|
|
template <class ChoiceType, class NumberType>
|
|
class field_asn1_choice_type_number : public field_asn1
|
|
{
|
|
ChoiceType* store_ptr;
|
|
|
|
public:
|
|
field_asn1_choice_type_number(const char* name_, ChoiceType* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
NumberType val;
|
|
if (root.lookupValue(name, val)) {
|
|
typename ChoiceType::types type;
|
|
bool found = asn1::number_to_enum(type, val);
|
|
if (not found) {
|
|
fprintf(stderr, "Invalid option: %s for field \"%s\"\n", val.c_str(), name);
|
|
fprintf(stderr, "Valid options: %d", ChoiceType::types(0).to_number());
|
|
for (uint32_t i = 1; i < ChoiceType::types::nof_types; i++) {
|
|
fprintf(stderr, ", %s", ChoiceType::types(i).to_number());
|
|
}
|
|
fprintf(stderr, "\n");
|
|
return -1;
|
|
}
|
|
store_ptr->set(type);
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
};
|
|
|
|
template <class EnumType>
|
|
bool parse_enum_by_number(EnumType& enum_val, const char* name, Setting& root)
|
|
{
|
|
typename EnumType::number_type val;
|
|
if (parser::lookupValue(root, name, &val)) {
|
|
bool found = asn1::number_to_enum(enum_val, val);
|
|
if (not found) {
|
|
std::ostringstream ss;
|
|
ss << val;
|
|
fprintf(stderr, "Invalid option: %s for field \"%s\"\n", ss.str().c_str(), name);
|
|
ss.str("");
|
|
ss << EnumType((typename EnumType::options)0).to_number();
|
|
fprintf(stderr, "Valid options: %s", ss.str().c_str());
|
|
for (uint32_t i = 1; i < EnumType::nof_types; i++) {
|
|
ss.str("");
|
|
ss << EnumType((typename EnumType::options)i).to_number();
|
|
fprintf(stderr, ", %s", ss.str().c_str());
|
|
}
|
|
fprintf(stderr, "\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
} else {
|
|
std::string str_val;
|
|
if (parser::lookupValue(root, name, &str_val)) {
|
|
fprintf(stderr, "Expected a number for field %s but received a string %s\n", name, str_val.c_str());
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class EnumType>
|
|
bool parse_enum_by_number_str(EnumType& enum_val, const char* name, Setting& root)
|
|
{
|
|
std::string val;
|
|
if (parser::lookupValue(root, name, &val)) {
|
|
bool found = asn1::number_string_to_enum(enum_val, val);
|
|
if (not found) {
|
|
fprintf(stderr, "PARSER ERROR: Invalid option: \"%s\" for field \"%s\"\n", val.c_str(), name);
|
|
fprintf(stderr, "Valid options: \"%s\"", EnumType((typename EnumType::options)0).to_number_string().c_str());
|
|
for (uint32_t i = 1; i < EnumType::nof_types; i++) {
|
|
fprintf(stderr, ", \"%s\"", EnumType((typename EnumType::options)i).to_number_string().c_str());
|
|
}
|
|
fprintf(stderr, "\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class EnumType>
|
|
bool nowhitespace_string_to_enum(EnumType& e, const std::string& s)
|
|
{
|
|
std::string s_nows = s;
|
|
std::transform(s_nows.begin(), s_nows.end(), s_nows.begin(), ::tolower);
|
|
s_nows.erase(std::remove(s_nows.begin(), s_nows.end(), ' '), s_nows.end());
|
|
s_nows.erase(std::remove(s_nows.begin(), s_nows.end(), '-'), s_nows.end());
|
|
for (uint32_t i = 0; i < EnumType::nof_types; ++i) {
|
|
e = (typename EnumType::options)i;
|
|
std::string s_nows2 = e.to_string();
|
|
std::transform(s_nows2.begin(), s_nows2.end(), s_nows2.begin(), ::tolower);
|
|
s_nows2.erase(std::remove(s_nows2.begin(), s_nows2.end(), ' '), s_nows2.end());
|
|
s_nows2.erase(std::remove(s_nows2.begin(), s_nows2.end(), '-'), s_nows2.end());
|
|
if (s_nows2 == s_nows) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class EnumType>
|
|
bool parse_enum_by_str(EnumType& enum_val, const char* name, Setting& root)
|
|
{
|
|
std::string val;
|
|
if (root.lookupValue(name, val)) {
|
|
bool found = nowhitespace_string_to_enum(enum_val, val);
|
|
if (not found) {
|
|
fprintf(stderr, "PARSER ERROR: Invalid option: \"%s\" for field \"%s\"\n", val.c_str(), name);
|
|
fprintf(stderr, "Valid options: \"%s\"", EnumType((typename EnumType::options)0).to_string().c_str());
|
|
for (uint32_t i = 1; i < EnumType::nof_types; i++) {
|
|
fprintf(stderr, ", \"%s\"", EnumType((typename EnumType::options)i).to_string().c_str());
|
|
}
|
|
fprintf(stderr, "\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class EnumType>
|
|
class field_asn1_enum_number : public field_asn1
|
|
{
|
|
EnumType* store_ptr;
|
|
|
|
public:
|
|
field_asn1_enum_number(const char* name_, EnumType* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
bool found = parse_enum_by_number(*store_ptr, name, root);
|
|
return found ? 0 : -1;
|
|
}
|
|
};
|
|
|
|
template <class EnumType>
|
|
field_asn1_enum_number<EnumType>* make_asn1_enum_number_parser(const char* name, EnumType* store_ptr,
|
|
bool* enabled = NULL)
|
|
{
|
|
return new field_asn1_enum_number<EnumType>(name, store_ptr, enabled);
|
|
}
|
|
|
|
template <class EnumType>
|
|
class field_asn1_enum_str : public field_asn1
|
|
{
|
|
EnumType* store_ptr;
|
|
|
|
public:
|
|
field_asn1_enum_str(const char* name_, EnumType* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
bool found = parse_enum_by_str(*store_ptr, name, root);
|
|
return found ? 0 : -1;
|
|
}
|
|
};
|
|
|
|
template <class EnumType>
|
|
field_asn1_enum_str<EnumType>* make_asn1_enum_str_parser(const char* name, EnumType* store_ptr, bool* enabled = NULL)
|
|
{
|
|
return new field_asn1_enum_str<EnumType>(name, store_ptr, enabled);
|
|
}
|
|
|
|
template <class EnumType>
|
|
class field_asn1_enum_number_str : public field_asn1
|
|
{
|
|
EnumType* store_ptr;
|
|
|
|
public:
|
|
field_asn1_enum_number_str(const char* name_, EnumType* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
bool found = parse_enum_by_number_str(*store_ptr, name, root);
|
|
return found ? 0 : -1;
|
|
}
|
|
};
|
|
|
|
template <class EnumType>
|
|
field_asn1_enum_number_str<EnumType>* make_asn1_enum_number_str_parser(const char* name, EnumType* store_ptr,
|
|
bool* enabled = NULL)
|
|
{
|
|
return new field_asn1_enum_number_str<EnumType>(name, store_ptr, enabled);
|
|
}
|
|
|
|
template <class ChoiceType>
|
|
class field_asn1_choice_str : public field_asn1
|
|
{
|
|
public:
|
|
typedef bool (*func_ptr)(ChoiceType*, const char*, Setting&);
|
|
field_asn1_choice_str(const char* name_, const char* choicetypename_, func_ptr f_, ChoiceType* store_ptr_,
|
|
bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_),
|
|
choicetypename(choicetypename_),
|
|
f(f_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
typename ChoiceType::types type;
|
|
bool found = parse_enum_by_str(type, choicetypename, root);
|
|
if (not found) {
|
|
return -1;
|
|
}
|
|
store_ptr->set(type);
|
|
if (f(store_ptr, name, root)) {
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
private:
|
|
ChoiceType* store_ptr;
|
|
const char* choicetypename;
|
|
func_ptr f;
|
|
};
|
|
|
|
template <class ChoiceType, class FuncOper>
|
|
field_asn1_choice_str<ChoiceType>* make_asn1_choice_str_parser(const char* name, const char* choicetypename,
|
|
ChoiceType* store_ptr, FuncOper f, bool* enabled = NULL)
|
|
{
|
|
return new field_asn1_choice_str<ChoiceType>(name, choicetypename, f, store_ptr, enabled);
|
|
}
|
|
|
|
template <class ChoiceType>
|
|
class field_asn1_choice_number : public field_asn1
|
|
{
|
|
public:
|
|
typedef bool (*func_ptr)(ChoiceType*, const char*, Setting&);
|
|
field_asn1_choice_number(const char* name_, const char* choicetypename_, func_ptr f_, ChoiceType* store_ptr_,
|
|
bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_),
|
|
choicetypename(choicetypename_),
|
|
f(f_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
typename ChoiceType::types type;
|
|
bool found = parse_enum_by_number(type, choicetypename, root);
|
|
if (not found) {
|
|
return -1;
|
|
}
|
|
store_ptr->set(type);
|
|
if (f(store_ptr, name, root)) {
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
private:
|
|
ChoiceType* store_ptr;
|
|
const char* choicetypename;
|
|
func_ptr f;
|
|
};
|
|
|
|
template <class BitString, class NumType>
|
|
class field_asn1_bitstring_number : public field_asn1
|
|
{
|
|
BitString* store_ptr;
|
|
|
|
public:
|
|
field_asn1_bitstring_number(const char* name_, BitString* store_ptr_, bool* enabled_value_ = NULL) :
|
|
field_asn1(name_, enabled_value_),
|
|
store_ptr(store_ptr_)
|
|
{
|
|
}
|
|
|
|
int parse_value(Setting& root)
|
|
{
|
|
NumType val;
|
|
if (parser::lookupValue(root, name, &val)) {
|
|
store_ptr->from_number(val);
|
|
return 0;
|
|
} else {
|
|
std::string str_val;
|
|
if (parser::lookupValue(root, name, &str_val)) {
|
|
fprintf(stderr, "PARSER ERROR: Expected number for field %s but got the string \"%s\"\n", name,
|
|
str_val.c_str());
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
};
|
|
|
|
template <class BitString>
|
|
field_asn1_bitstring_number<BitString, uint32_t>* make_asn1_bitstring_number_parser(const char* name,
|
|
BitString* store_ptr)
|
|
{
|
|
return new field_asn1_bitstring_number<BitString, uint32_t>(name, store_ptr);
|
|
}
|
|
|
|
class phr_cnfg_parser : public parser::field_itf
|
|
{
|
|
public:
|
|
phr_cnfg_parser(asn1::rrc::mac_main_cfg_s::phr_cfg_c_* phr_cfg_) { phr_cfg = phr_cfg_; }
|
|
~phr_cnfg_parser() {}
|
|
int parse(Setting& root);
|
|
const char* get_name() { return "phr_cnfg"; }
|
|
|
|
private:
|
|
asn1::rrc::mac_main_cfg_s::phr_cfg_c_* phr_cfg;
|
|
};
|
|
|
|
class mbsfn_sf_cfg_list_parser : public parser::field_itf
|
|
{
|
|
public:
|
|
mbsfn_sf_cfg_list_parser(asn1::rrc::mbsfn_sf_cfg_list_l* mbsfn_list_, bool* enabled_) :
|
|
mbsfn_list(mbsfn_list_),
|
|
enabled(enabled_)
|
|
{
|
|
}
|
|
~mbsfn_sf_cfg_list_parser() {}
|
|
int parse(Setting& root);
|
|
const char* get_name() { return "mbsfnSubframeConfigList"; }
|
|
|
|
private:
|
|
asn1::rrc::mbsfn_sf_cfg_list_l* mbsfn_list;
|
|
bool* enabled;
|
|
};
|
|
|
|
class mbsfn_area_info_list_parser : public parser::field_itf
|
|
{
|
|
public:
|
|
mbsfn_area_info_list_parser(asn1::rrc::mbsfn_area_info_list_r9_l* mbsfn_list_, bool* enabled_) :
|
|
mbsfn_list(mbsfn_list_),
|
|
enabled(enabled_)
|
|
{
|
|
}
|
|
~mbsfn_area_info_list_parser() {}
|
|
int parse(Setting& root);
|
|
const char* get_name() { return "mbsfn_area_info_list"; }
|
|
|
|
private:
|
|
asn1::rrc::mbsfn_area_info_list_r9_l* mbsfn_list;
|
|
bool* enabled;
|
|
};
|
|
}
|
|
|
|
#endif
|
|
|