mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-25 10:00:21 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			718 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			718 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
| @file Connection_uLimeSDR.cpp
 | |
| @author Lime Microsystems
 | |
| @brief Implementation of uLimeSDR board connection.
 | |
| */
 | |
| 
 | |
| #include "Connection_uLimeSDR.h"
 | |
| #include "ErrorReporting.h"
 | |
| #include <cstring>
 | |
| #include <iostream>
 | |
| 
 | |
| #include <thread>
 | |
| #include <chrono>
 | |
| #include <FPGA_common.h>
 | |
| #include <LMS7002M.h>
 | |
| #include <ciso646>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace lime;
 | |
| 
 | |
| #define __unix__
 | |
| 
 | |
| Connection_uLimeSDR::Connection_uLimeSDR(void *arg)
 | |
| {
 | |
|     RxLoopFunction = bind(&Connection_uLimeSDR::ReceivePacketsLoop, this, std::placeholders::_1);
 | |
|     TxLoopFunction = bind(&Connection_uLimeSDR::TransmitPacketsLoop, this, std::placeholders::_1);
 | |
| 
 | |
|     isConnected = false;
 | |
| 
 | |
|     mStreamWrEndPtAddr = 0x03;
 | |
|     mStreamRdEndPtAddr = 0x83;
 | |
|     isConnected = false;
 | |
|     txSize = 0;
 | |
|     rxSize = 0;
 | |
| #ifndef __unix__
 | |
| 	mFTHandle = NULL;
 | |
| #else
 | |
|     dev_handle = 0;
 | |
|     devs = 0;
 | |
| 	mUsbCounter = 0;
 | |
|     ctx = (libusb_context *)arg;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**	@brief Initializes port type and object necessary to communicate to usb device.
 | |
| */
 | |
| Connection_uLimeSDR::Connection_uLimeSDR(void *arg, const unsigned index, const int vid, const int pid)
 | |
| {
 | |
|     RxLoopFunction = bind(&Connection_uLimeSDR::ReceivePacketsLoop, this, std::placeholders::_1);
 | |
|     TxLoopFunction = bind(&Connection_uLimeSDR::TransmitPacketsLoop, this, std::placeholders::_1);
 | |
|     mTimestampOffset = 0;
 | |
|     rxLastTimestamp.store(0);
 | |
|     mExpectedSampleRate = 0;
 | |
|     generateData.store(false);
 | |
|     rxRunning.store(false);
 | |
|     txRunning.store(false);
 | |
|     isConnected = false;
 | |
|     terminateRx.store(false);
 | |
|     terminateTx.store(false);
 | |
|     rxDataRate_Bps.store(0);
 | |
|     txDataRate_Bps.store(0);
 | |
| 
 | |
|     mStreamWrEndPtAddr = 0x03;
 | |
|     mStreamRdEndPtAddr = 0x83;
 | |
|     isConnected = false;
 | |
|     txSize = 0;
 | |
|     rxSize = 0;
 | |
| #ifndef __unix__
 | |
|     mFTHandle = NULL;
 | |
| #else
 | |
|     dev_handle = 0;
 | |
|     devs = 0;
 | |
| 	mUsbCounter = 0;
 | |
|     ctx = (libusb_context *)arg;
 | |
| #endif
 | |
|     if (this->Open(index, vid, pid) != 0)
 | |
|         std::cerr << GetLastErrorMessage() << std::endl;
 | |
|     this->SetReferenceClockRate(52e6);
 | |
|     GetChipVersion();
 | |
| }
 | |
| 
 | |
| /**	@brief Closes connection to chip and deallocates used memory.
 | |
| */
 | |
| Connection_uLimeSDR::~Connection_uLimeSDR()
 | |
| {
 | |
|     for(auto i : mTxStreams)
 | |
|         ControlStream((size_t)i, false);
 | |
|     for(auto i : mRxStreams)
 | |
|         ControlStream((size_t)i, false);
 | |
|     for(auto i : mTxStreams)
 | |
|         CloseStream((size_t)i);
 | |
|     for(auto i : mRxStreams)
 | |
|         CloseStream((size_t)i);
 | |
|     UpdateThreads();
 | |
|     Close();
 | |
| }
 | |
| #ifdef __unix__
 | |
| int Connection_uLimeSDR::FT_FlushPipe(unsigned char ep)
 | |
| {
 | |
|     int actual = 0;
 | |
|     unsigned char wbuffer[20]={0};
 | |
| 
 | |
|     mUsbCounter++;
 | |
|     wbuffer[0] = (mUsbCounter)&0xFF;
 | |
|     wbuffer[1] = (mUsbCounter>>8)&0xFF;
 | |
|     wbuffer[2] = (mUsbCounter>>16)&0xFF;
 | |
|     wbuffer[3] = (mUsbCounter>>24)&0xFF;
 | |
|     wbuffer[4] = ep;
 | |
|     libusb_bulk_transfer(dev_handle, 0x01, wbuffer, 20, &actual, 1000);
 | |
|     if (actual != 20)
 | |
|         return -1;
 | |
| 
 | |
|     mUsbCounter++;
 | |
|     wbuffer[0] = (mUsbCounter)&0xFF;
 | |
|     wbuffer[1] = (mUsbCounter>>8)&0xFF;
 | |
|     wbuffer[2] = (mUsbCounter>>16)&0xFF;
 | |
|     wbuffer[3] = (mUsbCounter>>24)&0xFF;
 | |
|     wbuffer[4] = ep;
 | |
|     wbuffer[5] = 0x03;
 | |
|     libusb_bulk_transfer(dev_handle, 0x01, wbuffer, 20, &actual, 1000);
 | |
|     if (actual != 20)
 | |
|         return -1;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int Connection_uLimeSDR::FT_SetStreamPipe(unsigned char ep, size_t size)
 | |
| {
 | |
| 
 | |
|     int actual = 0;
 | |
|     unsigned char wbuffer[20]={0};
 | |
| 
 | |
|     mUsbCounter++;
 | |
|     wbuffer[0] = (mUsbCounter)&0xFF;
 | |
|     wbuffer[1] = (mUsbCounter>>8)&0xFF;
 | |
|     wbuffer[2] = (mUsbCounter>>16)&0xFF;
 | |
|     wbuffer[3] = (mUsbCounter>>24)&0xFF;
 | |
|     wbuffer[4] = ep;
 | |
|     libusb_bulk_transfer(dev_handle, 0x01, wbuffer, 20, &actual, 1000);
 | |
|     if (actual != 20)
 | |
|         return -1;
 | |
| 
 | |
|     mUsbCounter++;
 | |
|     wbuffer[0] = (mUsbCounter)&0xFF;
 | |
|     wbuffer[1] = (mUsbCounter>>8)&0xFF;
 | |
|     wbuffer[2] = (mUsbCounter>>16)&0xFF;
 | |
|     wbuffer[3] = (mUsbCounter>>24)&0xFF;
 | |
|     wbuffer[5] = 0x02;
 | |
|     wbuffer[8] = (size)&0xFF;
 | |
|     wbuffer[9] = (size>>8)&0xFF;
 | |
|     wbuffer[10] = (size>>16)&0xFF;
 | |
|     wbuffer[11] = (size>>24)&0xFF;
 | |
|     libusb_bulk_transfer(dev_handle, 0x01, wbuffer, 20, &actual, 1000);
 | |
|     if (actual != 20)
 | |
|         return -1;
 | |
|     return 0;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /**	@brief Tries to open connected USB device and find communication endpoints.
 | |
| @return Returns 0-Success, other-EndPoints not found or device didn't connect.
 | |
| */
 | |
| int Connection_uLimeSDR::Open(const unsigned index, const int vid, const int pid)
 | |
| {
 | |
| #ifndef __unix__
 | |
| 	DWORD devCount;
 | |
| 	FT_STATUS ftStatus = FT_OK;
 | |
| 	DWORD dwNumDevices = 0;
 | |
| 	// Open a device
 | |
| 	ftStatus = FT_Create(0, FT_OPEN_BY_INDEX, &mFTHandle);
 | |
| 	if (FT_FAILED(ftStatus))
 | |
| 	{
 | |
| 		ReportError(ENODEV, "Failed to list USB Devices");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	FT_AbortPipe(mFTHandle, mStreamRdEndPtAddr);
 | |
| 	FT_AbortPipe(mFTHandle, 0x82);
 | |
| 	FT_AbortPipe(mFTHandle, 0x02);
 | |
| 	FT_AbortPipe(mFTHandle, mStreamWrEndPtAddr);
 | |
| 	FT_SetStreamPipe(mFTHandle, FALSE, FALSE, 0x82, 64);
 | |
| 	FT_SetStreamPipe(mFTHandle, FALSE, FALSE, 0x02, 64);
 | |
|     FT_SetPipeTimeout(mFTHandle, 0x02, 500);
 | |
|     FT_SetPipeTimeout(mFTHandle, 0x82, 500);
 | |
| 	isConnected = true;
 | |
| 	return 0;
 | |
| #else
 | |
|     dev_handle = libusb_open_device_with_vid_pid(ctx, vid, pid);
 | |
| 
 | |
|     if(dev_handle == nullptr)
 | |
|         return ReportError(ENODEV, "libusb_open failed");
 | |
|     libusb_reset_device(dev_handle);
 | |
|     if(libusb_kernel_driver_active(dev_handle, 1) == 1)   //find out if kernel driver is attached
 | |
|     {
 | |
|         printf("Kernel Driver Active\n");
 | |
|         if(libusb_detach_kernel_driver(dev_handle, 1) == 0) //detach it
 | |
|             printf("Kernel Driver Detached!\n");
 | |
|     }
 | |
|     int r = libusb_claim_interface(dev_handle, 1); //claim interface 0 (the first) of device
 | |
|     if(r < 0)
 | |
|     {
 | |
|         printf("Cannot Claim Interface\n");
 | |
|         return ReportError(-1, "Cannot claim interface - %s", libusb_strerror(libusb_error(r)));
 | |
|     }
 | |
|     r = libusb_claim_interface(dev_handle, 1); //claim interface 0 (the first) of device
 | |
|     if(r < 0)
 | |
|     {
 | |
|         printf("Cannot Claim Interface\n");
 | |
|         return ReportError(-1, "Cannot claim interface - %s", libusb_strerror(libusb_error(r)));
 | |
|     }
 | |
|     printf("Claimed Interface\n");
 | |
| 
 | |
|     FT_SetStreamPipe(0x82,64);
 | |
|     FT_SetStreamPipe(0x02,64);
 | |
|     isConnected = true;
 | |
|     return 0;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**	@brief Closes communication to device.
 | |
| */
 | |
| void Connection_uLimeSDR::Close()
 | |
| {
 | |
| #ifndef __unix__
 | |
| 	FT_Close(mFTHandle);
 | |
| #else
 | |
|     if(dev_handle != 0)
 | |
|     {
 | |
|         FT_FlushPipe(mStreamRdEndPtAddr);
 | |
|         FT_FlushPipe(0x82);
 | |
|         libusb_release_interface(dev_handle, 1);
 | |
|         libusb_close(dev_handle);
 | |
|         dev_handle = 0;
 | |
|     }
 | |
| #endif
 | |
|     isConnected = false;
 | |
| }
 | |
| 
 | |
| /**	@brief Returns connection status
 | |
| @return 1-connection open, 0-connection closed.
 | |
| */
 | |
| bool Connection_uLimeSDR::IsOpen()
 | |
| {
 | |
|     return isConnected;
 | |
| }
 | |
| 
 | |
| #ifndef __unix__
 | |
| int Connection_uLimeSDR::ReinitPipe(unsigned char ep)
 | |
| {
 | |
|     FT_AbortPipe(mFTHandle, ep);
 | |
|     FT_FlushPipe(mFTHandle, ep);
 | |
|     FT_SetStreamPipe(mFTHandle, FALSE, FALSE, ep, 64);
 | |
|     return 0;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /**	@brief Sends given data buffer to chip through USB port.
 | |
| @param buffer data buffer, must not be longer than 64 bytes.
 | |
| @param length given buffer size.
 | |
| @param timeout_ms timeout limit for operation in milliseconds
 | |
| @return number of bytes sent.
 | |
| */
 | |
| int Connection_uLimeSDR::Write(const unsigned char *buffer, const int length, int timeout_ms)
 | |
| {
 | |
|     std::lock_guard<std::mutex> lock(mExtraUsbMutex);
 | |
|     long len = 0;
 | |
|     if (IsOpen() == false)
 | |
|         return 0;
 | |
| 
 | |
| #ifndef __unix__
 | |
|     // Write to channel 1 ep 0x02
 | |
|     ULONG ulBytesWrite = 0;
 | |
|     FT_STATUS ftStatus = FT_OK;
 | |
|     OVERLAPPED	vOverlapped = { 0 };
 | |
|     FT_InitializeOverlapped(mFTHandle, &vOverlapped);
 | |
|     ftStatus = FT_WritePipe(mFTHandle, 0x02, (unsigned char*)buffer, length, &ulBytesWrite, &vOverlapped);
 | |
|     if (ftStatus != FT_IO_PENDING)
 | |
|     {
 | |
|         FT_ReleaseOverlapped(mFTHandle, &vOverlapped);
 | |
|         ReinitPipe(0x02);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     DWORD dwRet = WaitForSingleObject(vOverlapped.hEvent, timeout_ms);
 | |
|     if (dwRet == WAIT_OBJECT_0 || dwRet == WAIT_TIMEOUT)
 | |
|     {
 | |
|         if (GetOverlappedResult(mFTHandle, &vOverlapped, &ulBytesWrite, FALSE) == FALSE)
 | |
|         {
 | |
|             ReinitPipe(0x02);
 | |
|             ulBytesWrite = -1;
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ReinitPipe(0x02);
 | |
|         ulBytesWrite = -1;
 | |
|     }
 | |
|     FT_ReleaseOverlapped(mFTHandle, &vOverlapped);
 | |
|     return ulBytesWrite;
 | |
| #else
 | |
|     unsigned char* wbuffer = new unsigned char[length];
 | |
|     memcpy(wbuffer, buffer, length);
 | |
|     int actual = 0;
 | |
|     libusb_bulk_transfer(dev_handle, 0x02, wbuffer, length, &actual, timeout_ms);
 | |
|     len = actual;
 | |
|     delete[] wbuffer;
 | |
|     return len;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**	@brief Reads data coming from the chip through USB port.
 | |
| @param buffer pointer to array where received data will be copied, array must be
 | |
| big enough to fit received data.
 | |
| @param length number of bytes to read from chip.
 | |
| @param timeout_ms timeout limit for operation in milliseconds
 | |
| @return number of bytes received.
 | |
| */
 | |
| 
 | |
| int Connection_uLimeSDR::Read(unsigned char *buffer, const int length, int timeout_ms)
 | |
| {
 | |
|     std::lock_guard<std::mutex> lock(mExtraUsbMutex);
 | |
|     long len = length;
 | |
|     if(IsOpen() == false)
 | |
|         return 0;
 | |
| #ifndef __unix__
 | |
|     //
 | |
|     // Read from channel 1 ep 0x82
 | |
|     //
 | |
|     ULONG ulBytesRead = 0;
 | |
|     FT_STATUS ftStatus = FT_OK;
 | |
|     OVERLAPPED	vOverlapped = { 0 };
 | |
|     FT_InitializeOverlapped(mFTHandle, &vOverlapped);
 | |
|     ftStatus = FT_ReadPipe(mFTHandle, 0x82, buffer, length, &ulBytesRead, &vOverlapped);
 | |
|     if (ftStatus != FT_IO_PENDING)
 | |
|     {
 | |
|         FT_ReleaseOverlapped(mFTHandle, &vOverlapped);
 | |
|         ReinitPipe(0x82);
 | |
|         return -1;;
 | |
|     }
 | |
| 
 | |
|     DWORD dwRet = WaitForSingleObject(vOverlapped.hEvent, timeout_ms);
 | |
|     if (dwRet == WAIT_OBJECT_0 || dwRet == WAIT_TIMEOUT)
 | |
|     {
 | |
|         if (GetOverlappedResult(mFTHandle, &vOverlapped, &ulBytesRead, FALSE)==FALSE)
 | |
|         {
 | |
|             ReinitPipe(0x82);
 | |
|             ulBytesRead = -1;
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ReinitPipe(0x82);
 | |
|         ulBytesRead = -1;
 | |
|     }
 | |
|     FT_ReleaseOverlapped(mFTHandle, &vOverlapped);
 | |
|     return ulBytesRead;
 | |
| #else
 | |
|     int actual = 0;
 | |
|     libusb_bulk_transfer(dev_handle, 0x82, buffer, len, &actual, timeout_ms);
 | |
|     len = actual;
 | |
| #endif
 | |
|     return len;
 | |
| }
 | |
| 
 | |
| #ifdef __unix__
 | |
| /**	@brief Function for handling libusb callbacks
 | |
| */
 | |
| static void callback_libusbtransfer(libusb_transfer *trans)
 | |
| {
 | |
|     Connection_uLimeSDR::USBTransferContext *context = reinterpret_cast<Connection_uLimeSDR::USBTransferContext*>(trans->user_data);
 | |
|     std::unique_lock<std::mutex> lck(context->transferLock);
 | |
|     switch(trans->status)
 | |
|     {
 | |
|         case LIBUSB_TRANSFER_CANCELLED:
 | |
|             //printf("Transfer %i canceled\n", context->id);
 | |
|             context->bytesXfered = trans->actual_length;
 | |
|             context->done.store(true);
 | |
|             //context->used = false;
 | |
|             //context->reset();
 | |
|             break;
 | |
|         case LIBUSB_TRANSFER_COMPLETED:
 | |
|             //if(trans->actual_length == context->bytesExpected)
 | |
|             {
 | |
|                 context->bytesXfered = trans->actual_length;
 | |
|                 context->done.store(true);
 | |
|             }
 | |
|         break;
 | |
|         case LIBUSB_TRANSFER_ERROR:
 | |
|             printf("TRANSFER ERRRO\n");
 | |
|             context->bytesXfered = trans->actual_length;
 | |
|             context->done.store(true);
 | |
|             //context->used = false;
 | |
|             break;
 | |
|         case LIBUSB_TRANSFER_TIMED_OUT:
 | |
|             //printf("transfer timed out %i\n", context->id);
 | |
|             context->bytesXfered = trans->actual_length;
 | |
|             context->done.store(true);
 | |
|             //context->used = false;
 | |
| 
 | |
|             break;
 | |
|         case LIBUSB_TRANSFER_OVERFLOW:
 | |
|             printf("transfer overflow\n");
 | |
| 
 | |
|             break;
 | |
|         case LIBUSB_TRANSFER_STALL:
 | |
|             printf("transfer stalled\n");
 | |
|             break;
 | |
|         case LIBUSB_TRANSFER_NO_DEVICE:
 | |
|             printf("transfer no device\n");
 | |
| 
 | |
|             break;
 | |
|     }
 | |
|     lck.unlock();
 | |
|     context->cv.notify_one();
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /**
 | |
| @brief Starts asynchronous data reading from board
 | |
| @param *buffer buffer where to store received data
 | |
| @param length number of bytes to read
 | |
| @return handle of transfer context
 | |
| */
 | |
| int Connection_uLimeSDR::BeginDataReading(char *buffer, uint32_t length)
 | |
| {
 | |
|     int i = 0;
 | |
|     bool contextFound = false;
 | |
|     //find not used context
 | |
|     for(i = 0; i<USB_MAX_CONTEXTS; i++)
 | |
|     {
 | |
|         if(!contexts[i].used)
 | |
|         {
 | |
|             contextFound = true;
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     if(!contextFound)
 | |
|     {
 | |
|         printf("No contexts left for reading data\n");
 | |
|         return -1;
 | |
|     }
 | |
|     contexts[i].used = true;
 | |
| 
 | |
| #ifndef __unix__
 | |
| 	if (length != rxSize)
 | |
| 	{
 | |
| 		rxSize = length;
 | |
| 		FT_SetStreamPipe(mFTHandle, FALSE, FALSE, mStreamRdEndPtAddr, rxSize);
 | |
| 	}
 | |
|     FT_InitializeOverlapped(mFTHandle, &contexts[i].inOvLap);
 | |
| 	ULONG ulActual;
 | |
|     FT_STATUS ftStatus = FT_OK;
 | |
|     ftStatus = FT_ReadPipe(mFTHandle, mStreamRdEndPtAddr, (unsigned char*)buffer, length, &ulActual, &contexts[i].inOvLap);
 | |
|     if (ftStatus != FT_IO_PENDING)
 | |
|         return -1;
 | |
| #else
 | |
|     if (length != rxSize)
 | |
|     {
 | |
|         rxSize = length;
 | |
|         FT_SetStreamPipe(mStreamRdEndPtAddr,rxSize);
 | |
|     }
 | |
|     unsigned int Timeout = 500;
 | |
|     libusb_transfer *tr = contexts[i].transfer;
 | |
|     libusb_fill_bulk_transfer(tr, dev_handle, mStreamRdEndPtAddr, (unsigned char*)buffer, length, callback_libusbtransfer, &contexts[i], Timeout);
 | |
|     contexts[i].done = false;
 | |
|     contexts[i].bytesXfered = 0;
 | |
|     contexts[i].bytesExpected = length;
 | |
|     int status = libusb_submit_transfer(tr);
 | |
|     if(status != 0)
 | |
|     {
 | |
|         printf("ERROR BEGIN DATA READING %s\n", libusb_error_name(status));
 | |
|         contexts[i].used = false;
 | |
|         return -1;
 | |
|     }
 | |
| #endif
 | |
|     return i;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Waits for asynchronous data reception
 | |
| @param contextHandle handle of which context data to wait
 | |
| @param timeout_ms number of miliseconds to wait
 | |
| @return 1-data received, 0-data not received
 | |
| */
 | |
| int Connection_uLimeSDR::WaitForReading(int contextHandle, unsigned int timeout_ms)
 | |
| {
 | |
|     if(contextHandle >= 0 && contexts[contextHandle].used == true)
 | |
|     {
 | |
| #ifndef __unix__
 | |
|         DWORD dwRet = WaitForSingleObject(contexts[contextHandle].inOvLap.hEvent, timeout_ms);
 | |
| 		if (dwRet == WAIT_OBJECT_0)
 | |
| 			return 1;
 | |
| #else
 | |
|         auto t1 = chrono::high_resolution_clock::now();
 | |
|         auto t2 = chrono::high_resolution_clock::now();
 | |
| 
 | |
|         std::unique_lock<std::mutex> lck(contexts[contextHandle].transferLock);
 | |
|         while(contexts[contextHandle].done.load() == false && std::chrono::duration_cast<std::chrono::milliseconds>(t2 - t1).count() < timeout_ms)
 | |
|         {
 | |
|             //blocking not to waste CPU
 | |
|             contexts[contextHandle].cv.wait_for(lck, chrono::milliseconds(timeout_ms));
 | |
|             t2 = chrono::high_resolution_clock::now();
 | |
|         }
 | |
|         return contexts[contextHandle].done.load() == true;
 | |
| #endif
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Finishes asynchronous data reading from board
 | |
| @param buffer array where to store received data
 | |
| @param length number of bytes to read
 | |
| @param contextHandle handle of which context to finish
 | |
| @return false failure, true number of bytes received
 | |
| */
 | |
| int Connection_uLimeSDR::FinishDataReading(char *buffer, uint32_t length, int contextHandle)
 | |
| {
 | |
|     if(contextHandle >= 0 && contexts[contextHandle].used == true)
 | |
|     {
 | |
| #ifndef __unix__
 | |
| 	ULONG ulActualBytesTransferred;
 | |
|         FT_STATUS ftStatus = FT_OK;
 | |
| 
 | |
|         ftStatus = FT_GetOverlappedResult(mFTHandle, &contexts[contextHandle].inOvLap, &ulActualBytesTransferred, FALSE);
 | |
|         if (ftStatus != FT_OK)
 | |
|             length = 0;
 | |
|         else
 | |
|             length = ulActualBytesTransferred;
 | |
|         FT_ReleaseOverlapped(mFTHandle, &contexts[contextHandle].inOvLap);
 | |
|         contexts[contextHandle].used = false;
 | |
|         return length;
 | |
| #else
 | |
|         length = contexts[contextHandle].bytesXfered;
 | |
|         contexts[contextHandle].used = false;
 | |
|         contexts[contextHandle].reset();
 | |
|         return length;
 | |
| #endif
 | |
|     }
 | |
|     else
 | |
|         return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Aborts reading operations
 | |
| */
 | |
| void Connection_uLimeSDR::AbortReading()
 | |
| {
 | |
| #ifndef __unix__
 | |
| 	FT_AbortPipe(mFTHandle, mStreamRdEndPtAddr);
 | |
| 	for (int i = 0; i < USB_MAX_CONTEXTS; ++i)
 | |
| 	{
 | |
| 		if (contexts[i].used == true)
 | |
| 		{
 | |
|             FT_ReleaseOverlapped(mFTHandle, &contexts[i].inOvLap);
 | |
| 			contexts[i].used = false;
 | |
| 		}
 | |
| 	}
 | |
|     FT_FlushPipe(mFTHandle, mStreamRdEndPtAddr);
 | |
| 	rxSize = 0;
 | |
| #else
 | |
| 
 | |
|     for(int i = 0; i<USB_MAX_CONTEXTS; ++i)
 | |
|     {
 | |
|         if(contexts[i].used)
 | |
|             libusb_cancel_transfer(contexts[i].transfer);
 | |
|     }
 | |
|     FT_FlushPipe(mStreamRdEndPtAddr);
 | |
|     rxSize = 0;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Starts asynchronous data Sending to board
 | |
| @param *buffer buffer to send
 | |
| @param length number of bytes to send
 | |
| @return handle of transfer context
 | |
| */
 | |
| int Connection_uLimeSDR::BeginDataSending(const char *buffer, uint32_t length)
 | |
| {
 | |
|     int i = 0;
 | |
|     //find not used context
 | |
|     bool contextFound = false;
 | |
|     for(i = 0; i<USB_MAX_CONTEXTS; i++)
 | |
|     {
 | |
|         if(!contextsToSend[i].used)
 | |
|         {
 | |
|             contextFound = true;
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     if(!contextFound)
 | |
|         return -1;
 | |
|     contextsToSend[i].used = true;
 | |
| 
 | |
| #ifndef __unix__
 | |
| 	FT_STATUS ftStatus = FT_OK;
 | |
| 	ULONG ulActualBytesSend;
 | |
| 	if (length != txSize)
 | |
| 	{
 | |
| 		txSize = length;
 | |
| 		FT_SetStreamPipe(mFTHandle, FALSE, FALSE, mStreamWrEndPtAddr, txSize);
 | |
| 	}
 | |
|     FT_InitializeOverlapped(mFTHandle, &contextsToSend[i].inOvLap);
 | |
| 	ftStatus = FT_WritePipe(mFTHandle, mStreamWrEndPtAddr, (unsigned char*)buffer, length, &ulActualBytesSend, &contextsToSend[i].inOvLap);
 | |
| 	if (ftStatus != FT_IO_PENDING)
 | |
| 		return -1;
 | |
| #else
 | |
|     if (length != txSize)
 | |
|     {
 | |
|         txSize = length;
 | |
|         FT_SetStreamPipe(mStreamWrEndPtAddr,txSize);
 | |
|     }
 | |
|     unsigned int Timeout = 500;
 | |
|     libusb_transfer *tr = contextsToSend[i].transfer;
 | |
|     libusb_fill_bulk_transfer(tr, dev_handle, mStreamWrEndPtAddr, (unsigned char*)buffer, length, callback_libusbtransfer, &contextsToSend[i], Timeout);
 | |
|     contextsToSend[i].done = false;
 | |
|     contextsToSend[i].bytesXfered = 0;
 | |
|     contextsToSend[i].bytesExpected = length;
 | |
|     int status = libusb_submit_transfer(tr);
 | |
|     if(status != 0)
 | |
|     {
 | |
|         printf("ERROR BEGIN DATA SENDING %s\n", libusb_error_name(status));
 | |
|         contextsToSend[i].used = false;
 | |
|         return -1;
 | |
|     }
 | |
| #endif
 | |
|     return i;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Waits for asynchronous data sending
 | |
| @param contextHandle handle of which context data to wait
 | |
| @param timeout_ms number of miliseconds to wait
 | |
| @return 1-data received, 0-data not received
 | |
| */
 | |
| int Connection_uLimeSDR::WaitForSending(int contextHandle, unsigned int timeout_ms)
 | |
| {
 | |
|     if(contextsToSend[contextHandle].used == true)
 | |
|     {
 | |
| #ifndef __unix__
 | |
|         DWORD dwRet = WaitForSingleObject(contextsToSend[contextHandle].inOvLap.hEvent, timeout_ms);
 | |
| 		if (dwRet == WAIT_OBJECT_0)
 | |
| 			return 1;
 | |
| #else
 | |
|         auto t1 = chrono::high_resolution_clock::now();
 | |
|         auto t2 = chrono::high_resolution_clock::now();
 | |
|         std::unique_lock<std::mutex> lck(contextsToSend[contextHandle].transferLock);
 | |
|         while(contextsToSend[contextHandle].done.load() == false && std::chrono::duration_cast<std::chrono::milliseconds>(t2 - t1).count() < timeout_ms)
 | |
|         {
 | |
|             //blocking not to waste CPU
 | |
|             contextsToSend[contextHandle].cv.wait_for(lck, chrono::milliseconds(timeout_ms));
 | |
|             t2 = chrono::high_resolution_clock::now();
 | |
|         }
 | |
|         return contextsToSend[contextHandle].done == true;
 | |
| #endif
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Finishes asynchronous data sending to board
 | |
| @param buffer array where to store received data
 | |
| @param length number of bytes to read
 | |
| @param contextHandle handle of which context to finish
 | |
| @return false failure, true number of bytes sent
 | |
| */
 | |
| int Connection_uLimeSDR::FinishDataSending(const char *buffer, uint32_t length, int contextHandle)
 | |
| {
 | |
|     if(contextsToSend[contextHandle].used == true)
 | |
|     {
 | |
| #ifndef __unix__
 | |
|         ULONG ulActualBytesTransferred ;
 | |
|         FT_STATUS ftStatus = FT_OK;
 | |
|         ftStatus = FT_GetOverlappedResult(mFTHandle, &contextsToSend[contextHandle].inOvLap, &ulActualBytesTransferred, FALSE);
 | |
|         if (ftStatus != FT_OK)
 | |
|             length = 0;
 | |
|         else
 | |
|         length = ulActualBytesTransferred;
 | |
|         FT_ReleaseOverlapped(mFTHandle, &contextsToSend[contextHandle].inOvLap);
 | |
| 	    contextsToSend[contextHandle].used = false;
 | |
| 	    return length;
 | |
| #else
 | |
|         length = contextsToSend[contextHandle].bytesXfered;
 | |
|         contextsToSend[contextHandle].used = false;
 | |
|         contextsToSend[contextHandle].reset();
 | |
|         return length;
 | |
| #endif
 | |
|     }
 | |
|     else
 | |
|         return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
| @brief Aborts sending operations
 | |
| */
 | |
| void Connection_uLimeSDR::AbortSending()
 | |
| {
 | |
| #ifndef __unix__
 | |
| 	FT_AbortPipe(mFTHandle, mStreamWrEndPtAddr);
 | |
| 	for (int i = 0; i < USB_MAX_CONTEXTS; ++i)
 | |
| 	{
 | |
| 		if (contextsToSend[i].used == true)
 | |
| 		{
 | |
|             FT_ReleaseOverlapped(mFTHandle, &contextsToSend[i].inOvLap);
 | |
| 			contextsToSend[i].used = false;
 | |
| 		}
 | |
| 	}
 | |
| 	txSize = 0;
 | |
| #else
 | |
|     for(int i = 0; i<USB_MAX_CONTEXTS; ++i)
 | |
|     {
 | |
|         if(contextsToSend[i].used)
 | |
|             libusb_cancel_transfer(contextsToSend[i].transfer);
 | |
|     }
 | |
|     FT_FlushPipe(mStreamWrEndPtAddr);
 | |
|     txSize = 0;
 | |
| #endif
 | |
| }
 |