X-Git-Url: http://sigrok.org/gitweb/?a=blobdiff_plain;ds=sidebyside;f=src%2Finput%2Fbinary.c;h=f58af58d16ac1d35e649a1db7761a0cf3fe495ec;hb=bcf9384d3d2cad0effb5ef2769697d8af05b8a17;hp=11a3b4768f626b2566197e14d243636f809f6e2f;hpb=b84cba4dbf2a78806848f9dbaf10238a18daf735;p=libsigrok.git
diff --git a/src/input/binary.c b/src/input/binary.c
index 11a3b476..f58af58d 100644
--- a/src/input/binary.c
+++ b/src/input/binary.c
@@ -17,20 +17,21 @@
* along with this program. If not, see .
*/
+#include
#include
#include
#include
#include
#include
#include
-#include "libsigrok.h"
+#include
#include "libsigrok-internal.h"
#define LOG_PREFIX "input/binary"
#define MAX_CHUNK_SIZE 4096
#define DEFAULT_NUM_CHANNELS 8
-#define DEFAULT_SAMPLERATE "0"
+#define DEFAULT_SAMPLERATE 0
struct context {
gboolean started;
@@ -39,11 +40,8 @@ struct context {
static int init(struct sr_input *in, GHashTable *options)
{
- struct sr_channel *ch;
struct context *inc;
- uint64_t samplerate;
int num_channels, i;
- const char *s;
char name[16];
num_channels = g_variant_get_int32(g_hash_table_lookup(options, "numchannels"));
@@ -52,26 +50,20 @@ static int init(struct sr_input *in, GHashTable *options)
return SR_ERR_ARG;
}
- s = g_variant_get_string(g_hash_table_lookup(options, "samplerate"), NULL);
- if (sr_parse_sizestring(s, &samplerate) != SR_OK) {
- sr_err("Invalid samplerate '%s'.", s);
- return SR_ERR_ARG;
- }
-
- in->sdi = sr_dev_inst_new(0, SR_ST_ACTIVE, NULL, NULL, NULL);
+ in->sdi = g_malloc0(sizeof(struct sr_dev_inst));
in->priv = inc = g_malloc0(sizeof(struct context));
- inc->samplerate = samplerate;
+
+ inc->samplerate = g_variant_get_uint64(g_hash_table_lookup(options, "samplerate"));
for (i = 0; i < num_channels; i++) {
snprintf(name, 16, "%d", i);
- ch = sr_channel_new(i, SR_CHANNEL_LOGIC, TRUE, name);
- in->sdi->channels = g_slist_append(in->sdi->channels, ch);
+ sr_channel_new(in->sdi, i, SR_CHANNEL_LOGIC, TRUE, name);
}
return SR_OK;
}
-static int receive(const struct sr_input *in, GString *buf)
+static int process_buffer(struct sr_input *in)
{
struct sr_datafeed_packet packet;
struct sr_datafeed_meta meta;
@@ -79,73 +71,91 @@ static int receive(const struct sr_input *in, GString *buf)
struct sr_config *src;
struct context *inc;
gsize chunk_size, i;
- int chunk, num_channels;
+ int chunk;
inc = in->priv;
-
- g_string_append_len(in->buf, buf->str, buf->len);
-
- num_channels = g_slist_length(in->sdi->channels);
-
- std_session_send_df_header(in->sdi, LOG_PREFIX);
- inc->started = TRUE;
-
- packet.type = SR_DF_META;
- packet.payload = &meta;
- src = sr_config_new(SR_CONF_SAMPLERATE, g_variant_new_uint64(inc->samplerate));
- meta.config = g_slist_append(NULL, src);
- sr_session_send(in->sdi, &packet);
- sr_config_free(src);
+ if (!inc->started) {
+ std_session_send_df_header(in->sdi, LOG_PREFIX);
+
+ if (inc->samplerate) {
+ packet.type = SR_DF_META;
+ packet.payload = &meta;
+ src = sr_config_new(SR_CONF_SAMPLERATE, g_variant_new_uint64(inc->samplerate));
+ meta.config = g_slist_append(NULL, src);
+ sr_session_send(in->sdi, &packet);
+ g_slist_free(meta.config);
+ sr_config_free(src);
+ }
+
+ inc->started = TRUE;
+ }
packet.type = SR_DF_LOGIC;
packet.payload = &logic;
- logic.unitsize = (num_channels + 7) / 8;
- logic.data = in->buf->str;
+ logic.unitsize = (g_slist_length(in->sdi->channels) + 7) / 8;
/* Cut off at multiple of unitsize. */
chunk_size = in->buf->len / logic.unitsize * logic.unitsize;
- chunk = 0;
for (i = 0; i < chunk_size; i += chunk) {
- chunk = MAX(MAX_CHUNK_SIZE, chunk_size - i);
+ logic.data = in->buf->str + i;
+ chunk = MIN(MAX_CHUNK_SIZE, chunk_size - i);
logic.length = chunk;
sr_session_send(in->sdi, &packet);
}
-
- if (in->buf->len > chunk_size)
- g_string_erase(in->buf, 0, in->buf->len - chunk_size);
+ g_string_erase(in->buf, 0, chunk_size);
return SR_OK;
}
-static int cleanup(struct sr_input *in)
+static int receive(struct sr_input *in, GString *buf)
+{
+ int ret;
+
+ g_string_append_len(in->buf, buf->str, buf->len);
+
+ if (!in->sdi_ready) {
+ /* sdi is ready, notify frontend. */
+ in->sdi_ready = TRUE;
+ return SR_OK;
+ }
+
+ ret = process_buffer(in);
+
+ return ret;
+}
+
+static int end(struct sr_input *in)
{
- struct sr_datafeed_packet packet;
struct context *inc;
+ struct sr_datafeed_packet packet;
+ int ret;
- inc = in->priv;
+ if (in->sdi_ready)
+ ret = process_buffer(in);
+ else
+ ret = SR_OK;
+ inc = in->priv;
if (inc->started) {
packet.type = SR_DF_END;
sr_session_send(in->sdi, &packet);
}
- g_free(in->priv);
- in->priv = NULL;
- return SR_OK;
+ return ret;
}
static struct sr_option options[] = {
{ "numchannels", "Number of channels", "Number of channels", NULL, NULL },
{ "samplerate", "Sample rate", "Sample rate", NULL, NULL },
- { 0 }
+ ALL_ZERO
};
-static struct sr_option *get_options(void)
+static const struct sr_option *get_options(void)
{
if (!options[0].def) {
options[0].def = g_variant_ref_sink(g_variant_new_int32(DEFAULT_NUM_CHANNELS));
- options[1].def = g_variant_ref_sink(g_variant_new_string(DEFAULT_SAMPLERATE));
+ options[1].def = g_variant_ref_sink(g_variant_new_uint64(DEFAULT_SAMPLERATE));
}
return options;
@@ -155,8 +165,9 @@ SR_PRIV struct sr_input_module input_binary = {
.id = "binary",
.name = "Binary",
.desc = "Raw binary",
+ .exts = NULL,
.options = get_options,
.init = init,
.receive = receive,
- .cleanup = cleanup,
+ .end = end,
};