diff options
Diffstat (limited to 'host/examples')
-rw-r--r-- | host/examples/CMakeLists.txt | 1 | ||||
-rw-r--r-- | host/examples/spi.cpp | 135 |
2 files changed, 136 insertions, 0 deletions
diff --git a/host/examples/CMakeLists.txt b/host/examples/CMakeLists.txt index 40a2eb7ad..856b867e5 100644 --- a/host/examples/CMakeLists.txt +++ b/host/examples/CMakeLists.txt @@ -27,6 +27,7 @@ set(example_sources usrp_list_sensors.cpp latency_test.cpp gpio.cpp + spi.cpp sync_to_gps.cpp rfnoc_nullsource_ce_rx.cpp rfnoc_rx_to_file.cpp diff --git a/host/examples/spi.cpp b/host/examples/spi.cpp new file mode 100644 index 000000000..e5372cac3 --- /dev/null +++ b/host/examples/spi.cpp @@ -0,0 +1,135 @@ +// +// Copyright 2022 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +// Example for SPI testing. +// +// This example shows how to work with SPI which is based on the GPIO +// interface of the X410. + +#include <uhd/features/spi_getter_iface.hpp> +#include <uhd/usrp/multi_usrp.hpp> +#include <uhd/utils/safe_main.hpp> +#include <stdlib.h> +#include <boost/program_options.hpp> +#include <iostream> + +static const std::string SPI_DEFAULT_GPIO = "GPIOA"; +static const size_t SPI_DEFAULT_CLK_PIN = 0; +static const size_t SPI_DEFAULT_MISO_PIN = 1; +static const size_t SPI_DEFAULT_MOSI_PIN = 2; +static const size_t SPI_DEFAULT_CS_PIN = 3; +static const size_t SPI_DEFAULT_PAYLOAD_LENGTH = 32; +static const std::string SPI_DEFAULT_PAYLOAD = "0xfefe"; +static const size_t SPI_DEFAULT_CLK_DIVIDER = 4; + +namespace po = boost::program_options; + +int UHD_SAFE_MAIN(int argc, char* argv[]) +{ + // variables to be set by po + std::string args; + size_t clk; + size_t miso; + size_t mosi; + size_t cs; + size_t payload_length; + size_t clk_divider; + std::string payload_str; + uint32_t payload; + + // setup the program options + po::options_description desc("Allowed options"); + // clang-format off + desc.add_options() + ("help", "help message") + ("args", po::value<std::string>(&args)->default_value(""), "multi uhd device address args") + ("list-banks", "print list of banks before running tests") + ("clk", po::value<size_t>(&clk)->default_value(SPI_DEFAULT_CLK_PIN), "number of pin for SPI clock") + ("mosi", po::value<size_t>(&mosi)->default_value(SPI_DEFAULT_MOSI_PIN), "number of pin for MOSI") + ("miso", po::value<size_t>(&miso)->default_value(SPI_DEFAULT_MISO_PIN), "number of pin for MISO") + ("cs", po::value<size_t>(&cs)->default_value(SPI_DEFAULT_CS_PIN), "number of pin for chip select") + ("payload", po::value<std::string>(&payload_str)->default_value(SPI_DEFAULT_PAYLOAD), "payload as integer value") + ("length", po::value<size_t>(&payload_length)->default_value(SPI_DEFAULT_PAYLOAD_LENGTH), "payload length in bits") + ("clk-div", po::value<size_t>(&clk_divider)->default_value(SPI_DEFAULT_CLK_DIVIDER), "clock divider for SPI") + ; + // clang-format on + po::variables_map vm; + po::store(po::parse_command_line(argc, argv, desc), vm); + po::notify(vm); + + // print the help message + if (vm.count("help")) { + std::cout << argv[0] << " " << desc << std::endl; + return ~0; + } + + // create a usrp device + std::cout << std::endl; + std::cout << "Creating the usrp device with: " << args << "..." + << std::endl; + auto usrp = uhd::usrp::multi_usrp::make(args); + + if (vm.count("list-banks")) { + std::cout << "Available GPIO banks: " << std::endl; + auto banks = usrp->get_gpio_banks(0); + for (auto& bank : banks) { + std::cout << "* " << bank << std::endl; + } + } + + // Get the SPI getter interface from where we'll get + auto& spi_getter_iface = + usrp->get_radio_control().get_feature<uhd::features::spi_getter_iface>(); + + // Set all available pins to SPI for GPIO0 and GPIO1 + std::vector<std::string> sources(12, "DB0_SPI"); + usrp->set_gpio_src("GPIO0", sources); + usrp->set_gpio_src("GPIO1", sources); + + // Create slave configuration per slave + uhd::features::spi_slave_config_t slave_cfg; + slave_cfg.slave_clk = clk; + slave_cfg.slave_miso = miso; + slave_cfg.slave_mosi = mosi; + slave_cfg.slave_ss = cs; + + // The vector holds the slave configs with index=slave number + std::vector<uhd::features::spi_slave_config_t> slave_cfgs; + slave_cfgs.push_back(slave_cfg); + + // Set the data direction register + uint32_t outputs = 0x0; + outputs |= 1 << slave_cfg.slave_clk; + outputs |= 1 << slave_cfg.slave_mosi; + outputs |= 1 << slave_cfg.slave_ss; + usrp->set_gpio_attr("GPIOA", "DDR", outputs & 0xFFFFFF); + auto spi_ref = spi_getter_iface.get_spi_ref(slave_cfgs); + + std::cout << "Using pins: " << std::endl + << " Clock = " << (int)(slave_cfg.slave_clk) << std::endl + << " MOSI = " << (int)(slave_cfg.slave_mosi) << std::endl + << " MISO = " << (int)(slave_cfg.slave_miso) << std::endl + << " CS = " << (int)(slave_cfg.slave_ss) << std::endl + << std::endl; + + payload = strtoul(payload_str.c_str(), NULL, 0); + std::cout << "Writing payload: 0x" << std::hex << payload << " with length " + << std::dec << payload_length << " bits" << std::endl; + // The spi_config_t holds items like the clock divider and the MISO + // and MOSI edges + uhd::spi_config_t config; + config.divider = clk_divider; + config.use_custom_divider = true; + config.mosi_edge = config.EDGE_RISE; + config.miso_edge = config.EDGE_FALL; + + // Do the SPI transaction. There are write() and read() methods available, too. + std::cout << "Performing SPI transaction..." << std::endl; + uint32_t read_data = spi_ref->transact_spi(0, config, payload, payload_length, true); + std::cout << "Data read: 0x" << std::hex << read_data << std::endl; + + return EXIT_SUCCESS; +} |