mirror of
https://github.com/rtlsdrblog/rtl-sdr-blog.git
synced 2025-01-03 23:07:15 +01:00
Merge pull request #13 from BatchDrake/feature/xfer-completion
lib: Fix race condition after cancellation of USB transfers
This commit is contained in:
commit
acb0b5cd2a
401
src/librtlsdr.c
401
src/librtlsdr.c
@ -92,18 +92,56 @@ static const int fir_default[FIR_LEN] = {
|
|||||||
101, 156, 215, 273, 327, 372, 404, 421 /* 12 bit signed */
|
101, 156, 215, 273, 327, 372, 404, 421 /* 12 bit signed */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct rtlsdr_dev;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* RTL-SDR transfer context.
|
||||||
|
*
|
||||||
|
* Proper transfer cancellation is a stateful operation that involves
|
||||||
|
* delivering an asynchronous cancellation request and waiting for its
|
||||||
|
* completion. This is necessary to prevent a race condition during
|
||||||
|
* cleanup when the device is closed (especially in Win32 systems).
|
||||||
|
*
|
||||||
|
* The rtlsdr_xfer_ctx structure encapsulates a transfer state, including
|
||||||
|
* its buffer, its completion status and a pointer to the device it
|
||||||
|
* belongs to.
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct rtlsdr_xfer_ctx {
|
||||||
|
struct rtlsdr_dev *dev;
|
||||||
|
struct libusb_transfer *xfer;
|
||||||
|
unsigned char *buffer;
|
||||||
|
int done;
|
||||||
|
int zero_copy;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct rtlsdr_xfer_ctx rtlsdr_xfer_ctx_t;
|
||||||
|
|
||||||
|
static rtlsdr_xfer_ctx_t *rtlsdr_xfer_ctx_new(struct rtlsdr_dev *dev);
|
||||||
|
static int rtlsdr_xfer_ctx_alloc_buffer(rtlsdr_xfer_ctx_t *self);
|
||||||
|
static void rtlsdr_xfer_ctx_free_buffer(rtlsdr_xfer_ctx_t *self);
|
||||||
|
static int rtlsdr_xfer_ctx_issue_transfer(rtlsdr_xfer_ctx_t *self);
|
||||||
|
static void LIBUSB_CALL _rtlsdr_xfer_libusb_callback(
|
||||||
|
struct libusb_transfer *xfer);
|
||||||
|
static void rtlsdr_xfer_ctx_destroy(rtlsdr_xfer_ctx_t *self);
|
||||||
|
|
||||||
|
#define rtlsdr_xfer_ctx_status(self) \
|
||||||
|
((self)->xfer->status)
|
||||||
|
|
||||||
|
/* Friend of rtlsdr_xfer_ctx */
|
||||||
struct rtlsdr_dev {
|
struct rtlsdr_dev {
|
||||||
libusb_context *ctx;
|
libusb_context *ctx;
|
||||||
struct libusb_device_handle *devh;
|
struct libusb_device_handle *devh;
|
||||||
uint32_t xfer_buf_num;
|
|
||||||
|
rtlsdr_xfer_ctx_t **xfer_ctx_list;
|
||||||
|
uint32_t xfer_ctx_num;
|
||||||
uint32_t xfer_buf_len;
|
uint32_t xfer_buf_len;
|
||||||
struct libusb_transfer **xfer;
|
|
||||||
unsigned char **xfer_buf;
|
|
||||||
rtlsdr_read_async_cb_t cb;
|
rtlsdr_read_async_cb_t cb;
|
||||||
void *cb_ctx;
|
void *cb_ctx;
|
||||||
enum rtlsdr_async_status async_status;
|
enum rtlsdr_async_status async_status;
|
||||||
int async_cancel;
|
int async_cancel;
|
||||||
int use_zerocopy;
|
|
||||||
/* rtl demod context */
|
/* rtl demod context */
|
||||||
uint32_t rate; /* Hz */
|
uint32_t rate; /* Hz */
|
||||||
uint32_t rtl_xtal; /* Hz */
|
uint32_t rtl_xtal; /* Hz */
|
||||||
@ -1718,22 +1756,184 @@ int rtlsdr_read_sync(rtlsdr_dev_t *dev, void *buf, int len, int *n_read)
|
|||||||
return libusb_bulk_transfer(dev->devh, 0x81, buf, len, n_read, BULK_TIMEOUT);
|
return libusb_bulk_transfer(dev->devh, 0x81, buf, len, n_read, BULK_TIMEOUT);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void LIBUSB_CALL _libusb_callback(struct libusb_transfer *xfer)
|
int rtlsdr_wait_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx)
|
||||||
{
|
{
|
||||||
rtlsdr_dev_t *dev = (rtlsdr_dev_t *)xfer->user_data;
|
return rtlsdr_read_async(dev, cb, ctx, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
if (LIBUSB_TRANSFER_COMPLETED == xfer->status) {
|
static void
|
||||||
|
rtlsdr_xfer_ctx_free_buffer(rtlsdr_xfer_ctx_t *self)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* This path is excersed only if we managed to
|
||||||
|
* allocate a buffer, which in turn only happens if
|
||||||
|
* self->dev->xfer_buf_len is set to something meaningful
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (self->buffer != NULL) {
|
||||||
|
if (self->zero_copy) {
|
||||||
|
#if defined (__linux__) && LIBUSB_API_VERSION >= 0x01000105
|
||||||
|
libusb_dev_mem_free(
|
||||||
|
self->dev->devh,
|
||||||
|
self->buffer,
|
||||||
|
self->dev->xfer_buf_len);
|
||||||
|
#endif /* __linux__ */
|
||||||
|
} else {
|
||||||
|
free(self->buffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
rtlsdr_xfer_ctx_cancel(rtlsdr_xfer_ctx_t *self)
|
||||||
|
{
|
||||||
|
int r = 0;
|
||||||
|
|
||||||
|
struct timeval zero_tv = {0, 0};
|
||||||
|
|
||||||
|
self->done = 0;
|
||||||
|
|
||||||
|
r = libusb_cancel_transfer(self->xfer);
|
||||||
|
|
||||||
|
while (!self->done) {
|
||||||
|
libusb_handle_events_timeout_completed(
|
||||||
|
self->dev->ctx,
|
||||||
|
&zero_tv,
|
||||||
|
&self->done);
|
||||||
|
}
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
rtlsdr_xfer_ctx_issue_transfer(rtlsdr_xfer_ctx_t *self)
|
||||||
|
{
|
||||||
|
self->done = 0;
|
||||||
|
|
||||||
|
libusb_fill_bulk_transfer(
|
||||||
|
self->xfer,
|
||||||
|
self->dev->devh,
|
||||||
|
0x81,
|
||||||
|
self->buffer,
|
||||||
|
self->dev->xfer_buf_len,
|
||||||
|
_rtlsdr_xfer_libusb_callback,
|
||||||
|
self,
|
||||||
|
BULK_TIMEOUT);
|
||||||
|
|
||||||
|
return libusb_submit_transfer(self->xfer);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
rtlsdr_xfer_ctx_alloc_buffer(rtlsdr_xfer_ctx_t *self)
|
||||||
|
{
|
||||||
|
int ok = 0;
|
||||||
|
uint32_t buf_len = self->dev->xfer_buf_len;
|
||||||
|
|
||||||
|
self->buffer = NULL;
|
||||||
|
self->zero_copy = 0;
|
||||||
|
|
||||||
|
#if defined(ENABLE_ZEROCOPY) && defined (__linux__) && LIBUSB_API_VERSION >= 0x01000105
|
||||||
|
self->buffer = libusb_dev_mem_alloc(self->dev->devh, buf_len);
|
||||||
|
|
||||||
|
if (self->buffer != NULL) {
|
||||||
|
/* Check if Kernel usbfs mmap() bug is present: if the
|
||||||
|
* mapping is correct, the buffers point to memory that
|
||||||
|
* was memset to 0 by the Kernel, otherwise, they point
|
||||||
|
* to random memory. We check if the buffers are zeroed
|
||||||
|
* and otherwise fall back to buffers in userspace.
|
||||||
|
*/
|
||||||
|
if (self->buffer[0] || memcmp(self->buffer,
|
||||||
|
self->buffer + 1,
|
||||||
|
buf_len - 1)) {
|
||||||
|
fprintf(stderr, "Detected Kernel usbfs mmap() "
|
||||||
|
"bug, falling back to buffers "
|
||||||
|
"in userspace\n");
|
||||||
|
} else {
|
||||||
|
/* All looks good! Green light to zerocopy */
|
||||||
|
self->zero_copy = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Zero copy buffer allocation failed. We just need to free the
|
||||||
|
* failed ones, as the current implementation allows mixed
|
||||||
|
* userspace / zero-copy buffers.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (!self->zero_copy && self->buffer != NULL) {
|
||||||
|
libusb_dev_mem_free(
|
||||||
|
self->dev->devh,
|
||||||
|
self->buffer,
|
||||||
|
buf_len);
|
||||||
|
self->buffer = NULL;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* No zero copy, allocate in userspace */
|
||||||
|
if (!self->zero_copy)
|
||||||
|
if ((self->buffer = malloc(buf_len)) == NULL)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
ok = 1;
|
||||||
|
|
||||||
|
done:
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
rtlsdr_xfer_ctx_t *rtlsdr_xfer_ctx_new(struct rtlsdr_dev *dev)
|
||||||
|
{
|
||||||
|
rtlsdr_xfer_ctx_t *new = NULL;
|
||||||
|
|
||||||
|
if (dev == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if ((new = calloc(1, sizeof(rtlsdr_xfer_ctx_t))) == NULL)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
new->dev = dev;
|
||||||
|
|
||||||
|
if ((new->xfer = libusb_alloc_transfer(0)) == NULL)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (!rtlsdr_xfer_ctx_alloc_buffer(new))
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
return new;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (new != NULL)
|
||||||
|
rtlsdr_xfer_ctx_destroy(new);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void LIBUSB_CALL _rtlsdr_xfer_libusb_callback(
|
||||||
|
struct libusb_transfer *xfer)
|
||||||
|
{
|
||||||
|
rtlsdr_xfer_ctx_t *ctx = (rtlsdr_xfer_ctx_t *) xfer->user_data;
|
||||||
|
rtlsdr_dev_t *dev = ctx->dev;
|
||||||
|
|
||||||
|
switch (xfer->status) {
|
||||||
|
case LIBUSB_TRANSFER_COMPLETED:
|
||||||
if (dev->cb)
|
if (dev->cb)
|
||||||
dev->cb(xfer->buffer, xfer->actual_length, dev->cb_ctx);
|
dev->cb(xfer->buffer, xfer->actual_length, dev->cb_ctx);
|
||||||
|
|
||||||
libusb_submit_transfer(xfer); /* resubmit transfer */
|
libusb_submit_transfer(xfer); /* resubmit transfer */
|
||||||
dev->xfer_errors = 0;
|
dev->xfer_errors = 0;
|
||||||
} else if (LIBUSB_TRANSFER_CANCELLED != xfer->status) {
|
ctx->done = 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case LIBUSB_TRANSFER_CANCELLED:
|
||||||
|
ctx->done = 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
/* Maybe an error */
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
if (LIBUSB_TRANSFER_ERROR == xfer->status)
|
if (LIBUSB_TRANSFER_ERROR == xfer->status)
|
||||||
dev->xfer_errors++;
|
dev->xfer_errors++;
|
||||||
|
|
||||||
if (dev->xfer_errors >= dev->xfer_buf_num ||
|
if (dev->xfer_errors >= dev->xfer_ctx_num ||
|
||||||
LIBUSB_TRANSFER_NO_DEVICE == xfer->status) {
|
LIBUSB_TRANSFER_NO_DEVICE == xfer->status) {
|
||||||
#endif
|
#endif
|
||||||
dev->dev_lost = 1;
|
dev->dev_lost = 1;
|
||||||
@ -1746,129 +1946,69 @@ static void LIBUSB_CALL _libusb_callback(struct libusb_transfer *xfer)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int rtlsdr_wait_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx)
|
void
|
||||||
|
rtlsdr_xfer_ctx_destroy(rtlsdr_xfer_ctx_t *self)
|
||||||
{
|
{
|
||||||
return rtlsdr_read_async(dev, cb, ctx, 0, 0);
|
if (self->xfer != NULL)
|
||||||
}
|
libusb_free_transfer(self->xfer);
|
||||||
|
|
||||||
static int _rtlsdr_alloc_async_buffers(rtlsdr_dev_t *dev)
|
rtlsdr_xfer_ctx_free_buffer(self);
|
||||||
{
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
if (!dev)
|
free(self);
|
||||||
return -1;
|
|
||||||
|
|
||||||
if (!dev->xfer) {
|
|
||||||
dev->xfer = malloc(dev->xfer_buf_num *
|
|
||||||
sizeof(struct libusb_transfer *));
|
|
||||||
|
|
||||||
for(i = 0; i < dev->xfer_buf_num; ++i)
|
|
||||||
dev->xfer[i] = libusb_alloc_transfer(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dev->xfer_buf)
|
|
||||||
return -2;
|
|
||||||
|
|
||||||
dev->xfer_buf = malloc(dev->xfer_buf_num * sizeof(unsigned char *));
|
|
||||||
memset(dev->xfer_buf, 0, dev->xfer_buf_num * sizeof(unsigned char *));
|
|
||||||
|
|
||||||
#if defined(ENABLE_ZEROCOPY) && defined (__linux__) && LIBUSB_API_VERSION >= 0x01000105
|
|
||||||
fprintf(stderr, "Allocating %d zero-copy buffers\n", dev->xfer_buf_num);
|
|
||||||
|
|
||||||
dev->use_zerocopy = 1;
|
|
||||||
for (i = 0; i < dev->xfer_buf_num; ++i) {
|
|
||||||
dev->xfer_buf[i] = libusb_dev_mem_alloc(dev->devh, dev->xfer_buf_len);
|
|
||||||
|
|
||||||
if (dev->xfer_buf[i]) {
|
|
||||||
/* Check if Kernel usbfs mmap() bug is present: if the
|
|
||||||
* mapping is correct, the buffers point to memory that
|
|
||||||
* was memset to 0 by the Kernel, otherwise, they point
|
|
||||||
* to random memory. We check if the buffers are zeroed
|
|
||||||
* and otherwise fall back to buffers in userspace.
|
|
||||||
*/
|
|
||||||
if (dev->xfer_buf[i][0] || memcmp(dev->xfer_buf[i],
|
|
||||||
dev->xfer_buf[i] + 1,
|
|
||||||
dev->xfer_buf_len - 1)) {
|
|
||||||
fprintf(stderr, "Detected Kernel usbfs mmap() "
|
|
||||||
"bug, falling back to buffers "
|
|
||||||
"in userspace\n");
|
|
||||||
dev->use_zerocopy = 0;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fprintf(stderr, "Failed to allocate zero-copy "
|
|
||||||
"buffer for transfer %d\nFalling "
|
|
||||||
"back to buffers in userspace\n", i);
|
|
||||||
dev->use_zerocopy = 0;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* zero-copy buffer allocation failed (partially or completely)
|
|
||||||
* we need to free the buffers again if already allocated */
|
|
||||||
if (!dev->use_zerocopy) {
|
|
||||||
for (i = 0; i < dev->xfer_buf_num; ++i) {
|
|
||||||
if (dev->xfer_buf[i])
|
|
||||||
libusb_dev_mem_free(dev->devh,
|
|
||||||
dev->xfer_buf[i],
|
|
||||||
dev->xfer_buf_len);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* no zero-copy available, allocate buffers in userspace */
|
|
||||||
if (!dev->use_zerocopy) {
|
|
||||||
for (i = 0; i < dev->xfer_buf_num; ++i) {
|
|
||||||
dev->xfer_buf[i] = malloc(dev->xfer_buf_len);
|
|
||||||
|
|
||||||
if (!dev->xfer_buf[i])
|
|
||||||
return -ENOMEM;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _rtlsdr_free_async_buffers(rtlsdr_dev_t *dev)
|
static int _rtlsdr_free_async_buffers(rtlsdr_dev_t *dev)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
if (!dev)
|
if (dev == NULL)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if (dev->xfer) {
|
if (dev->xfer_ctx_list != NULL) {
|
||||||
for(i = 0; i < dev->xfer_buf_num; ++i) {
|
for (i = 0; i < dev->xfer_ctx_num; ++i)
|
||||||
if (dev->xfer[i]) {
|
if (dev->xfer_ctx_list[i] != NULL)
|
||||||
libusb_free_transfer(dev->xfer[i]);
|
rtlsdr_xfer_ctx_destroy(dev->xfer_ctx_list[i]);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
free(dev->xfer);
|
free(dev->xfer_ctx_list);
|
||||||
dev->xfer = NULL;
|
dev->xfer_ctx_list = NULL;
|
||||||
}
|
|
||||||
|
|
||||||
if (dev->xfer_buf) {
|
|
||||||
for (i = 0; i < dev->xfer_buf_num; ++i) {
|
|
||||||
if (dev->xfer_buf[i]) {
|
|
||||||
if (dev->use_zerocopy) {
|
|
||||||
#if defined (__linux__) && LIBUSB_API_VERSION >= 0x01000105
|
|
||||||
libusb_dev_mem_free(dev->devh,
|
|
||||||
dev->xfer_buf[i],
|
|
||||||
dev->xfer_buf_len);
|
|
||||||
#endif
|
|
||||||
} else {
|
|
||||||
free(dev->xfer_buf[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
free(dev->xfer_buf);
|
|
||||||
dev->xfer_buf = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int _rtlsdr_alloc_async_buffers(rtlsdr_dev_t *dev)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
int ret = -1;
|
||||||
|
|
||||||
|
if (dev == NULL)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
if (dev->xfer_ctx_list == NULL) {
|
||||||
|
/* Lazy allocation of async buffers */
|
||||||
|
|
||||||
|
if ((dev->xfer_ctx_list = calloc(
|
||||||
|
dev->xfer_ctx_num,
|
||||||
|
sizeof(rtlsdr_xfer_ctx_t *))) == NULL)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
for (i = 0; i < dev->xfer_ctx_num; ++i) {
|
||||||
|
dev->xfer_ctx_list[i] = rtlsdr_xfer_ctx_new(dev);
|
||||||
|
if (dev->xfer_ctx_list[i] == NULL)
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
done:
|
||||||
|
if (ret != 0)
|
||||||
|
_rtlsdr_free_async_buffers(dev);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
int rtlsdr_read_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx,
|
int rtlsdr_read_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx,
|
||||||
uint32_t buf_num, uint32_t buf_len)
|
uint32_t buf_num, uint32_t buf_len)
|
||||||
{
|
{
|
||||||
@ -1891,9 +2031,9 @@ int rtlsdr_read_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx,
|
|||||||
dev->cb_ctx = ctx;
|
dev->cb_ctx = ctx;
|
||||||
|
|
||||||
if (buf_num > 0)
|
if (buf_num > 0)
|
||||||
dev->xfer_buf_num = buf_num;
|
dev->xfer_ctx_num = buf_num;
|
||||||
else
|
else
|
||||||
dev->xfer_buf_num = DEFAULT_BUF_NUMBER;
|
dev->xfer_ctx_num = DEFAULT_BUF_NUMBER;
|
||||||
|
|
||||||
if (buf_len > 0 && buf_len % 512 == 0) /* len must be multiple of 512 */
|
if (buf_len > 0 && buf_len % 512 == 0) /* len must be multiple of 512 */
|
||||||
dev->xfer_buf_len = buf_len;
|
dev->xfer_buf_len = buf_len;
|
||||||
@ -1902,17 +2042,9 @@ int rtlsdr_read_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx,
|
|||||||
|
|
||||||
_rtlsdr_alloc_async_buffers(dev);
|
_rtlsdr_alloc_async_buffers(dev);
|
||||||
|
|
||||||
for(i = 0; i < dev->xfer_buf_num; ++i) {
|
for(i = 0; i < dev->xfer_ctx_num; ++i) {
|
||||||
libusb_fill_bulk_transfer(dev->xfer[i],
|
r = rtlsdr_xfer_ctx_issue_transfer(dev->xfer_ctx_list[i]);
|
||||||
dev->devh,
|
|
||||||
0x81,
|
|
||||||
dev->xfer_buf[i],
|
|
||||||
dev->xfer_buf_len,
|
|
||||||
_libusb_callback,
|
|
||||||
(void *)dev,
|
|
||||||
BULK_TIMEOUT);
|
|
||||||
|
|
||||||
r = libusb_submit_transfer(dev->xfer[i]);
|
|
||||||
if (r < 0) {
|
if (r < 0) {
|
||||||
fprintf(stderr, "Failed to submit transfer %i\n"
|
fprintf(stderr, "Failed to submit transfer %i\n"
|
||||||
"Please increase your allowed "
|
"Please increase your allowed "
|
||||||
@ -1938,21 +2070,16 @@ int rtlsdr_read_async(rtlsdr_dev_t *dev, rtlsdr_read_async_cb_t cb, void *ctx,
|
|||||||
if (RTLSDR_CANCELING == dev->async_status) {
|
if (RTLSDR_CANCELING == dev->async_status) {
|
||||||
next_status = RTLSDR_INACTIVE;
|
next_status = RTLSDR_INACTIVE;
|
||||||
|
|
||||||
if (!dev->xfer)
|
if (dev->xfer_ctx_list == NULL)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
for(i = 0; i < dev->xfer_buf_num; ++i) {
|
for(i = 0; i < dev->xfer_ctx_num; ++i) {
|
||||||
if (!dev->xfer[i])
|
if (dev->xfer_ctx_list[i] == NULL)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (LIBUSB_TRANSFER_CANCELLED !=
|
if (LIBUSB_TRANSFER_CANCELLED !=
|
||||||
dev->xfer[i]->status) {
|
rtlsdr_xfer_ctx_status(dev->xfer_ctx_list[i])) {
|
||||||
r = libusb_cancel_transfer(dev->xfer[i]);
|
r = rtlsdr_xfer_ctx_cancel(dev->xfer_ctx_list[i]);
|
||||||
/* handle events after canceling
|
|
||||||
* to allow transfer status to
|
|
||||||
* propagate */
|
|
||||||
libusb_handle_events_timeout_completed(dev->ctx,
|
|
||||||
&zerotv, NULL);
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user