aboutsummaryrefslogtreecommitdiffstats
path: root/source/stmdsp/stmdsp.cpp
diff options
context:
space:
mode:
authorClyne Sullivan <clyne@clyne-lp.lan>2021-08-08 22:02:52 -0400
committerClyne Sullivan <clyne@clyne-lp.lan>2021-08-08 22:02:52 -0400
commit707b24dd07236243269cf092728f85172e94e8a4 (patch)
treec136716a5fc9ed9cbf570e24f8f6ab715adc73a2 /source/stmdsp/stmdsp.cpp
initial commit
Diffstat (limited to 'source/stmdsp/stmdsp.cpp')
-rw-r--r--source/stmdsp/stmdsp.cpp214
1 files changed, 214 insertions, 0 deletions
diff --git a/source/stmdsp/stmdsp.cpp b/source/stmdsp/stmdsp.cpp
new file mode 100644
index 0000000..b3fc8c3
--- /dev/null
+++ b/source/stmdsp/stmdsp.cpp
@@ -0,0 +1,214 @@
+/**
+ * @file stmdsp.cpp
+ * @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/>.
+ */
+
+#include "stmdsp.hpp"
+
+#include <serial/serial.h>
+
+namespace stmdsp
+{
+ std::list<std::string>& scanner::scan()
+ {
+ auto devices = serial::list_ports();
+ for (auto& device : devices) {
+ if (device.hardware_id.find(STMDSP_USB_ID) != std::string::npos)
+ m_available_devices.emplace_front(device.port);
+ }
+
+ return m_available_devices;
+ }
+
+ device::device(const std::string& file) :
+ m_serial(file, 1000000/*230400*/, serial::Timeout::simpleTimeout(50))
+ {
+ if (m_serial.isOpen()) {
+ m_serial.flush();
+ m_serial.write("i");
+ if (auto id = m_serial.read(7); id.starts_with("stmdsp")) {
+ if (id.back() == 'h')
+ m_platform = platform::H7;
+ else if (id.back() == 'l')
+ m_platform = platform::L4;
+ else
+ m_serial.close();
+ } else {
+ m_serial.close();
+ }
+ }
+ }
+
+ void device::continuous_set_buffer_size(unsigned int size) {
+ if (connected()) {
+ m_buffer_size = size;
+
+ uint8_t request[3] = {
+ 'B',
+ static_cast<uint8_t>(size),
+ static_cast<uint8_t>(size >> 8)
+ };
+ m_serial.write(request, 3);
+ }
+ }
+
+ void device::set_sample_rate(unsigned int id) {
+ if (connected()) {
+ uint8_t request[2] = {
+ 'r',
+ static_cast<uint8_t>(id)
+ };
+ m_serial.write(request, 2);
+ }
+ }
+
+ unsigned int device::get_sample_rate() {
+ unsigned char result = 0xFF;
+
+ if (connected()) {
+ uint8_t request[2] = {
+ 'r', 0xFF
+ };
+ m_serial.write(request, 2);
+ m_serial.read(&result, 1);
+ }
+
+ return result;
+ }
+
+ void device::continuous_start() {
+ if (connected()) {
+ m_serial.write("R");
+ m_is_running = true;
+ }
+ }
+
+ void device::continuous_start_measure() {
+ if (connected()) {
+ m_serial.write("M");
+ m_is_running = true;
+ }
+ }
+
+ uint32_t device::continuous_start_get_measurement() {
+ uint32_t count = 0;
+ if (connected()) {
+ m_serial.write("m");
+ m_serial.read(reinterpret_cast<uint8_t *>(&count), sizeof(uint32_t));
+ }
+
+ return count / 2;
+ }
+
+ std::vector<adcsample_t> device::continuous_read() {
+ if (connected()) {
+ m_serial.write("s");
+ unsigned char sizebytes[2];
+ m_serial.read(sizebytes, 2);
+ unsigned int size = sizebytes[0] | (sizebytes[1] << 8);
+ if (size > 0) {
+ std::vector<adcsample_t> data (size);
+ unsigned int total = size * sizeof(adcsample_t);
+ unsigned int offset = 0;
+
+ while (total > 512) {
+ m_serial.read(reinterpret_cast<uint8_t *>(&data[0]) + offset, 512);
+ m_serial.write("n");
+ offset += 512;
+ total -= 512;
+ }
+ m_serial.read(reinterpret_cast<uint8_t *>(&data[0]) + offset, total);
+ m_serial.write("n");
+ return data;
+
+ }
+ }
+
+ return {};
+ }
+
+ std::vector<adcsample_t> device::continuous_read_input() {
+ if (connected()) {
+ m_serial.write("t");
+ unsigned char sizebytes[2];
+ m_serial.read(sizebytes, 2);
+ unsigned int size = sizebytes[0] | (sizebytes[1] << 8);
+ if (size > 0) {
+ std::vector<adcsample_t> data (size);
+ unsigned int total = size * sizeof(adcsample_t);
+ unsigned int offset = 0;
+
+ while (total > 512) {
+ m_serial.read(reinterpret_cast<uint8_t *>(&data[0]) + offset, 512);
+ m_serial.write("n");
+ offset += 512;
+ total -= 512;
+ }
+ m_serial.read(reinterpret_cast<uint8_t *>(&data[0]) + offset, total);
+ m_serial.write("n");
+ return data;
+
+ }
+ }
+
+ return {};
+ }
+
+ void device::continuous_stop() {
+ if (connected()) {
+ m_serial.write("S");
+ m_is_running = false;
+ }
+ }
+
+ void device::siggen_upload(dacsample_t *buffer, unsigned int size) {
+ if (connected()) {
+ uint8_t request[3] = {
+ 'D',
+ static_cast<uint8_t>(size),
+ static_cast<uint8_t>(size >> 8)
+ };
+ m_serial.write(request, 3);
+
+ m_serial.write((uint8_t *)buffer, size * sizeof(dacsample_t));
+ }
+ }
+
+ void device::siggen_start() {
+ if (connected()) {
+ m_is_siggening = true;
+ m_serial.write("W");
+ }
+ }
+
+ void device::siggen_stop() {
+ if (connected()) {
+ m_is_siggening = false;
+ m_serial.write("w");
+ }
+ }
+
+ void device::upload_filter(unsigned char *buffer, size_t size) {
+ if (connected()) {
+ uint8_t request[3] = {
+ 'E',
+ static_cast<uint8_t>(size),
+ static_cast<uint8_t>(size >> 8)
+ };
+ m_serial.write(request, 3);
+
+ m_serial.write(buffer, size);
+ }
+ }
+
+ void device::unload_filter() {
+ if (connected())
+ m_serial.write("e");
+ }
+}