// // Copyright 2013-2014 Ettus Research LLC // // 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 // "push" and "pop" introduced in GCC 4.6; works with all clang #if defined(__clang__) || defined(__GNUC__) && (__GNUC__ > 3) && (__GNUC_MINOR__ > 5) #pragma GCC diagnostic push #endif #if defined(__clang__) || defined(__GNUC__) #pragma GCC diagnostic ignored "-Wmissing-field-initializers" #endif namespace uhd { namespace niusrprio { //------------------------------------------------------- // niriok_proxy_impl_v1 //------------------------------------------------------- niriok_proxy_impl_v1::niriok_proxy_impl_v1() { } niriok_proxy_impl_v1::~niriok_proxy_impl_v1() { close(); } nirio_status niriok_proxy_impl_v1::open(const std::string& interface_path) { WRITER_LOCK if (interface_path.empty()) return NiRio_Status_ResourceNotFound; //close if already open. // use non-locking _close since we already have the lock _close(); nirio_status status = NiRio_Status_Success; nirio_status_chain(nirio_driver_iface::rio_open( interface_path, _device_handle), status); if (nirio_status_not_fatal(status)) { nirio_status_chain(nirio_driver_iface::rio_ioctl(_device_handle, nirio_driver_iface::NIRIO_IOCTL_POST_OPEN, NULL, 0, NULL, 0), status); nirio_ioctl_packet_t out(&_interface_num, sizeof(_interface_num), 0); nirio_status_chain(nirio_driver_iface::rio_ioctl(_device_handle, nirio_driver_iface::NIRIO_IOCTL_GET_IFACE_NUM, NULL, 0, &out, sizeof(out)), status); if (nirio_status_fatal(status)) _close(); } return status; } void niriok_proxy_impl_v1::close(void) { WRITER_LOCK _close(); } // this protected _close doesn't acquire the lock, so it can be used in methods // that already have the lock void niriok_proxy_impl_v1::_close() { if(nirio_driver_iface::rio_isopen(_device_handle)) { nirio_driver_iface::rio_ioctl( _device_handle, nirio_driver_iface::NIRIO_IOCTL_PRE_CLOSE, NULL, 0, NULL, 0); nirio_driver_iface::rio_close(_device_handle); } } nirio_status niriok_proxy_impl_v1::reset() { READER_LOCK nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::RESET; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::get_cached_session( uint32_t& session) { READER_LOCK nirio_ioctl_packet_t out(&session, sizeof(session), 0); return nirio_driver_iface::rio_ioctl(_device_handle, nirio_driver_iface::NIRIO_IOCTL_GET_SESSION, NULL, 0, &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::get_version( nirio_version_t type, uint32_t& major, uint32_t& upgrade, uint32_t& maintenance, char& phase, uint32_t& build) { nirio_device_attribute32_t version_attr = (type==CURRENT)?RIO_CURRENT_VERSION:RIO_OLDEST_COMPATIBLE_VERSION; uint32_t raw_version = 0; nirio_status status = get_attribute(version_attr, raw_version); major = (raw_version & VERSION_MAJOR_MASK) >> VERSION_MAJOR_SHIFT; upgrade = (raw_version & VERSION_UPGRD_MASK) >> VERSION_UPGRD_SHIFT; maintenance = (raw_version & VERSION_MAINT_MASK) >> VERSION_MAINT_SHIFT; build = (raw_version & VERSION_BUILD_MASK) >> VERSION_BUILD_SHIFT; uint32_t phase_num = (raw_version & VERSION_PHASE_MASK) >> VERSION_PHASE_SHIFT; switch (phase_num) { case 0: phase = 'd'; break; case 1: phase = 'a'; break; case 2: phase = 'b'; break; case 3: phase = 'f'; break; } return status; } nirio_status niriok_proxy_impl_v1::sync_operation( const void *writeBuffer, size_t writeBufferLength, void *readBuffer, size_t readBufferLength) { READER_LOCK nirio_ioctl_packet_t out(readBuffer, readBufferLength, 0); nirio_status ioctl_status = nirio_driver_iface::rio_ioctl(_device_handle, nirio_driver_iface::NIRIO_IOCTL_SYNCOP, writeBuffer, writeBufferLength, &out, sizeof(out)); if (nirio_status_fatal(ioctl_status)) return ioctl_status; return out.statusCode; } nirio_status niriok_proxy_impl_v1::get_attribute( const nirio_device_attribute32_t attribute, uint32_t& attrValue) { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::GET32; in.params.attribute32.attribute = static_cast (attribute); nirio_status status = sync_operation(&in, sizeof(in), &out, sizeof(out)); attrValue = out.params.attribute32.value; return status; } nirio_status niriok_proxy_impl_v1::set_attribute( const nirio_device_attribute32_t attribute, const uint32_t value) { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::SET32; in.params.attribute32.attribute = static_cast (attribute); in.params.attribute32.value = value; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::peek(uint32_t offset, uint32_t& value) { if (offset % 4 != 0) return NiRio_Status_MisalignedAccess; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::IO; in.subfunction = NIRIO_IO::PEEK32; in.params.io.offset = offset; nirio_status status = sync_operation(&in, sizeof(in), &out, sizeof(out)); value = out.params.io.value.value32; return status; } nirio_status niriok_proxy_impl_v1::peek(uint32_t offset, uint64_t& value) { if (offset % 8 != 0) return NiRio_Status_MisalignedAccess; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::IO; in.subfunction = NIRIO_IO::PEEK64; in.params.io.offset = offset; nirio_status status = sync_operation(&in, sizeof(in), &out, sizeof(out)); value = out.params.io.value.value64; return status; } nirio_status niriok_proxy_impl_v1::poke(uint32_t offset, const uint32_t& value) { if (offset % 4 != 0) return NiRio_Status_MisalignedAccess; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::IO; in.subfunction = NIRIO_IO::POKE32; in.params.io.offset = offset; in.params.io.value.value32 = value; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::poke(uint32_t offset, const uint64_t& value) { if (offset % 8 != 0) return NiRio_Status_MisalignedAccess; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::IO; in.subfunction = NIRIO_IO::POKE64; in.params.io.offset = offset; in.params.io.value.value64 = value; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::map_fifo_memory( uint32_t fifo_instance, size_t size, nirio_driver_iface::rio_mmap_t& map) { READER_LOCK return nirio_driver_iface::rio_mmap(_device_handle, GET_FIFO_MEMORY_TYPE(fifo_instance), size, true, map); } nirio_status niriok_proxy_impl_v1::unmap_fifo_memory( nirio_driver_iface::rio_mmap_t& map) { READER_LOCK return nirio_driver_iface::rio_munmap(map); } nirio_status niriok_proxy_impl_v1::stop_all_fifos() { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO_STOP_ALL; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::add_fifo_resource(const nirio_fifo_info_t& fifo_info) { niriok_proxy_impl_v1::nirio_syncop_in_params_t in = {}; niriok_proxy_impl_v1::nirio_syncop_out_params_t out = {}; in.function = niriok_proxy_impl_v1::NIRIO_FUNC::ADD_RESOURCE; if (fifo_info.direction == OUTPUT_FIFO) in.subfunction = niriok_proxy_impl_v1::NIRIO_RESOURCE::OUTPUT_FIFO; else in.subfunction = niriok_proxy_impl_v1::NIRIO_RESOURCE::INPUT_FIFO; in.params.add.fifoWithDataType.channel = fifo_info.channel; in.params.add.fifoWithDataType.baseAddress = fifo_info.base_addr; in.params.add.fifoWithDataType.depthInSamples = fifo_info.depth; in.params.add.fifoWithDataType.scalarType = static_cast (fifo_info.scalar_type); in.params.add.fifoWithDataType.bitWidth = fifo_info.bitWidth; in.params.add.fifoWithDataType.version = fifo_info.version; //fifo_info.integerWordLength is not needed by the v1 kernel interface return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::set_device_config() { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = niriok_proxy_impl_v1::NIRIO_FUNC::SET_DRIVER_CONFIG; in.subfunction = 0; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::start_fifo( uint32_t channel) { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::START; in.params.fifo.channel = channel; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::stop_fifo( uint32_t channel) { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::STOP; in.params.fifo.channel = channel; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::configure_fifo( uint32_t channel, uint32_t requested_depth, uint8_t requires_actuals, uint32_t& actual_depth, uint32_t& actual_size) { nirio_status status = NiRio_Status_Success; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::CONFIGURE; in.params.fifo.channel = channel; in.params.fifo.op.config.requestedDepth = requested_depth; in.params.fifo.op.config.requiresActuals = requires_actuals; status = sync_operation(&in, sizeof(in), &out, sizeof(out)); if (nirio_status_fatal(status)) return status; actual_depth = out.params.fifo.op.config.actualDepth; actual_size = out.params.fifo.op.config.actualSize; return status; } nirio_status niriok_proxy_impl_v1::wait_on_fifo( uint32_t channel, uint32_t elements_requested, uint32_t scalar_type, uint32_t bit_width, uint32_t timeout, uint8_t output, void*& data_pointer, uint32_t& elements_acquired, uint32_t& elements_remaining) { nirio_status status = NiRio_Status_Success; nirio_syncop_in_params_t in = {}; uint32_t stuffed[2]; nirio_syncop_out_params_t out = {}; init_syncop_out_params(out, stuffed, sizeof(stuffed)); in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::WAIT; in.params.fifo.channel = channel; in.params.fifo.op.wait.elementsRequested = elements_requested; in.params.fifo.op.wait.scalarType = scalar_type; in.params.fifo.op.wait.bitWidth = bit_width; in.params.fifo.op.wait.output = output; in.params.fifo.op.wait.timeout = timeout; status = sync_operation(&in, sizeof(in), &out, sizeof(out)); if (nirio_status_fatal(status)) return status; data_pointer = out.params.fifo.op.wait.elements.pointer; elements_acquired = stuffed[0]; elements_remaining = stuffed[1]; return status; } nirio_status niriok_proxy_impl_v1::grant_fifo( uint32_t channel, uint32_t elements_to_grant) { nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::GRANT; in.params.fifo.channel = channel; in.params.fifo.op.grant.elements = elements_to_grant; return sync_operation(&in, sizeof(in), &out, sizeof(out)); } nirio_status niriok_proxy_impl_v1::read_fifo( uint32_t channel, uint32_t elements_to_read, void* buffer, uint32_t buffer_datatype_width, uint32_t scalar_type, uint32_t bit_width, uint32_t timeout, uint32_t& number_read, uint32_t& number_remaining) { nirio_status status = NiRio_Status_Success; nirio_syncop_in_params_t in = {}; nirio_syncop_out_params_t out = {}; init_syncop_out_params(out, buffer, elements_to_read * buffer_datatype_width); in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::READ; in.params.fifo.channel = channel; in.params.fifo.op.readWithDataType.timeout = timeout; in.params.fifo.op.readWithDataType.scalarType = scalar_type; in.params.fifo.op.readWithDataType.bitWidth = bit_width; status = sync_operation(&in, sizeof(in), &out, sizeof(out)); if (nirio_status_fatal(status) && (status != NiRio_Status_FifoTimeout)) return status; number_read = out.params.fifo.op.read.numberRead; number_remaining = out.params.fifo.op.read.numberRemaining; return status; } nirio_status niriok_proxy_impl_v1::write_fifo( uint32_t channel, uint32_t elements_to_write, void* buffer, uint32_t buffer_datatype_width, uint32_t scalar_type, uint32_t bit_width, uint32_t timeout, uint32_t& number_remaining) { nirio_status status = NiRio_Status_Success; nirio_syncop_in_params_t in = {}; init_syncop_in_params(in, buffer, elements_to_write * buffer_datatype_width); nirio_syncop_out_params_t out = {}; in.function = NIRIO_FUNC::FIFO; in.subfunction = NIRIO_FIFO::WRITE; in.params.fifo.channel = channel; in.params.fifo.op.writeWithDataType.timeout = timeout; in.params.fifo.op.writeWithDataType.scalarType = scalar_type; in.params.fifo.op.writeWithDataType.bitWidth = bit_width; status = sync_operation(&in, sizeof(in), &out, sizeof(out)); if (nirio_status_fatal(status) && (status != NiRio_Status_FifoTimeout)) return status; number_remaining = out.params.fifo.op.write.numberRemaining; return status; } }} #if defined(__clang__) || defined(__GNUC__) && (__GNUC__ > 3) && (__GNUC_MINOR__ > 5) #pragma GCC diagnostic pop #endif