/** * Copyright 2013-2021 Software Radio Systems Limited * * This file is part of srsRAN. * * srsRAN 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. * * srsRAN 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 SRSRAN_RF_BUFFER_H #define SRSRAN_RF_BUFFER_H #include "srsran/interfaces/radio_interfaces.h" namespace srsran { /** * Implemenation of the rf_buffer_interface for the current radio implementation which uses flat arrays. * @see rf_buffer_interface * @see radio * */ class rf_buffer_t : public rf_buffer_interface { public: /** * Creates an object and allocates memory for nof_subframes_ assuming the * largest system bandwidth * @param nof_subframes_ Number of subframes to allocate */ explicit rf_buffer_t(uint32_t nof_subframes_) { if (nof_subframes_ > 0) { // Allocate buffers for an integer number of subframes for (uint32_t i = 0; i < SRSRAN_MAX_CHANNELS; i++) { sample_buffer[i] = srsran_vec_cf_malloc(nof_subframes_ * SRSRAN_SF_LEN_MAX); srsran_vec_cf_zero(sample_buffer[i], SRSRAN_SF_LEN_MAX); } allocated = true; nof_subframes = nof_subframes_; } } /** * Creates an object and sets the buffers to the flat array pointed by data. Note that data must * contain up to SRSRAN_MAX_CHANNELS pointers * @param data Flat array to use as initializer for the internal buffer pointers */ explicit rf_buffer_t(cf_t* data[SRSRAN_MAX_CHANNELS], uint32_t nof_samples_) { for (uint32_t i = 0; i < SRSRAN_MAX_CHANNELS; i++) { sample_buffer[i] = data[i]; } nof_samples = nof_samples_; } /** * Creates an object from a single array pointer. The rest of the channel pointers will be left to NULL * @param data Flat array to use as initializer for the internal buffer pointers */ explicit rf_buffer_t(cf_t* data, uint32_t nof_samples_) { sample_buffer[0] = data; nof_samples = nof_samples_; } /** * Default constructor leaves the internal pointers to NULL */ rf_buffer_t() = default; /** * The destructor will deallocate memory only if it was allocated passing nof_subframes > 0 */ ~rf_buffer_t() { if (allocated) { free_all(); } } /** * Overrides the = operator such that the lvalue internal buffers point to the pointers inside rvalue. * If memory has already been allocated in the lvalue object, it will free it before pointing the * buffers to the lvalue. * After this operator, when the lvalue is destroyed no memory will be freed. * @param other rvalue * @return lvalue */ rf_buffer_t& operator=(const rf_buffer_t& other) { if (this == &other) { return *this; } if (this->allocated) { free_all(); this->allocated = false; } for (int i = 0; i < SRSRAN_MAX_CHANNELS; i++) { this->sample_buffer[i] = other.sample_buffer[i]; } return *this; } rf_buffer_t(const rf_buffer_t& other) = delete; cf_t* get(const uint32_t& channel_idx) const override { return sample_buffer.at(channel_idx); } void set(const uint32_t& channel_idx, cf_t* ptr) override { sample_buffer.at(channel_idx) = ptr; } cf_t* get(const uint32_t& logical_ch, const uint32_t& port_idx, const uint32_t& nof_antennas) const override { return sample_buffer.at(logical_ch * nof_antennas + port_idx); } void set(const uint32_t& logical_ch, const uint32_t& port_idx, const uint32_t& nof_antennas, cf_t* ptr) override { sample_buffer.at(logical_ch * nof_antennas + port_idx) = ptr; } void** to_void() override { return (void**)sample_buffer.data(); } cf_t** to_cf_t() override { return sample_buffer.data(); } uint32_t size() override { return nof_subframes * SRSRAN_SF_LEN_MAX; } void set_nof_samples(uint32_t n) override { nof_samples = n; } uint32_t get_nof_samples() const override { return nof_samples; } private: std::array sample_buffer = {}; bool allocated = false; uint32_t nof_subframes = 0; uint32_t nof_samples = 0; void free_all() { for (uint32_t i = 0; i < SRSRAN_MAX_CHANNELS; i++) { if (sample_buffer[i]) { free(sample_buffer[i]); } } } }; } // namespace srsran #endif // SRSRAN_RF_BUFFER_H