hantek-4032l: Increase speed of data getting.
- add support for multiple transfers. - set nummber of samples to 1 for FPGA FW version 0 - increase size of data transfer buffer to 2kB. Signed-off-by: Andrej Valek <andy@skyrain.eu>
This commit is contained in:
parent
43d8603571
commit
2958315ded
|
@ -219,8 +219,6 @@ static GSList *scan(struct sr_dev_driver *di, GSList *options)
|
||||||
|
|
||||||
devc->capture_ratio = 5;
|
devc->capture_ratio = 5;
|
||||||
|
|
||||||
devc->usb_transfer = libusb_alloc_transfer(0);
|
|
||||||
|
|
||||||
sdi->priv = devc;
|
sdi->priv = devc;
|
||||||
devices = g_slist_append(devices, sdi);
|
devices = g_slist_append(devices, sdi);
|
||||||
|
|
||||||
|
@ -407,6 +405,7 @@ static int dev_acquisition_start(const struct sr_dev_inst *sdi)
|
||||||
|
|
||||||
/* Initialize variables. */
|
/* Initialize variables. */
|
||||||
devc->acq_aborted = FALSE;
|
devc->acq_aborted = FALSE;
|
||||||
|
devc->submitted_transfers = 0;
|
||||||
|
|
||||||
/* Calculate packet ratio. */
|
/* Calculate packet ratio. */
|
||||||
cmd_pkt->pre_trigger_size = (cmd_pkt->sample_size * devc->capture_ratio) / 100;
|
cmd_pkt->pre_trigger_size = (cmd_pkt->sample_size * devc->capture_ratio) / 100;
|
||||||
|
@ -509,15 +508,8 @@ static int dev_acquisition_start(const struct sr_dev_inst *sdi)
|
||||||
|
|
||||||
static int dev_acquisition_stop(struct sr_dev_inst *sdi)
|
static int dev_acquisition_stop(struct sr_dev_inst *sdi)
|
||||||
{
|
{
|
||||||
struct dev_context *devc = sdi->priv;
|
/* Stop capturing. */
|
||||||
|
return h4032l_stop(sdi);
|
||||||
devc->acq_aborted = TRUE;
|
|
||||||
if (devc->usb_transfer)
|
|
||||||
libusb_cancel_transfer(devc->usb_transfer);
|
|
||||||
|
|
||||||
devc->status = H4032L_STATUS_IDLE;
|
|
||||||
|
|
||||||
return SR_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
SR_PRIV struct sr_dev_driver hantek_4032l_driver_info = {
|
SR_PRIV struct sr_dev_driver hantek_4032l_driver_info = {
|
||||||
|
|
|
@ -39,24 +39,55 @@ struct h4032l_status_packet {
|
||||||
uint32_t fpga_version;
|
uint32_t fpga_version;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void abort_acquisition(struct dev_context *devc)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
devc->acq_aborted = TRUE;
|
||||||
|
|
||||||
|
for (i = devc->num_transfers - 1; i >= 0; i--) {
|
||||||
|
if (devc->transfers[i])
|
||||||
|
libusb_cancel_transfer(devc->transfers[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
devc->status = H4032L_STATUS_IDLE;
|
||||||
|
}
|
||||||
|
|
||||||
static void finish_acquisition(struct sr_dev_inst *sdi)
|
static void finish_acquisition(struct sr_dev_inst *sdi)
|
||||||
{
|
{
|
||||||
|
struct dev_context *devc = sdi->priv;
|
||||||
struct drv_context *drvc = sdi->driver->context;
|
struct drv_context *drvc = sdi->driver->context;
|
||||||
|
|
||||||
std_session_send_df_end(sdi);
|
std_session_send_df_end(sdi);
|
||||||
usb_source_remove(sdi->session, drvc->sr_ctx);
|
usb_source_remove(sdi->session, drvc->sr_ctx);
|
||||||
|
|
||||||
|
devc->num_transfers = 0;
|
||||||
|
g_free(devc->transfers);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void free_transfer(struct libusb_transfer *transfer)
|
static void free_transfer(struct libusb_transfer *transfer)
|
||||||
{
|
{
|
||||||
struct sr_dev_inst *sdi = transfer->user_data;
|
struct sr_dev_inst *sdi = transfer->user_data;
|
||||||
struct dev_context *devc = sdi->priv;
|
struct dev_context *devc = sdi->priv;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
if ((transfer->buffer != (unsigned char *)&devc->cmd_pkt) &&
|
||||||
|
(transfer->buffer != devc->buffer)) {
|
||||||
|
g_free(transfer->buffer);
|
||||||
|
}
|
||||||
|
|
||||||
transfer->buffer = NULL;
|
transfer->buffer = NULL;
|
||||||
libusb_free_transfer(transfer);
|
libusb_free_transfer(transfer);
|
||||||
devc->usb_transfer = NULL;
|
|
||||||
|
|
||||||
finish_acquisition(sdi);
|
for (i = 0; i < devc->num_transfers; i++) {
|
||||||
|
if (devc->transfers[i] == transfer) {
|
||||||
|
devc->transfers[i] = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (--devc->submitted_transfers == 0)
|
||||||
|
finish_acquisition(sdi);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void resubmit_transfer(struct libusb_transfer *transfer)
|
static void resubmit_transfer(struct libusb_transfer *transfer)
|
||||||
|
@ -90,7 +121,6 @@ void LIBUSB_CALL h4032l_data_transfer_callback(struct libusb_transfer *transfer)
|
||||||
{
|
{
|
||||||
const struct sr_dev_inst *sdi = transfer->user_data;
|
const struct sr_dev_inst *sdi = transfer->user_data;
|
||||||
struct dev_context *devc = sdi->priv;
|
struct dev_context *devc = sdi->priv;
|
||||||
struct drv_context *drvc = sdi->driver->context;
|
|
||||||
uint32_t max_samples = transfer->actual_length / sizeof(uint32_t);
|
uint32_t max_samples = transfer->actual_length / sizeof(uint32_t);
|
||||||
struct sr_datafeed_packet packet;
|
struct sr_datafeed_packet packet;
|
||||||
struct sr_datafeed_logic logic;
|
struct sr_datafeed_logic logic;
|
||||||
|
@ -131,13 +161,17 @@ void LIBUSB_CALL h4032l_data_transfer_callback(struct libusb_transfer *transfer)
|
||||||
|
|
||||||
/* Close data receiving. */
|
/* Close data receiving. */
|
||||||
if (devc->remaining_samples == 0) {
|
if (devc->remaining_samples == 0) {
|
||||||
std_session_send_df_end(sdi);
|
|
||||||
usb_source_remove(sdi->session, drvc->sr_ctx);
|
|
||||||
devc->status = H4032L_STATUS_IDLE;
|
|
||||||
if (buffer[number_samples] != H4032L_END_PACKET_MAGIC)
|
if (buffer[number_samples] != H4032L_END_PACKET_MAGIC)
|
||||||
sr_err("Mismatch magic number of end poll.");
|
sr_err("Mismatch magic number of end poll.");
|
||||||
|
|
||||||
|
abort_acquisition(devc);
|
||||||
|
free_transfer(transfer);
|
||||||
} else {
|
} else {
|
||||||
resubmit_transfer(transfer);
|
if (((devc->submitted_transfers - 1) * H4032L_DATA_BUFFER_SIZE) <
|
||||||
|
(int32_t)(devc->remaining_samples * sizeof(uint32_t)))
|
||||||
|
resubmit_transfer(transfer);
|
||||||
|
else
|
||||||
|
free_transfer(transfer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -303,27 +337,57 @@ SR_PRIV int h4032l_start_data_transfers(const struct sr_dev_inst *sdi)
|
||||||
struct dev_context *devc = sdi->priv;
|
struct dev_context *devc = sdi->priv;
|
||||||
struct sr_usb_dev_inst *usb = sdi->conn;
|
struct sr_usb_dev_inst *usb = sdi->conn;
|
||||||
struct libusb_transfer *transfer;
|
struct libusb_transfer *transfer;
|
||||||
|
uint8_t *buffer;
|
||||||
|
unsigned int num_transfers;
|
||||||
|
unsigned int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
transfer = libusb_alloc_transfer(0);
|
devc->submitted_transfers = 0;
|
||||||
|
|
||||||
libusb_fill_bulk_transfer(transfer, usb->devhdl,
|
/*
|
||||||
6 | LIBUSB_ENDPOINT_IN,
|
* Set number of data transfers regarding to size of buffer.
|
||||||
devc->buffer, ARRAY_SIZE(devc->buffer),
|
* FPGA version 0 can't transfer multiple transfers at once.
|
||||||
h4032l_data_transfer_callback,
|
*/
|
||||||
(void *)sdi, H4032L_USB_TIMEOUT);
|
if ((num_transfers = MIN(devc->remaining_samples * sizeof(uint32_t) /
|
||||||
|
H4032L_DATA_BUFFER_SIZE, devc->fpga_version ?
|
||||||
|
H4032L_DATA_TRANSFER_MAX_NUM : 1)) == 0)
|
||||||
|
num_transfers = 1;
|
||||||
|
|
||||||
/* Send prepared usb packet. */
|
g_free(devc->transfers);
|
||||||
if ((ret = libusb_submit_transfer(transfer)) != 0) {
|
devc->transfers = g_try_malloc(sizeof(*devc->transfers) * num_transfers);
|
||||||
sr_err("Failed to submit transfer: %s.",
|
if (!devc->transfers) {
|
||||||
libusb_error_name(ret));
|
sr_err("USB transfers malloc failed.");
|
||||||
devc->status = H4032L_STATUS_IDLE;
|
return SR_ERR_MALLOC;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (devc->status == H4032L_STATUS_IDLE)
|
devc->num_transfers = num_transfers;
|
||||||
free_transfer(transfer);
|
for (i = 0; i < num_transfers; i++) {
|
||||||
|
if (!(buffer = g_malloc(H4032L_DATA_BUFFER_SIZE))) {
|
||||||
|
sr_err("USB transfer buffer malloc failed.");
|
||||||
|
return SR_ERR_MALLOC;
|
||||||
|
}
|
||||||
|
transfer = libusb_alloc_transfer(0);
|
||||||
|
|
||||||
return (ret ? SR_ERR : SR_OK);
|
libusb_fill_bulk_transfer(transfer, usb->devhdl,
|
||||||
|
6 | LIBUSB_ENDPOINT_IN,
|
||||||
|
buffer, H4032L_DATA_BUFFER_SIZE,
|
||||||
|
h4032l_data_transfer_callback,
|
||||||
|
(void *)sdi, H4032L_USB_TIMEOUT);
|
||||||
|
|
||||||
|
/* Send prepared usb packet. */
|
||||||
|
if ((ret = libusb_submit_transfer(transfer)) != 0) {
|
||||||
|
sr_err("Failed to submit transfer: %s.",
|
||||||
|
libusb_error_name(ret));
|
||||||
|
libusb_free_transfer(transfer);
|
||||||
|
g_free(buffer);
|
||||||
|
abort_acquisition(devc);
|
||||||
|
return SR_ERR;
|
||||||
|
}
|
||||||
|
devc->transfers[i] = transfer;
|
||||||
|
devc->submitted_transfers++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SR_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
SR_PRIV int h4032l_start(const struct sr_dev_inst *sdi)
|
SR_PRIV int h4032l_start(const struct sr_dev_inst *sdi)
|
||||||
|
@ -365,6 +429,23 @@ SR_PRIV int h4032l_start(const struct sr_dev_inst *sdi)
|
||||||
return SR_ERR;
|
return SR_ERR;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
devc->transfers = g_malloc0(sizeof(*devc->transfers));
|
||||||
|
if (!devc->transfers) {
|
||||||
|
sr_err("USB start transfer malloc failed.");
|
||||||
|
return SR_ERR_MALLOC;
|
||||||
|
}
|
||||||
|
|
||||||
|
devc->submitted_transfers++;
|
||||||
|
devc->num_transfers = 1;
|
||||||
|
devc->transfers[0] = transfer;
|
||||||
|
|
||||||
|
return SR_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
SR_PRIV int h4032l_stop(struct sr_dev_inst *sdi)
|
||||||
|
{
|
||||||
|
abort_acquisition(sdi->priv);
|
||||||
|
|
||||||
return SR_OK;
|
return SR_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,6 +33,9 @@
|
||||||
#define H4032L_USB_VENDOR 0x04b5
|
#define H4032L_USB_VENDOR 0x04b5
|
||||||
#define H4032L_USB_PRODUCT 0x4032
|
#define H4032L_USB_PRODUCT 0x4032
|
||||||
|
|
||||||
|
#define H4032L_DATA_BUFFER_SIZE (2 * 1024)
|
||||||
|
#define H4032L_DATA_TRANSFER_MAX_NUM 32
|
||||||
|
|
||||||
#define H4032L_CMD_PKT_MAGIC 0x017f
|
#define H4032L_CMD_PKT_MAGIC 0x017f
|
||||||
#define H4032L_STATUS_PACKET_MAGIC 0x2B1A037F
|
#define H4032L_STATUS_PACKET_MAGIC 0x2B1A037F
|
||||||
#define H4032L_START_PACKET_MAGIC 0x2B1A027F
|
#define H4032L_START_PACKET_MAGIC 0x2B1A027F
|
||||||
|
@ -120,10 +123,12 @@ struct h4032l_cmd_pkt {
|
||||||
|
|
||||||
struct dev_context {
|
struct dev_context {
|
||||||
enum h4032l_status status;
|
enum h4032l_status status;
|
||||||
|
int submitted_transfers;
|
||||||
uint32_t remaining_samples;
|
uint32_t remaining_samples;
|
||||||
gboolean acq_aborted;
|
gboolean acq_aborted;
|
||||||
struct h4032l_cmd_pkt cmd_pkt;
|
struct h4032l_cmd_pkt cmd_pkt;
|
||||||
struct libusb_transfer *usb_transfer;
|
unsigned int num_transfers;
|
||||||
|
struct libusb_transfer **transfers;
|
||||||
uint8_t buffer[512];
|
uint8_t buffer[512];
|
||||||
uint64_t capture_ratio;
|
uint64_t capture_ratio;
|
||||||
uint32_t fpga_version;
|
uint32_t fpga_version;
|
||||||
|
@ -135,6 +140,7 @@ SR_PRIV void LIBUSB_CALL h4032l_usb_callback(struct libusb_transfer *transfer);
|
||||||
SR_PRIV void LIBUSB_CALL h4032l_data_transfer_callback(struct libusb_transfer *transfer);
|
SR_PRIV void LIBUSB_CALL h4032l_data_transfer_callback(struct libusb_transfer *transfer);
|
||||||
SR_PRIV int h4032l_start_data_transfers(const struct sr_dev_inst *sdi);
|
SR_PRIV int h4032l_start_data_transfers(const struct sr_dev_inst *sdi);
|
||||||
SR_PRIV int h4032l_start(const struct sr_dev_inst *sdi);
|
SR_PRIV int h4032l_start(const struct sr_dev_inst *sdi);
|
||||||
|
SR_PRIV int h4032l_stop(struct sr_dev_inst *sdi);
|
||||||
SR_PRIV int h4032l_dev_open(struct sr_dev_inst *sdi);
|
SR_PRIV int h4032l_dev_open(struct sr_dev_inst *sdi);
|
||||||
SR_PRIV int h4032l_get_fpga_version(const struct sr_dev_inst *sdi);
|
SR_PRIV int h4032l_get_fpga_version(const struct sr_dev_inst *sdi);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue