2014-01-15 19:48:01 +08:00
|
|
|
/*
|
|
|
|
* This file is part of the DSLogic project.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <glib.h>
|
|
|
|
#include "libsigrok.h"
|
|
|
|
#include "libsigrok-internal.h"
|
|
|
|
|
|
|
|
/* Message logging helpers with subsystem-specific prefix string. */
|
|
|
|
#define LOG_PREFIX "session: "
|
|
|
|
#define sr_log(l, s, args...) sr_log(l, LOG_PREFIX s, ## args)
|
|
|
|
#define sr_spew(s, args...) sr_spew(LOG_PREFIX s, ## args)
|
|
|
|
#define sr_dbg(s, args...) sr_dbg(LOG_PREFIX s, ## args)
|
|
|
|
#define sr_info(s, args...) sr_info(LOG_PREFIX s, ## args)
|
|
|
|
#define sr_warn(s, args...) sr_warn(LOG_PREFIX s, ## args)
|
|
|
|
#define sr_err(s, args...) sr_err(LOG_PREFIX s, ## args)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
*
|
|
|
|
* init, set DSLogic trigger.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup Trigger handling
|
|
|
|
*
|
|
|
|
* init, set DSLogic trigger
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2016-06-10 10:02:08 +08:00
|
|
|
struct ds_trigger *trigger = NULL;
|
2014-01-15 19:48:01 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* recovery trigger to initial status.
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_init(void)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
2015-12-10 21:46:08 +08:00
|
|
|
if (!trigger) {
|
|
|
|
if (!(trigger = g_try_malloc0(sizeof(struct ds_trigger)))) {
|
|
|
|
sr_err("Trigger malloc failed.");
|
|
|
|
return SR_ERR_MALLOC;
|
|
|
|
}
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
trigger->trigger_en = 0;
|
|
|
|
trigger->trigger_mode = SIMPLE_TRIGGER;
|
|
|
|
trigger->trigger_pos = 0;
|
|
|
|
trigger->trigger_stages = 0;
|
|
|
|
|
|
|
|
for (i = 0; i <= TriggerStages; i++) {
|
|
|
|
for (j = 0; j < TriggerProbes; j++) {
|
|
|
|
trigger->trigger0[i][j] = 'X';
|
|
|
|
trigger->trigger1[i][j] = 'X';
|
|
|
|
}
|
|
|
|
trigger->trigger0_count[i] = 0;
|
|
|
|
trigger->trigger1_count[i] = 0;
|
|
|
|
trigger->trigger0_inv[i] = 0;
|
|
|
|
trigger->trigger1_inv[i] = 0;
|
|
|
|
trigger->trigger_logic[i] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
SR_API int ds_trigger_destroy(void)
|
|
|
|
{
|
|
|
|
if (trigger)
|
|
|
|
g_free(trigger);
|
2016-06-10 10:02:08 +08:00
|
|
|
trigger = NULL;
|
2014-01-15 19:48:01 +08:00
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
2015-06-06 22:24:00 +08:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
SR_API struct ds_trigger *ds_trigger_get(void)
|
|
|
|
{
|
|
|
|
return trigger;
|
|
|
|
}
|
|
|
|
|
2014-01-15 19:48:01 +08:00
|
|
|
/**
|
|
|
|
* set trigger based on stage
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_stage_set_value(uint16_t stage, uint16_t probes, char *trigger0, char *trigger1)
|
|
|
|
{
|
|
|
|
assert(stage < TriggerStages);
|
|
|
|
assert(probes <= TriggerProbes);
|
|
|
|
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j< probes; j++) {
|
|
|
|
trigger->trigger0[stage][probes - j - 1] = *(trigger0 + j * 2);
|
|
|
|
trigger->trigger1[stage][probes - j - 1] = *(trigger1 + j * 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
SR_API int ds_trigger_stage_set_logic(uint16_t stage, uint16_t probes, unsigned char trigger_logic)
|
|
|
|
{
|
|
|
|
assert(stage < TriggerStages);
|
|
|
|
assert(probes <= TriggerProbes);
|
|
|
|
|
|
|
|
trigger->trigger_logic[stage] = trigger_logic;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
SR_API int ds_trigger_stage_set_inv(uint16_t stage, uint16_t probes, unsigned char trigger0_inv, unsigned char trigger1_inv)
|
|
|
|
{
|
|
|
|
assert(stage < TriggerStages);
|
|
|
|
assert(probes <= TriggerProbes);
|
|
|
|
|
|
|
|
trigger->trigger0_inv[stage] = trigger0_inv;
|
|
|
|
trigger->trigger1_inv[stage] = trigger1_inv;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
2015-05-21 11:13:50 +08:00
|
|
|
SR_API int ds_trigger_stage_set_count(uint16_t stage, uint16_t probes, uint32_t trigger0_count, uint32_t trigger1_count)
|
2014-01-15 19:48:01 +08:00
|
|
|
{
|
|
|
|
assert(stage < TriggerStages);
|
|
|
|
assert(probes <= TriggerProbes);
|
|
|
|
|
|
|
|
trigger->trigger0_count[stage] = trigger0_count;
|
|
|
|
trigger->trigger1_count[stage] = trigger1_count;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set trigger based on probe
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_probe_set(uint16_t probe, unsigned char trigger0, unsigned char trigger1)
|
|
|
|
{
|
|
|
|
assert(probe < TriggerProbes);
|
|
|
|
|
|
|
|
trigger->trigger0[TriggerStages][probe] = trigger0;
|
|
|
|
trigger->trigger1[TriggerStages][probe] = trigger1;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set trigger stage
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_set_stage(uint16_t stages)
|
|
|
|
{
|
|
|
|
assert(stages <= TriggerStages);
|
|
|
|
|
|
|
|
trigger->trigger_stages = stages;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set trigger position
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_set_pos(uint16_t position)
|
|
|
|
{
|
|
|
|
assert(position <= 100);
|
|
|
|
|
|
|
|
trigger->trigger_pos = position;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
2015-04-17 00:21:14 +08:00
|
|
|
/**
|
|
|
|
* get trigger position
|
|
|
|
*
|
|
|
|
* @return uint16_t trigger_pos.
|
|
|
|
*/
|
|
|
|
SR_API uint16_t ds_trigger_get_pos()
|
|
|
|
{
|
|
|
|
return trigger->trigger_pos;
|
|
|
|
}
|
|
|
|
|
2014-01-15 19:48:01 +08:00
|
|
|
/**
|
|
|
|
* set trigger en
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_set_en(uint16_t enable)
|
|
|
|
{
|
|
|
|
|
|
|
|
trigger->trigger_en = enable;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
2016-05-15 22:51:18 +08:00
|
|
|
/**
|
|
|
|
* get trigger en
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API uint16_t ds_trigger_get_en()
|
|
|
|
{
|
2016-06-10 10:02:08 +08:00
|
|
|
if (trigger == NULL)
|
|
|
|
return 0;
|
|
|
|
else
|
|
|
|
return trigger->trigger_en;
|
2016-05-15 22:51:18 +08:00
|
|
|
}
|
|
|
|
|
2014-01-15 19:48:01 +08:00
|
|
|
/**
|
|
|
|
* set trigger mode
|
|
|
|
*
|
|
|
|
* @return SR_OK upon success.
|
|
|
|
*/
|
|
|
|
SR_API int ds_trigger_set_mode(uint16_t mode)
|
|
|
|
{
|
|
|
|
|
|
|
|
trigger->trigger_mode = mode;
|
|
|
|
|
|
|
|
return SR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_mask0(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t mask = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
mask = (mask << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
mask += (trigger->trigger0[stage][i] == 'X' || trigger->trigger0[stage][i] == 'C');
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_mask1(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t mask = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
mask = (mask << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
mask += ((trigger->trigger1[stage][i] == 'X') | (trigger->trigger1[stage][i] == 'C'));
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_value0(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t value = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
value = (value << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
value += ((trigger->trigger0[stage][i] == '1') | (trigger->trigger0[stage][i] == 'R'));
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_value1(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t value = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
value = (value << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
value += ((trigger->trigger1[stage][i] == '1') | (trigger->trigger1[stage][i] == 'R'));
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_edge0(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t edge = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
edge = (edge << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
edge += ((trigger->trigger0[stage][i] == 'R') | (trigger->trigger0[stage][i] == 'F') |
|
|
|
|
(trigger->trigger0[stage][i] == 'C'));
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return edge;
|
|
|
|
}
|
|
|
|
SR_PRIV uint64_t ds_trigger_get_edge1(uint16_t stage)
|
|
|
|
{
|
|
|
|
assert(stage <= TriggerStages);
|
|
|
|
|
|
|
|
uint64_t edge = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = TriggerProbes - 1; i >= 0 ; i--) {
|
|
|
|
edge = (edge << 1);
|
2017-05-23 22:39:00 +08:00
|
|
|
edge += ((trigger->trigger1[stage][i] == 'R') | (trigger->trigger1[stage][i] == 'F') |
|
|
|
|
(trigger->trigger1[stage][i] == 'C'));
|
2014-01-15 19:48:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return edge;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @} */
|