You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
219 lines
6.0 KiB
C++
219 lines
6.0 KiB
C++
/**
|
|
* @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, 8'000'000, 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() {
|
|
if (connected() && !is_running()) {
|
|
uint8_t request[2] = {
|
|
'r', 0xFF
|
|
};
|
|
m_serial.write(request, 2);
|
|
|
|
unsigned char result = 0xFF;
|
|
m_serial.read(&result, 1);
|
|
m_sample_rate = result;
|
|
}
|
|
|
|
return m_sample_rate;
|
|
}
|
|
|
|
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));
|
|
// TODO
|
|
if (!m_is_running)
|
|
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");
|
|
}
|
|
}
|