2017-10-21 06:11:49 +01:00
|
|
|
#ifndef ISOBUFFER_H
|
|
|
|
#define ISOBUFFER_H
|
|
|
|
|
2019-01-23 05:13:40 +00:00
|
|
|
// TODO: Move headers used only in implementation to isobuffer.cpp
|
2019-01-28 00:14:04 +00:00
|
|
|
#include <memory>
|
|
|
|
|
2017-10-21 06:11:49 +01:00
|
|
|
#include <QWidget>
|
|
|
|
#include <QString>
|
|
|
|
#include <QByteArray>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <QPlainTextEdit>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
|
|
|
|
#include "xmega.h"
|
|
|
|
#include "desktop_settings.h"
|
|
|
|
#include "genericusbdriver.h"
|
|
|
|
|
|
|
|
class isoDriver;
|
2017-11-28 00:17:11 +00:00
|
|
|
class uartStyleDecoder;
|
2018-12-11 09:25:14 +00:00
|
|
|
enum class UartParity : uint8_t;
|
2017-10-21 06:11:49 +01:00
|
|
|
|
2019-01-28 02:39:48 +00:00
|
|
|
enum class TriggerType : uint8_t
|
|
|
|
{
|
|
|
|
Disabled,
|
|
|
|
Rising,
|
|
|
|
Falling
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class TriggerSeekState : uint8_t
|
|
|
|
{
|
|
|
|
BelowTriggerLevel,
|
|
|
|
AboveTriggerLevel
|
|
|
|
};
|
|
|
|
|
2019-01-23 05:13:40 +00:00
|
|
|
// isoBuffer is a generic class that enables O(1) read times (!!!) on all
|
|
|
|
// read/write operations, while maintaining a huge buffer size.
|
|
|
|
// Imagine it as a circular buffer, but with access functions specifically
|
|
|
|
// designed for isochronous data from an Xmega.
|
2017-10-21 06:11:49 +01:00
|
|
|
|
2019-01-23 05:14:24 +00:00
|
|
|
constexpr uint32_t CONSOLE_UPDATE_TIMER_PERIOD = ISO_PACKETS_PER_CTX * 4;
|
2017-10-21 06:11:49 +01:00
|
|
|
|
2019-01-23 05:13:40 +00:00
|
|
|
// TODO: Make private what should be private
|
|
|
|
// TODO: Change integer types to cstdint types
|
2017-10-21 06:11:49 +01:00
|
|
|
class isoBuffer : public QWidget
|
|
|
|
{
|
2019-01-23 05:13:40 +00:00
|
|
|
Q_OBJECT
|
2017-10-21 06:11:49 +01:00
|
|
|
public:
|
2019-01-23 05:13:40 +00:00
|
|
|
isoBuffer(QWidget* parent = 0, int bufferLen = 0, isoDriver* caller = 0, unsigned char channel_value = 0);
|
2019-01-28 00:14:04 +00:00
|
|
|
~isoBuffer() = default;
|
2019-01-23 05:13:40 +00:00
|
|
|
|
|
|
|
// Basic buffer operations
|
|
|
|
short bufferAt(int idx) const;
|
|
|
|
void insertIntoBuffer(short item);
|
|
|
|
void clearBuffer();
|
|
|
|
void gainBuffer(int gain_log);
|
|
|
|
|
|
|
|
// Advanced buffer operations
|
|
|
|
private:
|
|
|
|
template<typename T, typename Function>
|
|
|
|
void writeBuffer(T* data, int len, int TOP, Function transform);
|
|
|
|
public:
|
|
|
|
void writeBuffer_char(char* data, int len);
|
|
|
|
void writeBuffer_short(short* data, int len);
|
|
|
|
|
2019-01-28 00:14:04 +00:00
|
|
|
std::unique_ptr<short[]> readBuffer(double sampleWindow, int numSamples, bool singleBit, double delayOffset);
|
2019-01-23 05:13:40 +00:00
|
|
|
|
|
|
|
// file I/O
|
2019-01-28 00:14:04 +00:00
|
|
|
private:
|
|
|
|
void outputSampleToFile(double averageSample);
|
|
|
|
void maybeOutputSampleToFile(double convertedSample);
|
|
|
|
public:
|
2019-01-23 05:13:40 +00:00
|
|
|
double sampleConvert(short sample, int TOP, bool AC) const;
|
|
|
|
short inverseSampleConvert(double voltageLevel, int TOP, bool AC) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
template<typename Function>
|
|
|
|
int capSample(int offset, int target, double seconds, double value, Function comp);
|
2019-01-28 02:39:48 +00:00
|
|
|
void checkTriggered();
|
2019-01-23 05:13:40 +00:00
|
|
|
public:
|
|
|
|
int cap_x0fromLast(double seconds, double vbot);
|
|
|
|
int cap_x1fromLast(double seconds, int x0, double vbot);
|
|
|
|
int cap_x2fromLast(double seconds, int x1, double vtop);
|
|
|
|
void serialManage(double baudRate, UartParity parity);
|
2019-01-28 02:39:48 +00:00
|
|
|
void setTriggerType(TriggerType newType);
|
|
|
|
void setTriggerLevel(double voltageLevel, uint16_t top, bool acCoupled);
|
2019-01-23 05:13:40 +00:00
|
|
|
|
|
|
|
// ---- MEMBER VARIABLES ----
|
|
|
|
|
2019-01-28 00:14:04 +00:00
|
|
|
// Presentation?
|
|
|
|
// TODO: Add consoles as constructor arguments
|
|
|
|
// NOTE: These are initialized in mainwindow.cpp
|
2019-01-23 05:13:40 +00:00
|
|
|
QPlainTextEdit* m_console1;
|
|
|
|
QPlainTextEdit* m_console2;
|
|
|
|
unsigned char m_channel = 255;
|
|
|
|
bool m_serialAutoScroll = true;
|
|
|
|
|
2019-01-28 00:14:04 +00:00
|
|
|
// Internal Storage
|
|
|
|
std::unique_ptr<short[]> m_buffer;
|
|
|
|
int m_back = 0;
|
|
|
|
int m_insertedCount = 0;
|
|
|
|
int m_bufferEnd;
|
|
|
|
|
2019-01-23 05:13:40 +00:00
|
|
|
// Conversion And Sampling
|
|
|
|
double m_voltage_ref = 1.65;
|
|
|
|
double m_frontendGain = (R4 / (R3 + R4));
|
|
|
|
int m_samplesPerSecond;
|
|
|
|
int m_sampleRate_bit;
|
2019-01-28 02:39:48 +00:00
|
|
|
TriggerType m_triggerType = TriggerType::Disabled;
|
|
|
|
TriggerSeekState m_triggerSeekState = TriggerSeekState::BelowTriggerLevel;
|
|
|
|
short m_triggerLevel = 0;
|
|
|
|
short m_triggerSensitivity = 0;
|
|
|
|
std::vector<uint32_t> m_triggerPositionList = {};
|
2019-01-23 05:13:40 +00:00
|
|
|
// UARTS decoding
|
|
|
|
uartStyleDecoder* m_decoder = NULL;
|
2019-01-28 00:14:04 +00:00
|
|
|
bool m_isDecoding = true;
|
2017-10-21 06:11:49 +01:00
|
|
|
private:
|
2019-01-23 05:13:40 +00:00
|
|
|
// File I/O
|
|
|
|
bool m_fileIOEnabled = false;
|
|
|
|
QFile* m_currentFile;
|
2019-01-28 00:14:04 +00:00
|
|
|
int m_fileIO_sampleCountPerWrite;
|
2019-01-23 05:13:40 +00:00
|
|
|
int m_fileIO_sampleCount;
|
2019-01-28 00:14:04 +00:00
|
|
|
double m_fileIO_sampleAccumulator;
|
|
|
|
qulonglong m_fileIO_maxFileSize;
|
2019-01-23 05:13:40 +00:00
|
|
|
qulonglong m_fileIO_numBytesWritten;
|
|
|
|
unsigned int m_currentColumn = 0;
|
2019-01-28 00:14:04 +00:00
|
|
|
|
2019-01-23 05:13:40 +00:00
|
|
|
isoDriver* m_virtualParent;
|
2017-12-24 03:08:27 +00:00
|
|
|
signals:
|
2019-01-23 05:13:40 +00:00
|
|
|
void fileIOinternalDisable();
|
2017-10-21 06:11:49 +01:00
|
|
|
public slots:
|
2019-01-23 05:13:40 +00:00
|
|
|
void enableFileIO(QFile* file, int samplesToAverage, qulonglong max_file_size);
|
|
|
|
void disableFileIO();
|
2017-10-21 06:11:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // ISOBUFFER_H
|