mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-11-03 13:11:20 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			505 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			505 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
///////////////////////////////////////////////////////////////////////////////////
 | 
						|
// Copyright (C) 2020, 2022 Edouard Griffiths, F4EXB <f4exb06@gmail.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 as version 3 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 V3 for more details.                               //
 | 
						|
//                                                                               //
 | 
						|
// You should have received a copy of the GNU General Public License             //
 | 
						|
// along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | 
						|
///////////////////////////////////////////////////////////////////////////////////
 | 
						|
 | 
						|
#ifndef INCLUDE_SIGMFFILEINPUT_H
 | 
						|
#define INCLUDE_SIGMFFILEINPUT_H
 | 
						|
 | 
						|
#include <ctime>
 | 
						|
#include <iostream>
 | 
						|
#include <fstream>
 | 
						|
 | 
						|
#include <QString>
 | 
						|
#include <QByteArray>
 | 
						|
#include <QTimer>
 | 
						|
#include <QThread>
 | 
						|
#include <QNetworkRequest>
 | 
						|
 | 
						|
#include "dsp/sigmf_forward.h"
 | 
						|
 | 
						|
#include "dsp/devicesamplesource.h"
 | 
						|
#include "sigmffileinputsettings.h"
 | 
						|
#include "sigmffiledata.h"
 | 
						|
 | 
						|
class QNetworkAccessManager;
 | 
						|
class QNetworkReply;
 | 
						|
class SigMFFileInputWorker;
 | 
						|
class DeviceAPI;
 | 
						|
 | 
						|
class SigMFFileInput : public DeviceSampleSource {
 | 
						|
	Q_OBJECT
 | 
						|
public:
 | 
						|
    /**
 | 
						|
     * Communicate settings
 | 
						|
     */
 | 
						|
	class MsgConfigureSigMFFileInput : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		const SigMFFileInputSettings& getSettings() const { return m_settings; }
 | 
						|
        const QList<QString>& getSettingsKeys() const { return m_settingsKeys; }
 | 
						|
		bool getForce() const { return m_force; }
 | 
						|
 | 
						|
		static MsgConfigureSigMFFileInput* create(const SigMFFileInputSettings& settings, const QList<QString>& settingsKeys, bool force) {
 | 
						|
			return new MsgConfigureSigMFFileInput(settings, settingsKeys, force);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		SigMFFileInputSettings m_settings;
 | 
						|
        QList<QString> m_settingsKeys;
 | 
						|
        bool m_force;
 | 
						|
 | 
						|
		MsgConfigureSigMFFileInput(const SigMFFileInputSettings& settings, const QList<QString>& settingsKeys, bool force) :
 | 
						|
			Message(),
 | 
						|
			m_settings(settings),
 | 
						|
            m_settingsKeys(settingsKeys),
 | 
						|
			m_force(force)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Start/stop track play
 | 
						|
     */
 | 
						|
	class MsgConfigureTrackWork : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		bool isWorking() const { return m_working; }
 | 
						|
 | 
						|
		static MsgConfigureTrackWork* create(bool working) {
 | 
						|
			return new MsgConfigureTrackWork(working);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		bool m_working;
 | 
						|
 | 
						|
		explicit MsgConfigureTrackWork(bool working) :
 | 
						|
			Message(),
 | 
						|
			m_working(working)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
 | 
						|
    /**
 | 
						|
     * Start/stop full file play
 | 
						|
     */
 | 
						|
	class MsgConfigureFileWork : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		bool isWorking() const { return m_working; }
 | 
						|
 | 
						|
		static MsgConfigureFileWork* create(bool working) {
 | 
						|
			return new MsgConfigureFileWork(working);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		bool m_working;
 | 
						|
 | 
						|
		explicit MsgConfigureFileWork(bool working) :
 | 
						|
			Message(),
 | 
						|
			m_working(working)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Move to track
 | 
						|
     */
 | 
						|
	class MsgConfigureTrackIndex : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
        int getTrackIndex() const { return m_trackIndex; }
 | 
						|
 | 
						|
		static MsgConfigureTrackIndex* create(int trackIndex)
 | 
						|
		{
 | 
						|
			return new MsgConfigureTrackIndex(trackIndex);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
        int m_trackIndex;
 | 
						|
 | 
						|
		explicit MsgConfigureTrackIndex(int trackIndex) :
 | 
						|
			Message(),
 | 
						|
            m_trackIndex(trackIndex)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Seek position in track
 | 
						|
     */
 | 
						|
	class MsgConfigureTrackSeek : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		int getMillis() const { return m_seekMillis; }
 | 
						|
 | 
						|
		static MsgConfigureTrackSeek* create(int seekMillis)
 | 
						|
		{
 | 
						|
			return new MsgConfigureTrackSeek(seekMillis);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		int m_seekMillis; //!< millis of seek position from the beginning 0..1000
 | 
						|
 | 
						|
		explicit MsgConfigureTrackSeek(int seekMillis) :
 | 
						|
			Message(),
 | 
						|
			m_seekMillis(seekMillis)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Seek position in full file
 | 
						|
     */
 | 
						|
	class MsgConfigureFileSeek : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		int getMillis() const { return m_seekMillis; }
 | 
						|
 | 
						|
		static MsgConfigureFileSeek* create(int seekMillis)
 | 
						|
		{
 | 
						|
			return new MsgConfigureFileSeek(seekMillis);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		int m_seekMillis; //!< millis of seek position from the beginning 0..1000
 | 
						|
 | 
						|
		explicit MsgConfigureFileSeek(int seekMillis) :
 | 
						|
			Message(),
 | 
						|
			m_seekMillis(seekMillis)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Pull stram timing information
 | 
						|
     */
 | 
						|
	class MsgConfigureFileInputStreamTiming : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
 | 
						|
		static MsgConfigureFileInputStreamTiming* create()
 | 
						|
		{
 | 
						|
			return new MsgConfigureFileInputStreamTiming();
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
 | 
						|
		MsgConfigureFileInputStreamTiming() :
 | 
						|
			Message()
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Start/stop plugin
 | 
						|
     */
 | 
						|
    class MsgStartStop : public Message {
 | 
						|
        MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
    public:
 | 
						|
        bool getStartStop() const { return m_startStop; }
 | 
						|
 | 
						|
        static MsgStartStop* create(bool startStop) {
 | 
						|
            return new MsgStartStop(startStop);
 | 
						|
        }
 | 
						|
 | 
						|
    private:
 | 
						|
        bool m_startStop;
 | 
						|
 | 
						|
        explicit MsgStartStop(bool startStop) :
 | 
						|
            Message(),
 | 
						|
            m_startStop(startStop)
 | 
						|
        { }
 | 
						|
    };
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push start/stop information
 | 
						|
     */
 | 
						|
	class MsgReportStartStop : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		bool getStartStop() const { return m_startStop; }
 | 
						|
 | 
						|
		static MsgReportStartStop* create(bool startStop)
 | 
						|
		{
 | 
						|
			return new MsgReportStartStop(startStop);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		bool m_startStop;
 | 
						|
 | 
						|
		explicit MsgReportStartStop(bool startStop) :
 | 
						|
			Message(),
 | 
						|
			m_startStop(startStop)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push meta data information
 | 
						|
     */
 | 
						|
    class MsgReportMetaData : public Message {
 | 
						|
        MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
    public:
 | 
						|
        const SigMFFileMetaInfo& getMetaInfo() const { return m_metaInfo; }
 | 
						|
        const QList<SigMFFileCapture>& getCaptures() const { return m_captures; }
 | 
						|
 | 
						|
        static MsgReportMetaData* create(const SigMFFileMetaInfo& metaInfo, const QList<SigMFFileCapture>& captures) {
 | 
						|
            return new MsgReportMetaData(metaInfo, captures);
 | 
						|
        }
 | 
						|
 | 
						|
    private:
 | 
						|
        SigMFFileMetaInfo m_metaInfo;
 | 
						|
        QList<SigMFFileCapture> m_captures;
 | 
						|
 | 
						|
        MsgReportMetaData(const SigMFFileMetaInfo& metaInfo, const QList<SigMFFileCapture>& captures) :
 | 
						|
            Message(),
 | 
						|
            m_metaInfo(metaInfo),
 | 
						|
            m_captures(captures)
 | 
						|
        {}
 | 
						|
    };
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push track change
 | 
						|
     */
 | 
						|
    class MsgReportTrackChange : public Message {
 | 
						|
        MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
    public:
 | 
						|
        int getTrackIndex() const { return m_trackIndex; }
 | 
						|
        static MsgReportTrackChange* create(int trackIndex) {
 | 
						|
            return new MsgReportTrackChange(trackIndex);
 | 
						|
        }
 | 
						|
 | 
						|
    private:
 | 
						|
        int m_trackIndex;
 | 
						|
        explicit MsgReportTrackChange(int trackIndex) :
 | 
						|
            Message(),
 | 
						|
            m_trackIndex(trackIndex)
 | 
						|
        { }
 | 
						|
    };
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push stream timing information
 | 
						|
     */
 | 
						|
	class MsgReportFileInputStreamTiming : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
        quint64 getSamplesCount() const { return m_samplesCount; }
 | 
						|
        quint64 getTrackSamplesCount() const { return m_trackSamplesCount; }
 | 
						|
        quint64 getTrackTimeStart() const { return m_trackTimeStart; }
 | 
						|
        int getTrackNumber() const { return m_trackNumber; }
 | 
						|
 | 
						|
        static MsgReportFileInputStreamTiming* create(
 | 
						|
            quint64 samplesCount,
 | 
						|
            quint64 trackSamplesCount,
 | 
						|
            quint64 trackTimeStart,
 | 
						|
            int trackNumber
 | 
						|
        )
 | 
						|
		{
 | 
						|
			return new MsgReportFileInputStreamTiming(
 | 
						|
                samplesCount,
 | 
						|
                trackSamplesCount,
 | 
						|
                trackTimeStart,
 | 
						|
                trackNumber
 | 
						|
            );
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
        quint64 m_samplesCount;
 | 
						|
        quint64 m_trackSamplesCount;
 | 
						|
        quint64 m_trackTimeStart;
 | 
						|
        int m_trackNumber;
 | 
						|
 | 
						|
        MsgReportFileInputStreamTiming(
 | 
						|
            quint64 samplesCount,
 | 
						|
            quint64 trackSamplesCount,
 | 
						|
            quint64 trackTimeStart,
 | 
						|
            int trackNumber
 | 
						|
        ) :
 | 
						|
			Message(),
 | 
						|
			m_samplesCount(samplesCount),
 | 
						|
            m_trackSamplesCount(trackSamplesCount),
 | 
						|
            m_trackTimeStart(trackTimeStart),
 | 
						|
            m_trackNumber(trackNumber)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push CRC (SHA512) information
 | 
						|
     */
 | 
						|
	class MsgReportCRC : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		bool isOK() const { return m_ok; }
 | 
						|
 | 
						|
		static MsgReportCRC* create(bool ok) {
 | 
						|
			return new MsgReportCRC(ok);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		bool m_ok;
 | 
						|
 | 
						|
		explicit MsgReportCRC(bool ok) :
 | 
						|
			Message(),
 | 
						|
			m_ok(ok)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
    /**
 | 
						|
     * Push record total check information
 | 
						|
     */
 | 
						|
	class MsgReportTotalSamplesCheck : public Message {
 | 
						|
		MESSAGE_CLASS_DECLARATION
 | 
						|
 | 
						|
	public:
 | 
						|
		bool isOK() const { return m_ok; }
 | 
						|
 | 
						|
		static MsgReportTotalSamplesCheck* create(bool ok) {
 | 
						|
			return new MsgReportTotalSamplesCheck(ok);
 | 
						|
		}
 | 
						|
 | 
						|
	private:
 | 
						|
		bool m_ok;
 | 
						|
 | 
						|
		explicit MsgReportTotalSamplesCheck(bool ok) :
 | 
						|
			Message(),
 | 
						|
			m_ok(ok)
 | 
						|
		{ }
 | 
						|
	};
 | 
						|
 | 
						|
	explicit SigMFFileInput(DeviceAPI *deviceAPI);
 | 
						|
	~SigMFFileInput() final;
 | 
						|
	void destroy() final;
 | 
						|
 | 
						|
    void init() final;
 | 
						|
	bool start() final;
 | 
						|
	void stop() final;
 | 
						|
 | 
						|
    QByteArray serialize() const final;
 | 
						|
    bool deserialize(const QByteArray& data) final;
 | 
						|
 | 
						|
    void setMessageQueueToGUI(MessageQueue *queue) final { m_guiMessageQueue = queue; }
 | 
						|
	const QString& getDeviceDescription() const final;
 | 
						|
	int getSampleRate() const final;
 | 
						|
    void setSampleRate(int sampleRate) final { (void) sampleRate; }
 | 
						|
	quint64 getCenterFrequency() const final;
 | 
						|
    void setCenterFrequency(qint64 centerFrequency) final;
 | 
						|
    quint64 getStartingTimeStamp() const;
 | 
						|
 | 
						|
	bool handleMessage(const Message& message) final;
 | 
						|
 | 
						|
	int webapiSettingsGet(
 | 
						|
        SWGSDRangel::SWGDeviceSettings& response,
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
	int webapiSettingsPutPatch(
 | 
						|
        bool force,
 | 
						|
        const QStringList& deviceSettingsKeys,
 | 
						|
        SWGSDRangel::SWGDeviceSettings& response, // query + response
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
    int webapiRunGet(
 | 
						|
        SWGSDRangel::SWGDeviceState& response,
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
    int webapiActionsPost(
 | 
						|
        const QStringList& deviceActionsKeys,
 | 
						|
        SWGSDRangel::SWGDeviceActions& query,
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
    int webapiRun(
 | 
						|
        bool run,
 | 
						|
        SWGSDRangel::SWGDeviceState& response,
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
    int webapiReportGet(
 | 
						|
        SWGSDRangel::SWGDeviceReport& response,
 | 
						|
        QString& errorMessage) final;
 | 
						|
 | 
						|
    static void webapiFormatDeviceSettings(
 | 
						|
        SWGSDRangel::SWGDeviceSettings& response,
 | 
						|
        const SigMFFileInputSettings& settings);
 | 
						|
 | 
						|
    static void webapiUpdateDeviceSettings(
 | 
						|
        SigMFFileInputSettings& settings,
 | 
						|
        const QStringList& deviceSettingsKeys,
 | 
						|
        SWGSDRangel::SWGDeviceSettings& response);
 | 
						|
 | 
						|
private:
 | 
						|
	DeviceAPI *m_deviceAPI;
 | 
						|
	QMutex m_mutex;
 | 
						|
    bool m_running = false;
 | 
						|
	SigMFFileInputSettings m_settings;
 | 
						|
	std::ifstream m_metaStream;
 | 
						|
    std::ifstream m_dataStream;
 | 
						|
    SigMFFileMetaInfo m_metaInfo;
 | 
						|
    QList<SigMFFileCapture> m_captures;
 | 
						|
    std::vector<uint64_t> m_captureStarts;
 | 
						|
    bool m_trackMode = false;
 | 
						|
    int m_currentTrackIndex = 0;
 | 
						|
    bool m_recordOpen = false;
 | 
						|
    bool m_crcAvailable = false;
 | 
						|
    bool m_crcOK = false;
 | 
						|
    bool m_recordLengthOK = false;
 | 
						|
    QString m_recordSummary;
 | 
						|
	SigMFFileInputWorker* m_fileInputWorker = nullptr;
 | 
						|
    QThread m_fileInputWorkerThread;
 | 
						|
	QString m_deviceDescription = "SigMFFileInput";
 | 
						|
	int m_sampleRate = 48000;
 | 
						|
	unsigned int m_sampleBytes = 1;
 | 
						|
	quint64 m_centerFrequency = 0;
 | 
						|
    quint64 m_recordLength = 0; //!< record length in seconds computed from file size
 | 
						|
    quint64 m_startingTimeStamp = 0;
 | 
						|
	QTimer m_masterTimer;
 | 
						|
    QNetworkAccessManager *m_networkManager;
 | 
						|
    QNetworkRequest m_networkRequest;
 | 
						|
 | 
						|
    void startWorker();
 | 
						|
    void stopWorker();
 | 
						|
	bool openFileStreams(const QString& fileName);
 | 
						|
    void extractMeta(
 | 
						|
        sigmf::SigMF<sigmf::Global<core::DescrT, sdrangel::DescrT>,
 | 
						|
        sigmf::Capture<core::DescrT, sdrangel::DescrT>,
 | 
						|
        sigmf::Annotation<core::DescrT> >* metaRecord,
 | 
						|
        uint64_t dataFileSize
 | 
						|
    );
 | 
						|
    void extractCaptures(
 | 
						|
        sigmf::SigMF<sigmf::Global<core::DescrT, sdrangel::DescrT>,
 | 
						|
        sigmf::Capture<core::DescrT, sdrangel::DescrT>,
 | 
						|
        sigmf::Annotation<core::DescrT> >* metaRecord
 | 
						|
    );
 | 
						|
    static void analyzeDataType(const std::string& dataTypeString, SigMFFileDataType& dataType);
 | 
						|
    uint64_t getTrackSampleStart(unsigned int trackIndex);
 | 
						|
    int getTrackIndex(uint64_t sampleIndex);
 | 
						|
	void seekFileStream(uint64_t sampleIndex);
 | 
						|
	void seekTrackMillis(int seekMillis);
 | 
						|
    void seekFileMillis(int seekMillis);
 | 
						|
	bool applySettings(const SigMFFileInputSettings& settings, const QList<QString>& settingsKeys, bool force = false);
 | 
						|
    void webapiFormatDeviceReport(SWGSDRangel::SWGDeviceReport& response);
 | 
						|
    void webapiReverseSendSettings(const QList<QString>& deviceSettingsKeys, const SigMFFileInputSettings& settings, bool force);
 | 
						|
    void webapiReverseSendStartStop(bool start);
 | 
						|
 | 
						|
private slots:
 | 
						|
    void networkManagerFinished(QNetworkReply *reply) const;
 | 
						|
};
 | 
						|
 | 
						|
#endif // INCLUDE_SIGMFFILEINPUT_H
 |