/* -*- c++ -*- */ /* * Copyright 2012-2013 Free Software Foundation, Inc. * * This file is part of GNU Radio * * SPDX-License-Identifier: GPL-3.0-or-later * */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "ctrlport_probe_psd_impl.h" #include <gnuradio/io_signature.h> namespace gr { namespace fft { ctrlport_probe_psd::sptr ctrlport_probe_psd::make(const std::string& id, const std::string& desc, int len) { return gnuradio::get_initial_sptr(new ctrlport_probe_psd_impl(id, desc, len)); } ctrlport_probe_psd_impl::ctrlport_probe_psd_impl(const std::string& id, const std::string& desc, int len) : gr::sync_block("probe_psd", gr::io_signature::make(1, 1, sizeof(gr_complex)), gr::io_signature::make(0, 0, 0)), d_id(id), d_desc(desc), d_len(len), d_fft(len, true, 1) { set_length(len); } ctrlport_probe_psd_impl::~ctrlport_probe_psd_impl() {} void ctrlport_probe_psd_impl::forecast(int noutput_items, gr_vector_int& ninput_items_required) { // make sure all inputs have noutput_items available unsigned ninputs = ninput_items_required.size(); for (unsigned i = 0; i < ninputs; i++) ninput_items_required[i] = d_len; } // boost::shared_mutex mutex_buffer; // mutable boost::mutex mutex_notify; // boost::condition_variable condition_buffer_ready; std::vector<gr_complex> ctrlport_probe_psd_impl::get() { mutex_buffer.lock(); d_buffer.clear(); mutex_buffer.unlock(); // wait for condition boost::mutex::scoped_lock lock(mutex_notify); condition_buffer_ready.wait(lock); mutex_buffer.lock(); memcpy(d_fft.get_inbuf(), &d_buffer[0], d_len * sizeof(gr_complex)); d_fft.execute(); std::vector<gr_complex> buf_copy; buf_copy.resize(d_len); gr_complex* out = d_fft.get_outbuf(); for (size_t i = 0; i < d_len; i++) { size_t idx = (i + d_len / 2) % d_len; float x = i / (d_len - 1.0f) - 0.5; buf_copy[i] = gr_complex(x, 10 * log10((out[idx] * std::conj(out[idx])).real())); } mutex_buffer.unlock(); return buf_copy; } void ctrlport_probe_psd_impl::set_length(int len) { if (len > 8191) { std::ostringstream msg; msg << "length " << len << " exceeds maximum buffer size of 8191"; GR_LOG_ERROR(d_logger, msg.str()); len = 8191; } d_len = len; d_buffer.reserve(d_len); } int ctrlport_probe_psd_impl::length() const { return (int)d_len; } int ctrlport_probe_psd_impl::work(int noutput_items, gr_vector_const_void_star& input_items, gr_vector_void_star& output_items) { const gr_complex* in = (const gr_complex*)input_items[0]; // copy samples to get buffer if we need samples mutex_buffer.lock(); if (d_buffer.size() < d_len) { // copy smaller of remaining buffer space and num inputs to work() int num_copy = std::min((int)(d_len - d_buffer.size()), noutput_items); // TODO: convert this to a copy operator for speed... for (int i = 0; i < num_copy; i++) { d_buffer.push_back(in[i]); } // notify the waiting get() if we fill up the buffer if (d_buffer.size() == d_len) { condition_buffer_ready.notify_one(); } } mutex_buffer.unlock(); return noutput_items; } void ctrlport_probe_psd_impl::setup_rpc() { #ifdef GR_CTRLPORT int len = static_cast<int>(d_len); d_rpc_vars.emplace_back( new rpcbasic_register_get<ctrlport_probe_psd, std::vector<std::complex<float>>>( alias(), d_id.c_str(), &ctrlport_probe_psd::get, pmt::make_c32vector(0, -2), pmt::make_c32vector(0, 2), pmt::make_c32vector(0, 0), "dB", d_desc.c_str(), RPC_PRIVLVL_MIN, DISPXY | DISPOPTSCATTER)); d_rpc_vars.emplace_back( new rpcbasic_register_get<ctrlport_probe_psd, int>(alias(), "length", &ctrlport_probe_psd::length, pmt::mp(1), pmt::mp(10 * len), pmt::mp(len), "samples", "get vector length", RPC_PRIVLVL_MIN, DISPNULL)); // d_rpc_vars.emplace_back( // new rpcbasic_register_set<ctrlport_probe_psd, int>( // alias(), "length", &ctrlport_probe_psd::set_length, // pmt::mp(1), pmt::mp(10*len), pmt::mp(len), // "samples", "set vector length", RPC_PRIVLVL_MIN, DISPNULL)); #endif /* GR_CTRLPORT */ } } /* namespace fft */ } /* namespace gr */