Files @ 71441196d1c5
Branch filter:

Location: tempo-plotter/src/asciireader.cpp - annotation

Hasan Yavuz ÖZDERYA
Merge with default
ecd02467c397
233fe22d52af
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ce36704ede6f
ce36704ede6f
ce36704ede6f
2a4985e6a454
2a4985e6a454
ecd02467c397
ecd02467c397
ecd02467c397
2a4985e6a454
2a4985e6a454
1dd76148084c
ce36704ede6f
ecd02467c397
ce36704ede6f
ce36704ede6f
2a4985e6a454
2a4985e6a454
2a4985e6a454
2a4985e6a454
ce36704ede6f
ce36704ede6f
ce36704ede6f
ce36704ede6f
ce36704ede6f
50d42af49411
1dd76148084c
1dd76148084c
1dd76148084c
1dd76148084c
1dd76148084c
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
ecd02467c397
70d0b81a44a5
70d0b81a44a5
70d0b81a44a5
70d0b81a44a5
70d0b81a44a5
70d0b81a44a5
70d0b81a44a5
81a003dc29e2
81a003dc29e2
81a003dc29e2
81a003dc29e2
81a003dc29e2
81a003dc29e2
233fe22d52af
233fe22d52af
81a003dc29e2
81a003dc29e2
233fe22d52af
ecd02467c397
233fe22d52af
233fe22d52af
ecd02467c397
ecd02467c397
233fe22d52af
233fe22d52af
233fe22d52af
ecd02467c397
50d42af49411
65256d18c617
50d42af49411
65256d18c617
50d42af49411
50d42af49411
50d42af49411
ecd02467c397
ecd02467c397
ecd02467c397
fa8cb0995e3e
ecd02467c397
ecd02467c397
50d42af49411
ecd02467c397
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
934c1d1050b2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
ce36704ede6f
eab37541abf2
eab37541abf2
ecd02467c397
5d1e33f3fcdc
eab37541abf2
193a625a7915
5d1e33f3fcdc
ecd02467c397
233fe22d52af
233fe22d52af
ecd02467c397
5b3f4916ebcf
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
eab37541abf2
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
5b3f4916ebcf
/*
  Copyright © 2019 Hasan Yavuz Özderya

  This file is part of serialplot.

  serialplot 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 3 of the License, or
  (at your option) any later version.

  serialplot 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 serialplot.  If not, see <http://www.gnu.org/licenses/>.
*/

#include <QtDebug>

#include "asciireader.h"

/// If set to this value number of channels is determined from input
#define NUMOFCHANNELS_AUTO   (0)

AsciiReader::AsciiReader(QIODevice* device, QObject* parent) :
    AbstractReader(device, parent)
{
    paused = false;

    _numChannels = _settingsWidget.numOfChannels();
    autoNumOfChannels = (_numChannels == NUMOFCHANNELS_AUTO);
    delimiter = _settingsWidget.delimiter();

    connect(&_settingsWidget, &AsciiReaderSettings::numOfChannelsChanged,
            [this](unsigned value)
            {
                _numChannels = value;
                updateNumChannels(); // TODO: setting numchannels = 0, should remove all buffers
                                     // do we want this?
                autoNumOfChannels = (_numChannels == NUMOFCHANNELS_AUTO);
                if (!autoNumOfChannels)
                {
                    emit numOfChannelsChanged(value);
                }
            });

    connect(&_settingsWidget, &AsciiReaderSettings::delimiterChanged,
            [this](QChar d)
            {
                delimiter = d;
            });
}

QWidget* AsciiReader::settingsWidget()
{
    return &_settingsWidget;
}

unsigned AsciiReader::numChannels() const
{
    // TODO: an alternative is to never set _numChannels to '0'
    // do not allow '0'
    return _numChannels == 0 ? 1 : _numChannels;
}

void AsciiReader::enable(bool enabled)
{
    if (enabled)
    {
        firstReadAfterEnable = true;
    }

    AbstractReader::enable(enabled);
}

unsigned AsciiReader::readData()
{
    unsigned numBytesRead = 0;

    while(_device->canReadLine())
    {
        QByteArray bytes = _device->readLine();
        QString line = QString(bytes);
        numBytesRead += bytes.size();

        // discard only once when we just started reading
        if (firstReadAfterEnable)
        {
            firstReadAfterEnable = false;
            continue;
        }

        // discard data if paused
        if (paused)
        {
            continue;
        }

        // parse data
        line = line.trimmed();

        // Note: When data coming from pseudo terminal is buffered by
        // system CR is converted to LF for some reason. This causes
        // empty lines in the input when the port is just opened.
        if (line.isEmpty())
        {
            continue;
        }

        const SamplePack* samples = parseLine(line);
        if (samples != nullptr) {
            // update number of channels if in auto mode
            if (autoNumOfChannels ) {
                unsigned nc = samples->numChannels();
                if (nc != _numChannels) {
                    _numChannels = nc;
                    updateNumChannels();
                    // TODO: is `numOfChannelsChanged` signal still used?
                    emit numOfChannelsChanged(nc);
                }
            }

            Q_ASSERT(samples->numChannels() == _numChannels);

            // commit data
            feedOut(*samples);
            delete samples;
        }
    }

    return numBytesRead;
}

SamplePack* AsciiReader::parseLine(const QString& line) const
{
    auto separatedValues = line.split(delimiter, QString::SkipEmptyParts);
    unsigned numComingChannels = separatedValues.length();

    // check number of channels (skipped if auto num channels is enabled)
    if ((!numComingChannels) || (!autoNumOfChannels && numComingChannels != _numChannels))
    {
        qWarning() << "Line parsing error: invalid number of channels!";
        qWarning() << "Read line: " << line;
        return nullptr;
    }

    // parse data per channel
    auto samples = new SamplePack(1, numComingChannels);
    for (unsigned ci = 0; ci < numComingChannels; ci++)
    {
        bool ok;
        samples->data(ci)[0] = separatedValues[ci].toDouble(&ok);
        if (!ok)
        {
            qWarning() << "Data parsing error for channel: " << ci;
            qWarning() << "Read line: " << line;

            delete samples;
            return nullptr;
        }
    }

    return samples;
}

void AsciiReader::saveSettings(QSettings* settings)
{
    _settingsWidget.saveSettings(settings);
}

void AsciiReader::loadSettings(QSettings* settings)
{
    _settingsWidget.loadSettings(settings);
}