//
// Copyright 2017 Ettus Research (National Instruments)
//
// 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 3 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, see .
//
#include 
#include 
#include 
#include 
using mpm::types::regs_iface;
/*! SPI implementation of the regs iface
 *
 * Uses spidev
 */
class spi_regs_iface_impl : public regs_iface
{
public:
    spi_regs_iface_impl(
        mpm::spi::spi_iface::sptr spi_iface,
        uint32_t addr_shift,
        uint32_t data_shift,
        uint32_t read_flags,
        uint32_t write_flags = 0
    ) : _spi_iface(spi_iface),
        _addr_shift(addr_shift),
        _data_shift(data_shift),
        _read_flags(read_flags),
        _write_flags(write_flags)
    {
        /* nop */
    }
    uint8_t peek8(
        const uint32_t addr
    ) {
        uint32_t transaction = 0
            | (addr << _addr_shift)
            | _read_flags
        ;
        uint32_t data = _spi_iface->transfer24_8(transaction);
        if ((data & 0xFFFFFF00) != 0) {
            throw mpm::runtime_error("SPI read returned too much data");
        }
        return data;
    }
    void poke8(
        const uint32_t addr,
        const uint8_t data
    ) {
        uint32_t transaction = 0
            | _write_flags
            | (addr << _addr_shift)
            | (data << _data_shift)
        ;
        _spi_iface->transfer24_8(transaction);
    }
    uint16_t peek16(
        const uint32_t addr
    ) {
        uint32_t transaction = 0
            | (addr << _addr_shift)
            | _read_flags
        ;
        uint32_t data = _spi_iface->transfer24_16(transaction);
        if ((data & 0xFFFF0000) != 0) {
            throw mpm::runtime_error("SPI read returned too much data");
        }
        return data;
    }
    void poke16(
        const uint32_t addr,
        const uint16_t data
    ) {
        uint32_t transaction = 0
            | _write_flags
            | (addr << _addr_shift)
            | (data << _data_shift)
        ;
        _spi_iface->transfer24_16(transaction);
    }
private:
    mpm::spi::spi_iface::sptr _spi_iface;
    uint32_t _addr_shift;
    uint32_t _data_shift;
    uint32_t _read_flags;
    uint32_t _write_flags;
};
regs_iface::sptr mpm::spi::make_spi_regs_iface(
    mpm::spi::spi_iface::sptr spi_iface,
    uint32_t addr_shift,
    uint32_t data_shift,
    uint32_t read_flags,
    uint32_t write_flags
) {
    return std::make_shared(
        spi_iface,
        addr_shift,
        data_shift,
        read_flags,
        write_flags
    );
}
mpm::types::regs_iface::sptr mpm::spi::make_spidev_regs_iface(
    const std::string &device,
    const int speed_hz,
    const int spi_mode,
    uint32_t addr_shift,
    uint32_t data_shift,
    uint32_t read_flags,
    uint32_t write_flags
) {
    auto spi_iface_sptr = mpm::spi::spi_iface::make_spidev(
        device, speed_hz, spi_mode
    );
    return std::make_shared(
        spi_iface_sptr,
        addr_shift,
        data_shift,
        read_flags,
        write_flags
    );
}