Fix output API receive() function

This changes the new output API's recv() to receive(), and has it
return an integer status code. The output of the function, if any,
is stored in a new parameter as a pointer to a newly allocated GString.

All output modules using this API have been adjusted.
This commit is contained in:
Bert Vermeulen 2013-04-27 17:29:46 +02:00
parent 9c48c2e91e
commit 17f63de642
5 changed files with 72 additions and 57 deletions

View File

@ -287,8 +287,8 @@ static int sanity_check_all_output_modules(void)
} }
/* All modules must provide a data or recv API callback. */ /* All modules must provide a data or recv API callback. */
if (!outputs[i]->data && !outputs[i]->recv) { if (!outputs[i]->data && !outputs[i]->receive) {
sr_err("No data/recv in module %d ('%s').", i, d); sr_err("No data/receive in module %d ('%s').", i, d);
errors++; errors++;
} }

View File

@ -454,7 +454,7 @@ struct sr_output_format {
* <code>data_out</code> parameter, so the caller knows not to try * <code>data_out</code> parameter, so the caller knows not to try
* and g_free() it. * and g_free() it.
* *
* Note: This API call is obsolete, use recv() instead. * Note: This API call is obsolete, use receive() instead.
* *
* @param o Pointer to the respective 'struct sr_output'. * @param o Pointer to the respective 'struct sr_output'.
* @param data_in Pointer to the input data buffer. * @param data_in Pointer to the input data buffer.
@ -486,7 +486,7 @@ struct sr_output_format {
* it stored in the <code>data_out</code> and <code>length_out</code> * it stored in the <code>data_out</code> and <code>length_out</code>
* parameters, or NULL if no output was generated. * parameters, or NULL if no output was generated.
* *
* Note: This API call is obsolete, use recv() instead. * Note: This API call is obsolete, use receive() instead.
* *
* @param o Pointer to the respective 'struct sr_output'. * @param o Pointer to the respective 'struct sr_output'.
* @param event_type Type of event that occured. * @param event_type Type of event that occured.
@ -498,9 +498,33 @@ struct sr_output_format {
int (*event) (struct sr_output *o, int event_type, uint8_t **data_out, int (*event) (struct sr_output *o, int event_type, uint8_t **data_out,
uint64_t *length_out); uint64_t *length_out);
GString *(*recv) (struct sr_output *o, const struct sr_dev_inst *sdi, /**
const struct sr_datafeed_packet *packet); * This function is passed a copy of every packed in the data feed.
* Any output generated by the output module in response to the
* packet should be returned in a newly allocated GString
* <code>out</code>, which will be freed by the caller.
*
* Packets not of interest to the output module can just be ignored,
* and the <code>out</code> parameter set to NULL.
*
* @param o Pointer to the respective 'struct sr_output'.
* @param sdi The device instance that generated the packet.
* @param packet The complete packet.
* @param out A pointer where a GString * should be stored if
* the module generates output, or NULL if not.
*
* @return SR_OK upon success, a negative error code otherwise.
*/
int (*receive) (struct sr_output *o, const struct sr_dev_inst *sdi,
const struct sr_datafeed_packet *packet, GString **out);
/**
* This function is called after the caller is finished using
* the output module, and can be used to free any internal
* resources the module may keep.
*
* @return SR_OK upon success, a negative error code otherwise.
*/
int (*cleanup) (struct sr_output *o); int (*cleanup) (struct sr_output *o);
}; };

View File

@ -36,7 +36,6 @@
struct context { struct context {
int num_enabled_probes; int num_enabled_probes;
GPtrArray *probelist; GPtrArray *probelist;
GString *out;
}; };
static int init(struct sr_output *o) static int init(struct sr_output *o)
@ -66,8 +65,6 @@ static int init(struct sr_output *o)
ctx->num_enabled_probes++; ctx->num_enabled_probes++;
} }
ctx->out = g_string_sized_new(512);
return SR_OK; return SR_OK;
} }
@ -188,10 +185,9 @@ static void fancyprint(int unit, int mqflags, float value, GString *out)
g_string_append_c(out, '\n'); g_string_append_c(out, '\n');
} }
static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi, static int receive(struct sr_output *o, const struct sr_dev_inst *sdi,
const struct sr_datafeed_packet *packet) const struct sr_datafeed_packet *packet, GString **out)
{ {
struct context *ctx;
const struct sr_datafeed_analog *analog; const struct sr_datafeed_analog *analog;
struct sr_probe *probe; struct sr_probe *probe;
GSList *l; GSList *l;
@ -200,35 +196,33 @@ static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi,
(void)sdi; (void)sdi;
*out = NULL;
if (!o || !o->sdi) if (!o || !o->sdi)
return NULL; return SR_ERR_ARG;
ctx = o->internal;
g_string_set_size(ctx->out, 0);
switch (packet->type) { switch (packet->type) {
case SR_DF_HEADER:
break;
case SR_DF_FRAME_BEGIN: case SR_DF_FRAME_BEGIN:
g_string_append_printf(ctx->out, "FRAME-BEGIN\n"); *out = g_string_new("FRAME-BEGIN\n");
break; break;
case SR_DF_FRAME_END: case SR_DF_FRAME_END:
g_string_append_printf(ctx->out, "FRAME-END\n"); *out = g_string_new("FRAME-END\n");
break; break;
case SR_DF_ANALOG: case SR_DF_ANALOG:
analog = packet->payload; analog = packet->payload;
fdata = (const float *)analog->data; fdata = (const float *)analog->data;
*out = g_string_sized_new(512);
for (i = 0; i < analog->num_samples; i++) { for (i = 0; i < analog->num_samples; i++) {
for (l = analog->probes, p = 0; l; l = l->next, p++) { for (l = analog->probes, p = 0; l; l = l->next, p++) {
probe = l->data; probe = l->data;
g_string_append_printf(ctx->out, "%s: ", probe->name); g_string_append_printf(*out, "%s: ", probe->name);
fancyprint(analog->unit, analog->mqflags, fancyprint(analog->unit, analog->mqflags,
fdata[i + p], ctx->out); fdata[i + p], *out);
} }
} }
break; break;
} }
return ctx->out; return SR_OK;
} }
static int cleanup(struct sr_output *o) static int cleanup(struct sr_output *o)
@ -240,7 +234,6 @@ static int cleanup(struct sr_output *o)
ctx = o->internal; ctx = o->internal;
g_ptr_array_free(ctx->probelist, 1); g_ptr_array_free(ctx->probelist, 1);
g_string_free(ctx->out, 1);
g_free(ctx); g_free(ctx);
o->internal = NULL; o->internal = NULL;
@ -252,6 +245,6 @@ SR_PRIV struct sr_output_format output_analog = {
.description = "Analog data", .description = "Analog data",
.df_type = SR_DF_ANALOG, .df_type = SR_DF_ANALOG,
.init = init, .init = init,
.recv = receive, .receive = receive,
.cleanup = cleanup .cleanup = cleanup
}; };

View File

@ -92,23 +92,22 @@ static GString *gen_header(const struct sr_dev_inst *sdi, struct context *ctx)
return s; return s;
} }
static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi, static int receive(struct sr_output *o, const struct sr_dev_inst *sdi,
const struct sr_datafeed_packet *packet) const struct sr_datafeed_packet *packet, GString **out)
{ {
struct context *ctx; struct context *ctx;
const struct sr_datafeed_meta *meta; const struct sr_datafeed_meta *meta;
const struct sr_datafeed_logic *logic; const struct sr_datafeed_logic *logic;
const struct sr_config *src; const struct sr_config *src;
GSList *l; GSList *l;
GString *out;
unsigned int i, j; unsigned int i, j;
uint8_t c; uint8_t c;
*out = NULL;
if (!o || !o->sdi) if (!o || !o->sdi)
return NULL; return SR_ERR_ARG;
ctx = o->internal; ctx = o->internal;
out = NULL;
switch (packet->type) { switch (packet->type) {
case SR_DF_META: case SR_DF_META:
meta = packet->payload; meta = packet->payload;
@ -122,21 +121,21 @@ static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi,
logic = packet->payload; logic = packet->payload;
if (ctx->num_samples == 0) { if (ctx->num_samples == 0) {
/* First logic packet in the feed. */ /* First logic packet in the feed. */
out = gen_header(sdi, ctx); *out = gen_header(sdi, ctx);
} else } else
out = g_string_sized_new(512); *out = g_string_sized_new(512);
for (i = 0; i <= logic->length - logic->unitsize; i += logic->unitsize) { for (i = 0; i <= logic->length - logic->unitsize; i += logic->unitsize) {
for (j = 0; j < logic->unitsize; j++) { for (j = 0; j < logic->unitsize; j++) {
/* The OLS format wants the samples presented MSB first. */ /* The OLS format wants the samples presented MSB first. */
c = *((uint8_t *)logic->data + i + logic->unitsize - 1 - j); c = *((uint8_t *)logic->data + i + logic->unitsize - 1 - j);
g_string_append_printf(out, "%02x", c); g_string_append_printf(*out, "%02x", c);
} }
g_string_append_printf(out, "@%"PRIu64"\n", ctx->num_samples++); g_string_append_printf(*out, "@%"PRIu64"\n", ctx->num_samples++);
} }
break; break;
} }
return out; return SR_OK;
} }
static int cleanup(struct sr_output *o) static int cleanup(struct sr_output *o)
@ -158,6 +157,6 @@ SR_PRIV struct sr_output_format output_ols = {
.description = "OpenBench Logic Sniffer", .description = "OpenBench Logic Sniffer",
.df_type = SR_DF_LOGIC, .df_type = SR_DF_LOGIC,
.init = init, .init = init,
.recv = receive, .receive = receive,
.cleanup = cleanup .cleanup = cleanup
}; };

View File

@ -151,12 +151,11 @@ static int init(struct sr_output *o)
return SR_OK; return SR_OK;
} }
static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi, static int receive(struct sr_output *o, const struct sr_dev_inst *sdi,
const struct sr_datafeed_packet *packet) const struct sr_datafeed_packet *packet, GString **out)
{ {
const struct sr_datafeed_logic *logic; const struct sr_datafeed_logic *logic;
struct context *ctx; struct context *ctx;
GString *text;
unsigned int i; unsigned int i;
int p, curbit, prevbit, index; int p, curbit, prevbit, index;
uint8_t *sample; uint8_t *sample;
@ -164,23 +163,23 @@ static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi,
(void)sdi; (void)sdi;
*out = NULL;
if (!o || !o->internal) if (!o || !o->internal)
return NULL; return SR_ERR_ARG;
ctx = o->internal; ctx = o->internal;
if (packet->type == SR_DF_END) { if (packet->type == SR_DF_END) {
text = g_string_sized_new(16); *out = g_string_new("$dumpoff\n$end\n");
g_string_printf(text, "$dumpoff\n$end\n"); return SR_OK;
return text;
} else if (packet->type != SR_DF_LOGIC) } else if (packet->type != SR_DF_LOGIC)
return NULL; return SR_OK;
if (ctx->header) { if (ctx->header) {
/* The header is still here, this must be the first packet. */ /* The header is still here, this must be the first packet. */
text = ctx->header; *out = ctx->header;
ctx->header = NULL; ctx->header = NULL;
} else { } else {
text = g_string_sized_new(512); *out = g_string_sized_new(512);
} }
logic = packet->payload; logic = packet->payload;
@ -199,7 +198,7 @@ static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi,
continue; continue;
/* Output which signal changed to which value. */ /* Output which signal changed to which value. */
g_string_append_printf(text, "#%" PRIu64 "\n%i%c\n", g_string_append_printf(*out, "#%" PRIu64 "\n%i%c\n",
(uint64_t)(((float)samplecount / ctx->samplerate) (uint64_t)(((float)samplecount / ctx->samplerate)
* ctx->period), curbit, (char)('!' + p)); * ctx->period), curbit, (char)('!' + p));
} }
@ -207,7 +206,7 @@ static GString *receive(struct sr_output *o, const struct sr_dev_inst *sdi,
memcpy(ctx->prevsample, sample, ctx->unitsize); memcpy(ctx->prevsample, sample, ctx->unitsize);
} }
return text; return SR_OK;
} }
static int cleanup(struct sr_output *o) static int cleanup(struct sr_output *o)
@ -228,6 +227,6 @@ struct sr_output_format output_vcd = {
.description = "Value Change Dump (VCD)", .description = "Value Change Dump (VCD)",
.df_type = SR_DF_LOGIC, .df_type = SR_DF_LOGIC,
.init = init, .init = init,
.recv = receive, .receive = receive,
.cleanup = cleanup, .cleanup = cleanup,
}; };