forked from Ivasoft/DSView
1038 lines
33 KiB
C++
Executable File
1038 lines
33 KiB
C++
Executable File
/*
|
|
* This file is part of the DSView project.
|
|
* DSView is based on PulseView.
|
|
*
|
|
* Copyright (C) 2013 DreamSourceLab <support@dreamsourcelab.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "samplingbar.h"
|
|
#include "../extdef.h"
|
|
|
|
#include <assert.h>
|
|
#include <libusb.h>
|
|
#include <QAction>
|
|
#include <QDebug>
|
|
#include <QLabel>
|
|
#include <QAbstractItemView>
|
|
#include <math.h>
|
|
|
|
#include "../devicemanager.h"
|
|
#include "../device/devinst.h"
|
|
#include "../dialogs/deviceoptions.h"
|
|
#include "../dialogs/waitingdialog.h"
|
|
#include "../dialogs/dsmessagebox.h"
|
|
#include "../view/dsosignal.h"
|
|
#include "../dialogs/interval.h"
|
|
#include "../config/appconfig.h"
|
|
|
|
|
|
using std::map;
|
|
using std::max;
|
|
using std::min;
|
|
using std::string;
|
|
|
|
using namespace pv::device;
|
|
|
|
namespace pv {
|
|
namespace toolbars {
|
|
|
|
const QString SamplingBar::RLEString = tr("(RLE)");
|
|
const QString SamplingBar::DIVString = tr(" / div");
|
|
|
|
SamplingBar::SamplingBar(SigSession *session, QWidget *parent) :
|
|
QToolBar("Sampling Bar", parent),
|
|
_session(session),
|
|
_enable(true),
|
|
_sampling(false),
|
|
_device_type(this),
|
|
_device_selector(this),
|
|
_updating_device_selector(false),
|
|
_configure_button(this),
|
|
_sample_count(this),
|
|
_sample_rate(this),
|
|
_updating_sample_rate(false),
|
|
_updating_sample_count(false),
|
|
_run_stop_button(this),
|
|
_instant_button(this),
|
|
_mode_button(this),
|
|
_instant(false)
|
|
{
|
|
setMovable(false);
|
|
setContentsMargins(0,0,0,0);
|
|
layout()->setSpacing(0);
|
|
|
|
_mode_button.setPopupMode(QToolButton::InstantPopup);
|
|
|
|
_device_selector.setSizeAdjustPolicy(DsComboBox::AdjustToContents);
|
|
_sample_rate.setSizeAdjustPolicy(DsComboBox::AdjustToContents);
|
|
_sample_count.setSizeAdjustPolicy(DsComboBox::AdjustToContents);
|
|
_device_selector.setMaximumWidth(ComboBoxMaxWidth);
|
|
|
|
//_run_stop_button.setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
|
|
_run_stop_button.setObjectName(tr("run_stop_button"));
|
|
|
|
QWidget *leftMargin = new QWidget(this);
|
|
leftMargin->setFixedWidth(4);
|
|
addWidget(leftMargin);
|
|
|
|
_device_type.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
addWidget(&_device_type);
|
|
addWidget(&_device_selector);
|
|
_configure_button.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
addWidget(&_configure_button);
|
|
|
|
addWidget(&_sample_count);
|
|
addWidget(new QLabel(tr(" @ ")));
|
|
addWidget(&_sample_rate);
|
|
|
|
_action_single = new QAction(this);
|
|
|
|
_action_repeat = new QAction(this);
|
|
|
|
_mode_menu = new QMenu(this);
|
|
_mode_menu->addAction(_action_single);
|
|
_mode_menu->addAction(_action_repeat);
|
|
_mode_button.setMenu(_mode_menu);
|
|
|
|
_mode_button.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
_mode_action = addWidget(&_mode_button);
|
|
|
|
_run_stop_button.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
_run_stop_action = addWidget(&_run_stop_button);
|
|
_instant_button.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
|
|
_instant_action = addWidget(&_instant_button);
|
|
|
|
set_sampling(false);
|
|
//retranslateUi();
|
|
|
|
connect(&_device_selector, SIGNAL(currentIndexChanged (int)), this, SLOT(on_device_selected()));
|
|
connect(&_configure_button, SIGNAL(clicked()),this, SLOT(on_configure()));
|
|
connect(&_run_stop_button, SIGNAL(clicked()),this, SLOT(on_run_stop()), Qt::DirectConnection);
|
|
connect(&_instant_button, SIGNAL(clicked()), this, SLOT(on_instant_stop()));
|
|
connect(&_sample_count, SIGNAL(currentIndexChanged(int)), this, SLOT(on_samplecount_sel(int)));
|
|
connect(_action_single, SIGNAL(triggered()), this, SLOT(on_mode()));
|
|
connect(_action_repeat, SIGNAL(triggered()), this, SLOT(on_mode()));
|
|
connect(&_sample_rate, SIGNAL(currentIndexChanged(int)), this, SLOT(on_samplerate_sel(int)));
|
|
}
|
|
|
|
void SamplingBar::changeEvent(QEvent *event)
|
|
{
|
|
if (event->type() == QEvent::LanguageChange)
|
|
retranslateUi();
|
|
else if (event->type() == QEvent::StyleChange)
|
|
reStyle();
|
|
QToolBar::changeEvent(event);
|
|
}
|
|
|
|
void SamplingBar::retranslateUi()
|
|
{
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst && dev_inst->dev_inst()) {
|
|
if (dev_inst->name().contains("virtual-demo"))
|
|
_device_type.setText(tr("Demo"));
|
|
else if (dev_inst->name().contains("virtual"))
|
|
_device_type.setText(tr("File"));
|
|
else {
|
|
int usb_speed = LIBUSB_SPEED_HIGH;
|
|
GVariant *gvar = dev_inst->get_config(NULL, NULL, SR_CONF_USB_SPEED);
|
|
if (gvar != NULL) {
|
|
usb_speed = g_variant_get_int32(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
if (usb_speed == LIBUSB_SPEED_HIGH)
|
|
_device_type.setText(tr("USB 2.0"));
|
|
else if (usb_speed == LIBUSB_SPEED_SUPER)
|
|
_device_type.setText(tr("USB 3.0"));
|
|
else
|
|
_device_type.setText(tr("USB UNKNOWN"));
|
|
}
|
|
}
|
|
_configure_button.setText(tr("Options"));
|
|
_mode_button.setText(tr("Mode"));
|
|
|
|
sr_dev_inst *dev_c = _session->get_dev_inst_c();
|
|
|
|
if (_instant) {
|
|
if (dev_c && dev_c->mode == DSO)
|
|
_instant_button.setText(_sampling ? tr("Stop") : tr("Single"));
|
|
else
|
|
_instant_button.setText(_sampling ? tr("Stop") : tr("Instant"));
|
|
_run_stop_button.setText(tr("Start"));
|
|
} else {
|
|
_run_stop_button.setText(_sampling ? tr("Stop") : tr("Start"));
|
|
if (dev_c && dev_c->mode == DSO)
|
|
_instant_button.setText(tr("Single"));
|
|
else
|
|
_instant_button.setText(tr("Instant"));
|
|
}
|
|
|
|
_action_single->setText(tr("&Single"));
|
|
_action_repeat->setText(tr("&Repetitive"));
|
|
}
|
|
|
|
void SamplingBar::reStyle()
|
|
{
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst && dev_inst->dev_inst()) {
|
|
if (dev_inst->name().contains("virtual-demo"))
|
|
_device_type.setIcon(QIcon(":/icons/demo.svg"));
|
|
else if (dev_inst->name().contains("virtual"))
|
|
_device_type.setIcon(QIcon(":/icons/data.svg"));
|
|
else {
|
|
int usb_speed = LIBUSB_SPEED_HIGH;
|
|
GVariant *gvar = dev_inst->get_config(NULL, NULL, SR_CONF_USB_SPEED);
|
|
if (gvar != NULL) {
|
|
usb_speed = g_variant_get_int32(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
if (usb_speed == LIBUSB_SPEED_SUPER)
|
|
_device_type.setIcon(QIcon(":/icons/usb3.svg"));
|
|
else
|
|
_device_type.setIcon(QIcon(":/icons/usb2.svg"));
|
|
|
|
}
|
|
}
|
|
|
|
if (true) {
|
|
QString iconPath = GetIconPath();
|
|
_configure_button.setIcon(QIcon(iconPath+"/params.svg"));
|
|
_mode_button.setIcon(_session->get_run_mode() == pv::SigSession::Single ? QIcon(iconPath+"/modes.svg") :
|
|
QIcon(iconPath+"/moder.svg"));
|
|
_run_stop_button.setIcon(_sampling ? QIcon(iconPath+"/stop.svg") :
|
|
QIcon(iconPath+"/start.svg"));
|
|
_instant_button.setIcon(QIcon(iconPath+"/single.svg"));
|
|
_action_single->setIcon(QIcon(iconPath+"/oneloop.svg"));
|
|
_action_repeat->setIcon(QIcon(iconPath+"/repeat.svg"));
|
|
}
|
|
}
|
|
|
|
void SamplingBar::set_device_list(const std::list<DevInst*> &devices, DevInst *selected)
|
|
{
|
|
int selected_index = -1;
|
|
|
|
assert(selected);
|
|
|
|
_updating_device_selector = true;
|
|
|
|
_device_selector.clear();
|
|
_device_selector_map.clear();
|
|
|
|
for (DevInst *dev_inst : devices) {
|
|
assert(dev_inst);
|
|
const QString title = dev_inst->format_device_title();
|
|
const void *id = dev_inst->get_id();
|
|
assert(id);
|
|
|
|
if (selected == dev_inst)
|
|
selected_index = _device_selector.count();
|
|
|
|
_device_selector_map[id] = dev_inst;
|
|
_device_selector.addItem(title,
|
|
QVariant::fromValue((void*)id));
|
|
}
|
|
int width = _device_selector.sizeHint().width();
|
|
_device_selector.setFixedWidth(min(width+15, _device_selector.maximumWidth()));
|
|
_device_selector.view()->setMinimumWidth(width+30);
|
|
|
|
// The selected device should have been in the list
|
|
assert(selected_index != -1);
|
|
_device_selector.setCurrentIndex(selected_index);
|
|
|
|
update_sample_rate_selector();
|
|
|
|
_updating_device_selector = false;
|
|
}
|
|
|
|
DevInst* SamplingBar::get_selected_device()
|
|
{
|
|
const int index = _device_selector.currentIndex();
|
|
if (index < 0)
|
|
return NULL;
|
|
|
|
const void *const id =
|
|
_device_selector.itemData(index).value<void*>();
|
|
assert(id);
|
|
|
|
auto it = _device_selector_map.find(id);
|
|
if (it == _device_selector_map.end())
|
|
return NULL;
|
|
|
|
return (*it).second;
|
|
}
|
|
|
|
void SamplingBar::on_configure()
|
|
{
|
|
sig_hide_calibration();
|
|
|
|
int ret;
|
|
DevInst *dev_inst = get_selected_device();
|
|
assert(dev_inst);
|
|
|
|
pv::dialogs::DeviceOptions dlg(this, dev_inst);
|
|
ret = dlg.exec();
|
|
if (ret == QDialog::Accepted) {
|
|
sig_device_updated();
|
|
update_sample_rate_selector();
|
|
|
|
GVariant* gvar;
|
|
if (dev_inst->dev_inst()->mode == DSO) {
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_ZERO);
|
|
if (gvar != NULL) {
|
|
bool zero = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
if (zero) {
|
|
zero_adj();
|
|
return;
|
|
}
|
|
}
|
|
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_CALI);
|
|
if (gvar != NULL) {
|
|
bool cali = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
if (cali) {
|
|
sig_show_calibration();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_TEST);
|
|
if (gvar != NULL) {
|
|
bool test = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
if (test) {
|
|
update_sample_rate_selector_value();
|
|
_sample_count.setDisabled(true);
|
|
_sample_rate.setDisabled(true);
|
|
} else {
|
|
_sample_count.setDisabled(false);
|
|
if (dev_inst->dev_inst()->mode != DSO)
|
|
_sample_rate.setDisabled(false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SamplingBar::zero_adj()
|
|
{
|
|
view::DsoSignal *dsoSig = NULL;
|
|
|
|
for(auto &s : _session->get_signals())
|
|
{
|
|
if ((dsoSig = dynamic_cast<view::DsoSignal*>(s)))
|
|
dsoSig->set_enable(true);
|
|
}
|
|
|
|
const int index_back = _sample_count.currentIndex();
|
|
int i = 0;
|
|
|
|
for (i = 0; i < _sample_count.count(); i++)
|
|
if (_sample_count.itemData(i).value<uint64_t>() == ZeroTimeBase)
|
|
break;
|
|
|
|
_sample_count.setCurrentIndex(i);
|
|
commit_hori_res();
|
|
|
|
sig_run_stop();
|
|
|
|
pv::dialogs::WaitingDialog wait(this, _session, SR_CONF_ZERO);
|
|
if (wait.start() == QDialog::Rejected) {
|
|
for(auto &s : _session->get_signals())
|
|
{
|
|
if ((dsoSig = dynamic_cast<view::DsoSignal*>(s)))
|
|
dsoSig->commit_settings();
|
|
}
|
|
}
|
|
|
|
if (_session->get_capture_state() == pv::SigSession::Running)
|
|
on_run_stop();
|
|
|
|
_sample_count.setCurrentIndex(index_back);
|
|
commit_hori_res();
|
|
}
|
|
|
|
bool SamplingBar::get_sampling()
|
|
{
|
|
return _sampling;
|
|
}
|
|
|
|
bool SamplingBar::get_instant()
|
|
{
|
|
return _instant;
|
|
}
|
|
|
|
void SamplingBar::set_sampling(bool sampling)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_sampling_mutex);
|
|
_sampling = sampling;
|
|
|
|
if (!sampling) {
|
|
enable_run_stop(true);
|
|
enable_instant(true);
|
|
} else {
|
|
if (_instant)
|
|
enable_instant(true);
|
|
else
|
|
enable_run_stop(true);
|
|
}
|
|
|
|
_mode_button.setEnabled(!sampling);
|
|
_configure_button.setEnabled(!sampling);
|
|
_device_selector.setEnabled(!sampling);
|
|
|
|
if (true) {
|
|
QString iconPath = GetIconPath();
|
|
if (_instant) {
|
|
_instant_button.setIcon(sampling ? QIcon(iconPath+"/stop.svg") : QIcon(iconPath+"/single.svg"));
|
|
} else {
|
|
_run_stop_button.setIcon(sampling ? QIcon(iconPath+"/stop.svg") : QIcon(iconPath+"/start.svg"));
|
|
}
|
|
_mode_button.setIcon(_session->get_run_mode() == pv::SigSession::Single ? QIcon(iconPath+"/modes.svg") :
|
|
QIcon(iconPath+"/moder.svg"));
|
|
}
|
|
|
|
retranslateUi();
|
|
}
|
|
|
|
void SamplingBar::set_sample_rate(uint64_t sample_rate)
|
|
{
|
|
for (int i = _sample_rate.count() - 1; i >= 0; i--) {
|
|
uint64_t cur_index_sample_rate = _sample_rate.itemData(
|
|
i).value<uint64_t>();
|
|
if (sample_rate >= cur_index_sample_rate) {
|
|
_sample_rate.setCurrentIndex(i);
|
|
break;
|
|
}
|
|
}
|
|
commit_settings();
|
|
}
|
|
|
|
void SamplingBar::update_sample_rate_selector()
|
|
{
|
|
GVariant *gvar_dict, *gvar_list;
|
|
const uint64_t *elements = NULL;
|
|
gsize num_elements;
|
|
|
|
if (_updating_sample_rate)
|
|
return;
|
|
|
|
disconnect(&_sample_rate, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplerate_sel(int)));
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (!dev_inst)
|
|
return;
|
|
|
|
assert(!_updating_sample_rate);
|
|
_updating_sample_rate = true;
|
|
|
|
if (!(gvar_dict = dev_inst->list_config(NULL, SR_CONF_SAMPLERATE)))
|
|
{
|
|
_sample_rate.clear();
|
|
_sample_rate.show();
|
|
_updating_sample_rate = false;
|
|
return;
|
|
}
|
|
|
|
if ((gvar_list = g_variant_lookup_value(gvar_dict,
|
|
"samplerates", G_VARIANT_TYPE("at"))))
|
|
{
|
|
elements = (const uint64_t *)g_variant_get_fixed_array(
|
|
gvar_list, &num_elements, sizeof(uint64_t));
|
|
_sample_rate.clear();
|
|
|
|
for (unsigned int i = 0; i < num_elements; i++)
|
|
{
|
|
char *const s = sr_samplerate_string(elements[i]);
|
|
_sample_rate.addItem(QString(s),
|
|
QVariant::fromValue(elements[i]));
|
|
g_free(s);
|
|
}
|
|
|
|
_sample_rate.show();
|
|
g_variant_unref(gvar_list);
|
|
}
|
|
|
|
_sample_rate.setMinimumWidth(_sample_rate.sizeHint().width()+15);
|
|
_sample_rate.view()->setMinimumWidth(_sample_rate.sizeHint().width()+30);
|
|
|
|
_updating_sample_rate = false;
|
|
g_variant_unref(gvar_dict);
|
|
|
|
update_sample_rate_selector_value();
|
|
connect(&_sample_rate, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplerate_sel(int)));
|
|
|
|
update_sample_count_selector();
|
|
}
|
|
|
|
void SamplingBar::update_sample_rate_selector_value()
|
|
{
|
|
if (_updating_sample_rate)
|
|
return;
|
|
|
|
const uint64_t samplerate = get_selected_device()->get_sample_rate();
|
|
|
|
assert(!_updating_sample_rate);
|
|
_updating_sample_rate = true;
|
|
|
|
if (samplerate != _sample_rate.itemData(
|
|
_sample_rate.currentIndex()).value<uint64_t>()) {
|
|
for (int i = _sample_rate.count() - 1; i >= 0; i--) {
|
|
if (samplerate >= _sample_rate.itemData(
|
|
i).value<uint64_t>()) {
|
|
_sample_rate.setCurrentIndex(i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
_updating_sample_rate = false;
|
|
}
|
|
|
|
void SamplingBar::on_samplerate_sel(int index)
|
|
{
|
|
(void)index;
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst->dev_inst()->mode != DSO)
|
|
update_sample_count_selector();
|
|
}
|
|
|
|
void SamplingBar::update_sample_count_selector()
|
|
{
|
|
bool stream_mode = false;
|
|
uint64_t hw_depth = 0;
|
|
uint64_t sw_depth;
|
|
uint64_t rle_depth = 0;
|
|
uint64_t max_timebase = 0;
|
|
uint64_t min_timebase = SR_NS(10);
|
|
double pre_duration = SR_SEC(1);
|
|
double duration;
|
|
bool rle_support = false;
|
|
|
|
if (_updating_sample_count)
|
|
return;
|
|
|
|
disconnect(&_sample_count, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplecount_sel(int)));
|
|
|
|
assert(!_updating_sample_count);
|
|
_updating_sample_count = true;
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
GVariant* gvar = dev_inst->get_config(NULL, NULL, SR_CONF_STREAM);
|
|
if (gvar != NULL) {
|
|
stream_mode = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_HW_DEPTH);
|
|
if (gvar != NULL) {
|
|
hw_depth = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
|
|
if (dev_inst->dev_inst()->mode == LOGIC) {
|
|
#if defined(__x86_64__) || defined(_M_X64)
|
|
sw_depth = LogicMaxSWDepth64;
|
|
#elif defined(__i386) || defined(_M_IX86)
|
|
int ch_num = _session->get_ch_num(SR_CHANNEL_LOGIC);
|
|
if (ch_num <= 0)
|
|
sw_depth = LogicMaxSWDepth32;
|
|
else
|
|
sw_depth = LogicMaxSWDepth32 / ch_num;
|
|
#endif
|
|
} else {
|
|
sw_depth = AnalogMaxSWDepth;
|
|
}
|
|
|
|
if (dev_inst->dev_inst()->mode == LOGIC) {
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_RLE_SUPPORT);
|
|
if (gvar != NULL) {
|
|
rle_support = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
if (rle_support)
|
|
rle_depth = min(hw_depth*SR_KB(1), sw_depth);
|
|
} else if (dev_inst->dev_inst()->mode == DSO) {
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_MAX_TIMEBASE);
|
|
if (gvar != NULL) {
|
|
max_timebase = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_MIN_TIMEBASE);
|
|
if (gvar != NULL) {
|
|
min_timebase = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
}
|
|
|
|
if (0 != _sample_count.count())
|
|
pre_duration = _sample_count.itemData(
|
|
_sample_count.currentIndex()).value<double>();
|
|
_sample_count.clear();
|
|
const uint64_t samplerate = _sample_rate.itemData(
|
|
_sample_rate.currentIndex()).value<uint64_t>();
|
|
const double hw_duration = hw_depth / (samplerate * (1.0 / SR_SEC(1)));
|
|
if (dev_inst->dev_inst()->mode == DSO)
|
|
duration = max_timebase;
|
|
else if (stream_mode)
|
|
duration = sw_depth / (samplerate * (1.0 / SR_SEC(1)));
|
|
else if (rle_support)
|
|
duration = rle_depth / (samplerate * (1.0 / SR_SEC(1)));
|
|
else
|
|
duration = hw_duration;
|
|
|
|
assert(duration > 0);
|
|
bool not_last = true;
|
|
do {
|
|
QString suffix = (dev_inst->dev_inst()->mode == DSO) ? DIVString :
|
|
(!stream_mode && duration > hw_duration) ? RLEString : "";
|
|
char *const s = sr_time_string(duration);
|
|
_sample_count.addItem(QString(s) + suffix,
|
|
QVariant::fromValue(duration));
|
|
g_free(s);
|
|
|
|
double unit;
|
|
if (duration >= SR_DAY(1))
|
|
unit = SR_DAY(1);
|
|
else if (duration >= SR_HOUR(1))
|
|
unit = SR_HOUR(1);
|
|
else if (duration >= SR_MIN(1))
|
|
unit = SR_MIN(1);
|
|
else
|
|
unit = 1;
|
|
const double log10_duration = pow(10,
|
|
floor(log10(duration / unit)));
|
|
if (duration > 5 * log10_duration * unit)
|
|
duration = 5 * log10_duration * unit;
|
|
else if (duration > 2 * log10_duration * unit)
|
|
duration = 2 * log10_duration * unit;
|
|
else if (duration > log10_duration * unit)
|
|
duration = log10_duration * unit;
|
|
else
|
|
duration = log10_duration > 1 ? duration * 0.5 :
|
|
(unit == SR_DAY(1) ? SR_HOUR(20) :
|
|
unit == SR_HOUR(1) ? SR_MIN(50) :
|
|
unit == SR_MIN(1) ? SR_SEC(50) : duration * 0.5);
|
|
|
|
if (dev_inst->dev_inst()->mode == DSO)
|
|
not_last = duration >= min_timebase;
|
|
else if (dev_inst->dev_inst()->mode == ANALOG)
|
|
not_last = (duration >= SR_MS(100)) &&
|
|
(duration / SR_SEC(1) * samplerate >= SR_KB(1));
|
|
else
|
|
not_last = (duration / SR_SEC(1) * samplerate >= SR_KB(1));
|
|
} while(not_last);
|
|
|
|
_updating_sample_count = true;
|
|
if (pre_duration > _sample_count.itemData(0).value<double>())
|
|
_sample_count.setCurrentIndex(0);
|
|
else if (pre_duration < _sample_count.itemData(_sample_count.count()-1).value<double>())
|
|
_sample_count.setCurrentIndex(_sample_count.count()-1);
|
|
else {
|
|
for (int i = 0; i < _sample_count.count(); i++)
|
|
if (pre_duration >= _sample_count.itemData(
|
|
i).value<double>()) {
|
|
_sample_count.setCurrentIndex(i);
|
|
break;
|
|
}
|
|
}
|
|
_updating_sample_count = false;
|
|
|
|
update_sample_count_selector_value();
|
|
on_samplecount_sel(_sample_count.currentIndex());
|
|
connect(&_sample_count, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplecount_sel(int)));
|
|
}
|
|
|
|
void SamplingBar::update_sample_count_selector_value()
|
|
{
|
|
if (_updating_sample_count)
|
|
return;
|
|
|
|
GVariant* gvar;
|
|
double duration;
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst->dev_inst()->mode == DSO) {
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_TIMEBASE);
|
|
if (gvar != NULL) {
|
|
duration = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
} else {
|
|
qDebug() << "ERROR: config_get SR_CONF_TIMEBASE failed.";
|
|
return;
|
|
}
|
|
} else {
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_LIMIT_SAMPLES);
|
|
if (gvar != NULL) {
|
|
duration = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
} else {
|
|
qDebug() << "ERROR: config_get SR_CONF_TIMEBASE failed.";
|
|
return;
|
|
}
|
|
const uint64_t samplerate = dev_inst->get_sample_rate();
|
|
duration = duration / samplerate * SR_SEC(1);
|
|
}
|
|
assert(!_updating_sample_count);
|
|
_updating_sample_count = true;
|
|
|
|
if (duration != _sample_count.itemData(
|
|
_sample_count.currentIndex()).value<double>()) {
|
|
for (int i = 0; i < _sample_count.count(); i++) {
|
|
if (duration >= _sample_count.itemData(
|
|
i).value<double>()) {
|
|
_sample_count.setCurrentIndex(i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
_updating_sample_count = false;
|
|
}
|
|
|
|
void SamplingBar::on_samplecount_sel(int index)
|
|
{
|
|
(void)index;
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst->dev_inst()->mode == DSO)
|
|
commit_hori_res();
|
|
sig_duration_changed();
|
|
}
|
|
|
|
double SamplingBar::get_hori_res()
|
|
{
|
|
return _sample_count.itemData(_sample_count.currentIndex()).value<double>();
|
|
}
|
|
|
|
double SamplingBar::hori_knob(int dir)
|
|
{
|
|
double hori_res = -1;
|
|
|
|
if (0 == dir) {
|
|
hori_res = commit_hori_res();
|
|
} else if ((dir > 0) && (_sample_count.currentIndex() > 0)) {
|
|
_sample_count.setCurrentIndex(_sample_count.currentIndex() - 1);
|
|
hori_res = commit_hori_res();
|
|
} else if ((dir < 0) && (_sample_count.currentIndex() < _sample_count.count() - 1)) {
|
|
_sample_count.setCurrentIndex(_sample_count.currentIndex() + 1);
|
|
hori_res = commit_hori_res();
|
|
}
|
|
|
|
disconnect(&_sample_count, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplecount_sel(int)));
|
|
|
|
connect(&_sample_count, SIGNAL(currentIndexChanged(int)),
|
|
this, SLOT(on_samplecount_sel(int)));
|
|
|
|
return hori_res;
|
|
}
|
|
|
|
double SamplingBar::commit_hori_res()
|
|
{
|
|
const double hori_res = _sample_count.itemData(
|
|
_sample_count.currentIndex()).value<double>();
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
const uint64_t sample_limit = dev_inst->get_sample_limit();
|
|
GVariant* gvar;
|
|
uint64_t max_sample_rate;
|
|
gvar = dev_inst->get_config(NULL, NULL, SR_CONF_MAX_DSO_SAMPLERATE);
|
|
if (gvar != NULL) {
|
|
max_sample_rate = g_variant_get_uint64(gvar);
|
|
g_variant_unref(gvar);
|
|
} else {
|
|
qDebug() << "ERROR: config_get SR_CONF_MAX_DSO_SAMPLERATE failed.";
|
|
return -1;
|
|
}
|
|
|
|
const uint64_t sample_rate = min((uint64_t)(sample_limit * SR_SEC(1) /
|
|
(hori_res * DS_CONF_DSO_HDIVS)),
|
|
(uint64_t)(max_sample_rate /
|
|
(_session->get_ch_num(SR_CHANNEL_DSO) ? _session->get_ch_num(SR_CHANNEL_DSO) : 1)));
|
|
set_sample_rate(sample_rate);
|
|
|
|
dev_inst->set_config(NULL, NULL, SR_CONF_TIMEBASE,
|
|
g_variant_new_uint64(hori_res));
|
|
|
|
return hori_res;
|
|
}
|
|
|
|
void SamplingBar::commit_settings()
|
|
{
|
|
bool test = false;
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst && dev_inst->owner()) {
|
|
GVariant *gvar = dev_inst->get_config(NULL, NULL, SR_CONF_TEST);
|
|
if (gvar != NULL) {
|
|
test = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
}
|
|
|
|
if (test) {
|
|
update_sample_rate_selector_value();
|
|
update_sample_count_selector_value();
|
|
} else {
|
|
const double sample_duration = _sample_count.itemData(
|
|
_sample_count.currentIndex()).value<double>();
|
|
const uint64_t sample_rate = _sample_rate.itemData(
|
|
_sample_rate.currentIndex()).value<uint64_t>();
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst) {
|
|
if (sample_rate != dev_inst->get_sample_rate())
|
|
dev_inst->set_config(NULL, NULL,
|
|
SR_CONF_SAMPLERATE,
|
|
g_variant_new_uint64(sample_rate));
|
|
if (dev_inst->dev_inst()->mode != DSO) {
|
|
const uint64_t sample_count = ((uint64_t)ceil(sample_duration / SR_SEC(1) *
|
|
sample_rate) + SAMPLES_ALIGN) & ~SAMPLES_ALIGN;
|
|
if (sample_count != dev_inst->get_sample_limit())
|
|
dev_inst->set_config(NULL, NULL,
|
|
SR_CONF_LIMIT_SAMPLES,
|
|
g_variant_new_uint64(sample_count));
|
|
|
|
bool rle_mode = _sample_count.currentText().contains(RLEString);
|
|
dev_inst->set_config(NULL, NULL,
|
|
SR_CONF_RLE,
|
|
g_variant_new_boolean(rle_mode));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//start or stop capture
|
|
void SamplingBar::on_run_stop()
|
|
{
|
|
if (get_sampling() || _session->isRepeating()) {
|
|
_session->exit_capture();
|
|
} else {
|
|
enable_run_stop(false);
|
|
enable_instant(false);
|
|
commit_settings();
|
|
_instant = false;
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (!dev_inst)
|
|
return;
|
|
|
|
if (dev_inst->dev_inst()->mode == DSO) {
|
|
GVariant* gvar = dev_inst->get_config(NULL, NULL, SR_CONF_ZERO);
|
|
if (gvar != NULL) {
|
|
bool zero = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
if (zero) {
|
|
dialogs::DSMessageBox msg(this);
|
|
msg.mBox()->setText(tr("Auto Calibration"));
|
|
msg.mBox()->setInformativeText(tr("Please adjust zero skew and save the result!"));
|
|
//msg.setStandardButtons(QMessageBox::Ok);
|
|
msg.mBox()->addButton(tr("Ok"), QMessageBox::AcceptRole);
|
|
msg.mBox()->addButton(tr("Skip"), QMessageBox::RejectRole);
|
|
msg.mBox()->setIcon(QMessageBox::Warning);
|
|
if (msg.exec()) {
|
|
zero_adj();
|
|
} else {
|
|
dev_inst->set_config(NULL, NULL, SR_CONF_ZERO, g_variant_new_boolean(false));
|
|
enable_run_stop(true);
|
|
enable_instant(true);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
sig_run_stop();
|
|
}
|
|
}
|
|
|
|
void SamplingBar::on_instant_stop()
|
|
{
|
|
if (get_sampling()) {
|
|
_session->set_repeating(false);
|
|
bool wait_upload = false;
|
|
if (_session->get_run_mode() != SigSession::Repetitive) {
|
|
GVariant *gvar = get_selected_device()->get_config(NULL, NULL, SR_CONF_WAIT_UPLOAD);
|
|
if (gvar != NULL) {
|
|
wait_upload = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
}
|
|
if (!wait_upload) {
|
|
_session->stop_capture();
|
|
_session->capture_state_changed(SigSession::Stopped);
|
|
}
|
|
} else {
|
|
enable_run_stop(false);
|
|
enable_instant(false);
|
|
commit_settings();
|
|
_instant = true;
|
|
|
|
DevInst *dev_inst = get_selected_device();
|
|
|
|
if (!dev_inst)
|
|
return;
|
|
|
|
if (dev_inst->dev_inst()->mode == DSO) {
|
|
GVariant* gvar = dev_inst->get_config(NULL, NULL, SR_CONF_ZERO);
|
|
|
|
if (gvar != NULL) {
|
|
bool zero = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
|
|
if (zero) {
|
|
dialogs::DSMessageBox msg(this);
|
|
msg.mBox()->setText(tr("Auto Calibration"));
|
|
msg.mBox()->setInformativeText(tr("Auto Calibration program will be started. Don't connect any probes. It can take a while!"));
|
|
|
|
msg.mBox()->addButton(tr("Ok"), QMessageBox::AcceptRole);
|
|
msg.mBox()->addButton(tr("Skip"), QMessageBox::RejectRole);
|
|
msg.mBox()->setIcon(QMessageBox::Warning);
|
|
if (msg.exec()) {
|
|
zero_adj();
|
|
} else {
|
|
dev_inst->set_config(NULL, NULL, SR_CONF_ZERO, g_variant_new_boolean(false));
|
|
enable_run_stop(true);
|
|
enable_instant(true);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
sig_instant_stop();
|
|
}
|
|
}
|
|
|
|
void SamplingBar::on_device_selected()
|
|
{
|
|
if (_updating_device_selector)
|
|
return;
|
|
|
|
_session->stop_capture();
|
|
_session->session_save();
|
|
|
|
DevInst* dev_inst = get_selected_device();
|
|
if (!dev_inst)
|
|
return;
|
|
|
|
try {
|
|
_session->set_device(dev_inst);
|
|
} catch(QString e) {
|
|
show_session_error(tr("Failed to select ") + dev_inst->dev_inst()->model, e);
|
|
}
|
|
sig_device_selected();
|
|
}
|
|
|
|
void SamplingBar::enable_toggle(bool enable)
|
|
{
|
|
bool test = false;
|
|
DevInst *dev_inst = get_selected_device();
|
|
if (dev_inst && dev_inst->owner()) {
|
|
GVariant *gvar = dev_inst->get_config(NULL, NULL, SR_CONF_TEST);
|
|
if (gvar != NULL) {
|
|
test = g_variant_get_boolean(gvar);
|
|
g_variant_unref(gvar);
|
|
}
|
|
}
|
|
if (!test) {
|
|
_sample_count.setDisabled(!enable);
|
|
if (dev_inst->dev_inst()->mode == DSO)
|
|
_sample_rate.setDisabled(true);
|
|
else
|
|
_sample_rate.setDisabled(!enable);
|
|
} else {
|
|
_sample_count.setDisabled(true);
|
|
_sample_rate.setDisabled(true);
|
|
}
|
|
|
|
if (_session->get_device()->name() == "virtual-session") {
|
|
_sample_count.setDisabled(true);
|
|
_sample_rate.setDisabled(true);
|
|
}
|
|
}
|
|
|
|
void SamplingBar::enable_run_stop(bool enable)
|
|
{
|
|
_run_stop_button.setDisabled(!enable);
|
|
}
|
|
|
|
void SamplingBar::enable_instant(bool enable)
|
|
{
|
|
_instant_button.setDisabled(!enable);
|
|
}
|
|
|
|
void SamplingBar::show_session_error(
|
|
const QString text, const QString info_text)
|
|
{
|
|
dialogs::DSMessageBox msg(this);
|
|
msg.mBox()->setText(text);
|
|
msg.mBox()->setInformativeText(info_text);
|
|
msg.mBox()->setStandardButtons(QMessageBox::Ok);
|
|
msg.mBox()->setIcon(QMessageBox::Warning);
|
|
msg.exec();
|
|
}
|
|
|
|
void SamplingBar::reload()
|
|
{
|
|
QString iconPath = GetIconPath();
|
|
if (_session->get_device()->dev_inst()->mode == LOGIC) {
|
|
if (_session->get_device()->name() == "virtual-session") {
|
|
_mode_action->setVisible(false);
|
|
} else {
|
|
_mode_button.setIcon(_session->get_run_mode() == pv::SigSession::Single ? QIcon(iconPath+"/modes.svg") :
|
|
QIcon(iconPath+"/moder.svg"));
|
|
_mode_action->setVisible(true);
|
|
}
|
|
_run_stop_action->setVisible(true);
|
|
_instant_action->setVisible(true);
|
|
enable_toggle(true);
|
|
} else if (_session->get_device()->dev_inst()->mode == ANALOG) {
|
|
_mode_action->setVisible(false);
|
|
_run_stop_action->setVisible(true);
|
|
_instant_action->setVisible(false);
|
|
enable_toggle(true);
|
|
} else if (_session->get_device()->dev_inst()->mode == DSO) {
|
|
_mode_action->setVisible(false);
|
|
_run_stop_action->setVisible(true);
|
|
_instant_action->setVisible(true);
|
|
enable_toggle(true);
|
|
}
|
|
|
|
retranslateUi();
|
|
reStyle();
|
|
update();
|
|
}
|
|
|
|
void SamplingBar::on_mode()
|
|
{
|
|
QString iconPath = GetIconPath();
|
|
QAction *act = qobject_cast<QAction *>(sender());
|
|
if (act == _action_single) {
|
|
_mode_button.setIcon(QIcon(iconPath+"/modes.svg"));
|
|
_session->set_run_mode(pv::SigSession::Single);
|
|
} else if (act == _action_repeat) {
|
|
_mode_button.setIcon(QIcon(iconPath+"/moder.svg"));
|
|
pv::dialogs::Interval interval_dlg(_session, this);
|
|
interval_dlg.exec();
|
|
_session->set_run_mode(pv::SigSession::Repetitive);
|
|
}
|
|
}
|
|
|
|
} // namespace toolbars
|
|
} // namespace pv
|