//
// Copyright 2014-2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0-or-later
//

#include <uhd/rfnoc/window_block_ctrl.hpp>
#include <uhd/convert.hpp>
#include <uhd/utils/log.hpp>

using namespace uhd::rfnoc;

class window_block_ctrl_impl : public window_block_ctrl
{
public:
    UHD_RFNOC_BLOCK_CONSTRUCTOR(window_block_ctrl),
        _item_type("sc16"), // We only support sc16 in this block
        _bpi(uhd::convert::get_bytes_per_item("sc16"))
    {
        _max_len = uint32_t(user_reg_read64(RB_MAX_WINDOW_LEN));
        UHD_LOGGER_DEBUG(unique_id())
            << "window_block::window_block() max_len ==" << _max_len << std::endl;
        UHD_ASSERT_THROW(_max_len);

        // TODO we need a coercer to check that spp on the prop tree doesn't get set to anything invalid
        _set_default_window(std::min<size_t>(get_arg<int>("spp"), _max_len));
    }

    //! Set window coefficients and length
    void set_window(const std::vector<int> &coeffs)
    {
        UHD_LOGGER_TRACE(unique_id())
            << "window_block::set_window()" << std::endl;
        if (coeffs.size() > _max_len) {
            throw uhd::value_error(str(
                boost::format("window_block::set_window(): Too many window "
                              "coefficients! Provided %d, window allows up to %d.\n")
                % coeffs.size() % _max_len
            ));
        }

        size_t window_len = coeffs.size();

        // Window block can take complex coefficients in sc16 format, but typical usage is
        // to have real(coeffs) == imag(coeffs)
        std::vector<uint32_t> coeffs_;
        for (size_t i = 0; i < window_len - 1; i++) {
            if (coeffs[i] > 32767 || coeffs[i] < -32768) {
                throw uhd::value_error(str(
                    boost::format("window_block::set_window(): Coefficient %d "
                                  "(index %d) outside coefficient range [-32768,32767].\n")
                    % coeffs[i] % i));
            }
            coeffs_.push_back(coeffs[i]);
        }

        // Write coefficients via the load bus
        for (size_t i = 0; i < window_len - 1; i++) {
            sr_write(AXIS_WINDOW_LOAD, coeffs_[i]);
        }
        // Assert tlast when sending the final coefficient (sorry, no joke here)
        sr_write(AXIS_WINDOW_LOAD_TLAST, coeffs_.back());
        // Set the window length
        sr_write(SR_WINDOW_LEN, window_len);

        // This block requires spp to match the window length:
        set_arg<int>("spp", int(window_len));
    }

    //! Returns the maximum window length of this block.
    size_t get_max_len() const
    {
        return _max_len;
    }

    size_t get_window_len() const
    {
        return size_t(get_arg<int>("spp"));
    }


private:
    const std::string _item_type;
    const size_t _bpi;
    size_t _max_len;

    //! Default is a rectangular window
    void _set_default_window(size_t window_len) {
        std::vector<int> default_coeffs(window_len, (1 << 15)-1);
        set_window(default_coeffs);
    }
};

UHD_RFNOC_BLOCK_REGISTER(window_block_ctrl, "Window");