aboutsummaryrefslogtreecommitdiffstats
path: root/gui/source/stmdsp/stmdsp.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'gui/source/stmdsp/stmdsp.hpp')
-rw-r--r--gui/source/stmdsp/stmdsp.hpp166
1 files changed, 166 insertions, 0 deletions
diff --git a/gui/source/stmdsp/stmdsp.hpp b/gui/source/stmdsp/stmdsp.hpp
new file mode 100644
index 0000000..efed8a3
--- /dev/null
+++ b/gui/source/stmdsp/stmdsp.hpp
@@ -0,0 +1,166 @@
+/**
+ * @file stmdsp.hpp
+ * @brief Interface for communication with stmdsp device over serial.
+ *
+ * Copyright (C) 2021 Clyne Sullivan
+ *
+ * Distributed under the GNU GPL v3 or later. You should have received a copy of
+ * the GNU General Public License along with this program.
+ * If not, see <https://www.gnu.org/licenses/>.
+ */
+
+#ifndef STMDSP_HPP_
+#define STMDSP_HPP_
+
+#include <serial/serial.h>
+
+#include <cstdint>
+#include <forward_list>
+#include <memory>
+#include <mutex>
+#include <string>
+#include <tuple>
+
+namespace stmdsp
+{
+ /**
+ * The largest possible size of an ADC or DAC sample buffer, as a sample count.
+ * Maximum byte size would be `SAMPLES_MAX * sizeof(XXXsample_t)`.
+ */
+ constexpr unsigned int SAMPLES_MAX = 4096;
+
+ /**
+ * ADC samples on all platforms are stored as 16-bit unsigned integers.
+ */
+ using adcsample_t = uint16_t;
+ /**
+ * DAC samples on all platforms are stored as 16-bit unsigned integers.
+ */
+ using dacsample_t = uint16_t;
+
+ /**
+ * List of all available platforms.
+ * Note that some platforms in this list may not have complete support.
+ */
+ enum class platform {
+ Unknown,
+ H7, /* Some feature support */
+ L4, /* Complete feature support */
+ G4 /* Unsupported, but planned */
+ };
+
+ /**
+ * Run status states, valued to match what the stmdsp firmware reports.
+ */
+ enum class RunStatus : char {
+ Idle = '1', /* Device ready for commands or execution. */
+ Running, /* Device currently executing its algorithm. */
+ Recovering /* Device recovering from fault caused by algorithm. */
+ };
+
+ /**
+ * Error messages that are reported by the firmware.
+ */
+ enum class Error : char {
+ None = 0,
+ BadParam, /* An invalid parameter was passed for a command. */
+ BadParamSize, /* An invaild param. size was given for a command. */
+ BadUserCodeLoad, /* Device failed to load the given algorithm. */
+ BadUserCodeSize, /* The given algorithm is too large for the device. */
+ NotIdle, /* An idle-only command was received while not Idle. */
+ ConversionAborted, /* A conversion was aborted due to a fault. */
+ NotRunning, /* A running-only command was received while not Running. */
+
+ GUIDisconnect = 100 /* The GUI lost connection with the device. */
+ };
+
+ /**
+ * Provides functionality to scan the system for stmdsp devices.
+ * A list of devices is returned, though the GUI only interacts with one
+ * device at a time.
+ */
+ class scanner
+ {
+ public:
+ /**
+ * Scans for connected devices, returning a list of ports with
+ * connected stmdsp devices.
+ */
+ const std::forward_list<std::string>& scan();
+
+ /**
+ * Retrieves the results of the last scan().
+ */
+ const std::forward_list<std::string>& devices() const noexcept {
+ return m_available_devices;
+ }
+
+ private:
+ constexpr static const char *STMDSP_USB_ID =
+#ifndef STMDSP_WIN32
+ "USB VID:PID=0483:5740";
+#else
+ "USB\\VID_0483&PID_5740";
+#endif
+
+ std::forward_list<std::string> m_available_devices;
+ };
+
+ class device
+ {
+ public:
+ device(const std::string& file);
+ ~device();
+
+ bool connected();
+ void disconnect();
+
+ auto get_platform() const { return m_platform; }
+
+ void continuous_set_buffer_size(unsigned int size);
+ unsigned int get_buffer_size() const { return m_buffer_size; }
+
+ void set_sample_rate(unsigned int rate);
+ unsigned int get_sample_rate();
+
+ void continuous_start();
+ void continuous_stop();
+
+ void measurement_start();
+ uint32_t measurement_read();
+
+ std::vector<adcsample_t> continuous_read();
+ std::vector<adcsample_t> continuous_read_input();
+
+ bool siggen_upload(dacsample_t *buffer, unsigned int size);
+ void siggen_start();
+ void siggen_stop();
+
+ bool is_siggening() const { return m_is_siggening; }
+ bool is_running() const { return m_is_running; }
+
+ // buffer is ELF binary
+ void upload_filter(unsigned char *buffer, size_t size);
+ void unload_filter();
+
+ std::pair<RunStatus, Error> get_status();
+
+ private:
+ std::unique_ptr<serial::Serial> m_serial;
+ platform m_platform = platform::Unknown;
+ unsigned int m_buffer_size = SAMPLES_MAX;
+ unsigned int m_sample_rate = 0;
+ bool m_is_siggening = false;
+ bool m_is_running = false;
+ bool m_disconnect_error_flag = false;
+
+ std::mutex m_lock;
+
+ bool try_command(std::basic_string<uint8_t> data);
+ bool try_read(std::basic_string<uint8_t> cmd, uint8_t *dest, unsigned int dest_size);
+ void handle_disconnect();
+ };
+}
+
+#endif // STMDSP_HPP_
+