mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-31 13:00:26 -04: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
 |