demo: User-configurable number of probes.

The sigrok and incremental patterns repeat every 8 probes, shifted by
one probe.
This commit is contained in:
Bert Vermeulen 2013-12-24 11:27:40 +01:00
parent bf90d4c666
commit c07f60e73d
1 changed files with 91 additions and 44 deletions

View File

@ -33,11 +33,10 @@
#define LOG_PREFIX "demo" #define LOG_PREFIX "demo"
/* TODO: Number of probes should be configurable. */ #define DEFAULT_NUM_LOGIC_PROBES 8
#define NUM_PROBES 8 #define DEFAULT_NUM_ANALOG_PROBES 4
/* The size of chunks to send through the session bus. */ /* The size of chunks to send through the session bus. */
/* TODO: Should be configurable. */
#define LOGIC_BUFSIZE 4096 #define LOGIC_BUFSIZE 4096
/* Patterns we can generate */ /* Patterns we can generate */
@ -45,14 +44,17 @@ enum {
/** /**
* Spells "sigrok" across 8 probes using '0's (with '1's as * Spells "sigrok" across 8 probes using '0's (with '1's as
* "background") when displayed using the 'bits' output format. * "background") when displayed using the 'bits' output format.
* The pattern is repeasted every 8 probes, shifted to the right
* in time by one bit.
*/ */
PATTERN_SIGROK, PATTERN_SIGROK,
/** Pseudo-)random values on all probes. */ /** Pseudo-random values on all probes. */
PATTERN_RANDOM, PATTERN_RANDOM,
/** /**
* Incrementing number across all probes. * Incrementing number across 8 probes. The pattern is repeasted
* every 8 probes, shifted to the right in time by one bit.
*/ */
PATTERN_INC, PATTERN_INC,
@ -78,12 +80,19 @@ struct dev_context {
uint64_t cur_samplerate; uint64_t cur_samplerate;
uint64_t limit_samples; uint64_t limit_samples;
uint64_t limit_msec; uint64_t limit_msec;
uint8_t sample_generator;
uint64_t samples_counter; uint64_t samples_counter;
void *cb_data;
int64_t starttime; int64_t starttime;
unsigned char logic_data[LOGIC_BUFSIZE];
uint64_t step; uint64_t step;
int32_t num_logic_probes;
unsigned int logic_unitsize;
uint8_t logic_pattern;
unsigned char logic_data[LOGIC_BUFSIZE];
int32_t num_analog_probes;
};
static const int32_t hwopts[] = {
SR_CONF_NUM_LOGIC_PROBES,
SR_CONF_NUM_ANALOG_PROBES,
}; };
static const int hwcaps[] = { static const int hwcaps[] = {
@ -113,15 +122,12 @@ static uint8_t pattern_sigrok[] = {
0xbe, 0xbe, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xbe, 0xbe, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
}; };
/* Private, per-device-instance driver context. */
/* TODO: struct context as with the other drivers. */
/* List of struct sr_dev_inst, maintained by dev_open()/dev_close(). */
SR_PRIV struct sr_dev_driver demo_driver_info; SR_PRIV struct sr_dev_driver demo_driver_info;
static struct sr_dev_driver *di = &demo_driver_info; static struct sr_dev_driver *di = &demo_driver_info;
static int dev_acquisition_stop(struct sr_dev_inst *sdi, void *cb_data); static int dev_acquisition_stop(struct sr_dev_inst *sdi, void *cb_data);
static int dev_clear(void) static int dev_clear(void)
{ {
return std_dev_clear(di, NULL); return std_dev_clear(di, NULL);
@ -134,20 +140,32 @@ static int init(struct sr_context *sr_ctx)
static GSList *scan(GSList *options) static GSList *scan(GSList *options)
{ {
struct sr_dev_inst *sdi;
struct sr_probe *probe;
struct drv_context *drvc; struct drv_context *drvc;
struct dev_context *devc; struct dev_context *devc;
GSList *devices; struct sr_dev_inst *sdi;
int i; struct sr_probe *probe;
struct sr_config *src;
GSList *devices, *l;
int num_logic_probes, num_analog_probes, i;
char probe_name[16]; char probe_name[16];
(void)options;
drvc = di->priv; drvc = di->priv;
devices = NULL; num_logic_probes = DEFAULT_NUM_LOGIC_PROBES;
num_analog_probes = DEFAULT_NUM_ANALOG_PROBES;
for (l = options; l; l = l->next) {
src = l->data;
switch (src->key) {
case SR_CONF_NUM_LOGIC_PROBES:
num_logic_probes = g_variant_get_int32(src->data);
break;
case SR_CONF_NUM_ANALOG_PROBES:
num_analog_probes = g_variant_get_int32(src->data);
break;
}
}
devices = NULL;
sdi = sr_dev_inst_new(0, SR_ST_ACTIVE, "Demo device", NULL, NULL); sdi = sr_dev_inst_new(0, SR_ST_ACTIVE, "Demo device", NULL, NULL);
if (!sdi) { if (!sdi) {
sr_err("Device instance creation failed."); sr_err("Device instance creation failed.");
@ -155,13 +173,20 @@ static GSList *scan(GSList *options)
} }
sdi->driver = di; sdi->driver = di;
for (i = 0; i < NUM_PROBES; i++) { for (i = 0; i < num_logic_probes; i++) {
sprintf(probe_name, "D%d", i); sprintf(probe_name, "D%d", i);
if (!(probe = sr_probe_new(i, SR_PROBE_LOGIC, TRUE, probe_name))) if (!(probe = sr_probe_new(i, SR_PROBE_LOGIC, TRUE, probe_name)))
return NULL; return NULL;
sdi->probes = g_slist_append(sdi->probes, probe); sdi->probes = g_slist_append(sdi->probes, probe);
} }
for (i = 0; i < num_analog_probes; i++) {
sprintf(probe_name, "A%d", i);
if (!(probe = sr_probe_new(i, SR_PROBE_ANALOG, TRUE, probe_name)))
return NULL;
sdi->probes = g_slist_append(sdi->probes, probe);
}
devices = g_slist_append(devices, sdi); devices = g_slist_append(devices, sdi);
drvc->instances = g_slist_append(drvc->instances, sdi); drvc->instances = g_slist_append(drvc->instances, sdi);
@ -169,12 +194,14 @@ static GSList *scan(GSList *options)
sr_err("Device context malloc failed."); sr_err("Device context malloc failed.");
return NULL; return NULL;
} }
devc->cur_samplerate = SR_KHZ(200); devc->cur_samplerate = SR_KHZ(200);
devc->limit_samples = 0; devc->limit_samples = 0;
devc->limit_msec = 0; devc->limit_msec = 0;
devc->sample_generator = PATTERN_SIGROK;
devc->step = 0; devc->step = 0;
devc->num_logic_probes = num_logic_probes;
devc->logic_unitsize = (devc->num_logic_probes + 7) / 8;
devc->logic_pattern = PATTERN_SIGROK;
devc->num_analog_probes = num_analog_probes;
sdi->priv = devc; sdi->priv = devc;
@ -208,10 +235,11 @@ static int cleanup(void)
static int config_get(int id, GVariant **data, const struct sr_dev_inst *sdi, static int config_get(int id, GVariant **data, const struct sr_dev_inst *sdi,
const struct sr_probe_group *probe_group) const struct sr_probe_group *probe_group)
{ {
struct dev_context *const devc = sdi->priv; struct dev_context *devc;
(void)probe_group; (void)probe_group;
devc = sdi->priv;
switch (id) { switch (id) {
case SR_CONF_SAMPLERATE: case SR_CONF_SAMPLERATE:
*data = g_variant_new_uint64(devc->cur_samplerate); *data = g_variant_new_uint64(devc->cur_samplerate);
@ -223,7 +251,13 @@ static int config_get(int id, GVariant **data, const struct sr_dev_inst *sdi,
*data = g_variant_new_uint64(devc->limit_msec); *data = g_variant_new_uint64(devc->limit_msec);
break; break;
case SR_CONF_PATTERN_MODE: case SR_CONF_PATTERN_MODE:
*data = g_variant_new_string(pattern_strings[devc->sample_generator]); *data = g_variant_new_string(pattern_strings[devc->logic_pattern]);
break;
case SR_CONF_NUM_LOGIC_PROBES:
*data = g_variant_new_int32(devc->num_logic_probes);
break;
case SR_CONF_NUM_ANALOG_PROBES:
*data = g_variant_new_int32(devc->num_analog_probes);
break; break;
default: default:
return SR_ERR_NA; return SR_ERR_NA;
@ -264,7 +298,7 @@ static int config_set(int id, GVariant *data, const struct sr_dev_inst *sdi,
ret = SR_ERR; ret = SR_ERR;
for (i = 0; i < ARRAY_SIZE(pattern_strings); i++) { for (i = 0; i < ARRAY_SIZE(pattern_strings); i++) {
if (!strcmp(stropt, pattern_strings[i])) { if (!strcmp(stropt, pattern_strings[i])) {
devc->sample_generator = i; devc->logic_pattern = i;
ret = SR_OK; ret = SR_OK;
break; break;
} }
@ -292,6 +326,10 @@ static int config_list(int key, GVariant **data, const struct sr_dev_inst *sdi,
(void)probe_group; (void)probe_group;
switch (key) { switch (key) {
case SR_CONF_SCAN_OPTIONS:
*data = g_variant_new_fixed_array(G_VARIANT_TYPE_INT32,
hwopts, ARRAY_SIZE(hwopts), sizeof(int32_t));
break;
case SR_CONF_DEVICE_OPTIONS: case SR_CONF_DEVICE_OPTIONS:
*data = g_variant_new_fixed_array(G_VARIANT_TYPE_INT32, *data = g_variant_new_fixed_array(G_VARIANT_TYPE_INT32,
hwcaps, ARRAY_SIZE(hwcaps), sizeof(int32_t)); hwcaps, ARRAY_SIZE(hwcaps), sizeof(int32_t));
@ -313,18 +351,23 @@ static int config_list(int key, GVariant **data, const struct sr_dev_inst *sdi,
return SR_OK; return SR_OK;
} }
static void sample_generator(struct sr_dev_inst *sdi, uint64_t size) static void logic_generator(struct sr_dev_inst *sdi, uint64_t size)
{ {
struct dev_context *devc; struct dev_context *devc;
uint64_t i; uint64_t i, j;
uint8_t pat;
devc = sdi->priv; devc = sdi->priv;
switch (devc->sample_generator) { switch (devc->logic_pattern) {
case PATTERN_SIGROK: case PATTERN_SIGROK:
for (i = 0; i < size; i++) { memset(devc->logic_data, 0x00, size);
devc->logic_data[i] = ~(pattern_sigrok[ for (i = 0; i < size; i += devc->logic_unitsize) {
devc->step++ % sizeof(pattern_sigrok)] >> 1); for (j = 0; j < devc->logic_unitsize; j++) {
pat = pattern_sigrok[(devc->step + j) % sizeof(pattern_sigrok)] >> 1;
devc->logic_data[i + j] = ~pat;
}
devc->step++;
} }
break; break;
case PATTERN_RANDOM: case PATTERN_RANDOM:
@ -332,15 +375,19 @@ static void sample_generator(struct sr_dev_inst *sdi, uint64_t size)
devc->logic_data[i] = (uint8_t)(rand() & 0xff); devc->logic_data[i] = (uint8_t)(rand() & 0xff);
break; break;
case PATTERN_INC: case PATTERN_INC:
for (i = 0; i < size; i++) for (i = 0; i < size; i++) {
devc->logic_data[i] = devc->step++; for (j = 0; j < devc->logic_unitsize; j++) {
devc->logic_data[i + j] = devc->step;
}
devc->step++;
}
break; break;
case PATTERN_ALL_LOW: case PATTERN_ALL_LOW:
case PATTERN_ALL_HIGH: case PATTERN_ALL_HIGH:
/* These were set when the pattern mode was selected. */ /* These were set when the pattern mode was selected. */
break; break;
default: default:
sr_err("Unknown pattern: %d.", devc->sample_generator); sr_err("Unknown pattern: %d.", devc->logic_pattern);
break; break;
} }
} }
@ -352,7 +399,7 @@ static int prepare_data(int fd, int revents, void *cb_data)
struct dev_context *devc; struct dev_context *devc;
struct sr_datafeed_packet packet; struct sr_datafeed_packet packet;
struct sr_datafeed_logic logic; struct sr_datafeed_logic logic;
static uint64_t samples_to_send, expected_samplenum, sending_now; uint64_t samples_to_send, expected_samplenum, sending_now;
int64_t time, elapsed; int64_t time, elapsed;
(void)fd; (void)fd;
@ -374,16 +421,17 @@ static int prepare_data(int fd, int revents, void *cb_data)
} }
while (samples_to_send > 0) { while (samples_to_send > 0) {
sending_now = MIN(samples_to_send, LOGIC_BUFSIZE); sending_now = MIN(samples_to_send, LOGIC_BUFSIZE / devc->logic_unitsize);
samples_to_send -= sending_now; samples_to_send -= sending_now;
sample_generator(sdi, sending_now);
/* Logic */
logic_generator(sdi, sending_now * devc->logic_unitsize);
packet.type = SR_DF_LOGIC; packet.type = SR_DF_LOGIC;
packet.payload = &logic; packet.payload = &logic;
logic.length = sending_now; logic.length = sending_now * devc->logic_unitsize;
logic.unitsize = 1; logic.unitsize = devc->logic_unitsize;
logic.data = devc->logic_data; logic.data = devc->logic_data;
sr_session_send(devc->cb_data, &packet); sr_session_send(sdi, &packet);
devc->samples_counter += sending_now; devc->samples_counter += sending_now;
} }
@ -405,7 +453,6 @@ static int dev_acquisition_start(const struct sr_dev_inst *sdi, void *cb_data)
return SR_ERR_DEV_CLOSED; return SR_ERR_DEV_CLOSED;
devc = sdi->priv; devc = sdi->priv;
devc->cb_data = cb_data;
devc->samples_counter = 0; devc->samples_counter = 0;
/* /*
@ -458,7 +505,7 @@ static int dev_acquisition_stop(struct sr_dev_inst *sdi, void *cb_data)
/* Send last packet. */ /* Send last packet. */
packet.type = SR_DF_END; packet.type = SR_DF_END;
sr_session_send(devc->cb_data, &packet); sr_session_send(sdi, &packet);
return SR_OK; return SR_OK;
} }