mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-19 14:25:27 +00:00
ea8e3006db
Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
604 lines
15 KiB
C++
604 lines
15 KiB
C++
// SPDX-License-Identifier: GPL-2.0
|
|
#include <errno.h>
|
|
|
|
#include <QtBluetooth/QBluetoothAddress>
|
|
#include <QtBluetooth/QBluetoothSocket>
|
|
#include <QEventLoop>
|
|
#include <QTimer>
|
|
#include <QDebug>
|
|
|
|
#include <libdivecomputer/version.h>
|
|
#include <libdivecomputer/context.h>
|
|
|
|
#if defined(SSRF_CUSTOM_IO)
|
|
|
|
#if defined(Q_OS_WIN)
|
|
#include <winsock2.h>
|
|
#include <windows.h>
|
|
#include <ws2bth.h>
|
|
#endif
|
|
|
|
#include <libdivecomputer/custom_io.h>
|
|
|
|
#ifdef BLE_SUPPORT
|
|
# include "qt-ble.h"
|
|
#endif
|
|
|
|
QList<QBluetoothUuid> registeredUuids;
|
|
|
|
static QBluetoothUuid getBtUuid()
|
|
{
|
|
return registeredUuids.first();
|
|
}
|
|
|
|
void addBtUuid(QBluetoothUuid uuid)
|
|
{
|
|
registeredUuids << uuid;
|
|
}
|
|
|
|
extern "C" {
|
|
typedef struct qt_serial_t {
|
|
/*
|
|
* RFCOMM socket used for Bluetooth Serial communication.
|
|
*/
|
|
#if defined(Q_OS_WIN)
|
|
SOCKET socket;
|
|
#else
|
|
QBluetoothSocket *socket;
|
|
#endif
|
|
long timeout;
|
|
} qt_serial_t;
|
|
|
|
#ifdef BLE_SUPPORT
|
|
|
|
static dc_status_t ble_serial_open(dc_custom_io_t *io, dc_context_t *, const char* devaddr);
|
|
static dc_status_t ble_serial_close(dc_custom_io_t *io);
|
|
static dc_status_t ble_serial_read(dc_custom_io_t *io, void* data, size_t size, size_t *actual);
|
|
static dc_status_t ble_serial_write(dc_custom_io_t *io, const void* data, size_t size, size_t *actual);
|
|
static dc_status_t ble_serial_purge(dc_custom_io_t *io, dc_direction_t queue);
|
|
static dc_status_t ble_serial_get_available(dc_custom_io_t *io, size_t *available);
|
|
static dc_status_t ble_serial_set_timeout(dc_custom_io_t *io, long timeout);
|
|
|
|
static dc_custom_io_t ble_serial_ops = {
|
|
.userdata = NULL,
|
|
.user_device = NULL,
|
|
|
|
.serial_open = ble_serial_open,
|
|
.serial_close = ble_serial_close,
|
|
.serial_read = ble_serial_read,
|
|
.serial_write = ble_serial_write,
|
|
.serial_purge = ble_serial_purge,
|
|
.serial_get_available = ble_serial_get_available,
|
|
.serial_set_timeout = ble_serial_set_timeout,
|
|
// These doesn't make sense over bluetooth
|
|
// NULL means NOP
|
|
.serial_configure = NULL,
|
|
.serial_set_dtr = NULL,
|
|
.serial_set_rts = NULL,
|
|
.serial_set_halfduplex = NULL,
|
|
.serial_set_break = NULL,
|
|
|
|
.packet_size = 20,
|
|
.packet_open = qt_ble_open,
|
|
.packet_close = qt_ble_close,
|
|
.packet_read = qt_ble_read,
|
|
.packet_write = qt_ble_write,
|
|
};
|
|
|
|
|
|
static dc_status_t ble_serial_open(dc_custom_io_t *io, dc_context_t *context, const char* devaddr)
|
|
{
|
|
dc_context_set_custom_io(context, &ble_serial_ops, io->user_device);
|
|
return qt_ble_open(&ble_serial_ops, context, devaddr);
|
|
}
|
|
|
|
#define BUFSZ 1024
|
|
static struct {
|
|
unsigned int out_bytes, in_bytes, in_pos;
|
|
unsigned char in[BUFSIZ];
|
|
unsigned char out[BUFSIZ];
|
|
} buffer;
|
|
|
|
static dc_status_t ble_serial_flush_write(void)
|
|
{
|
|
int bytes = buffer.out_bytes;
|
|
|
|
if (!bytes)
|
|
return DC_STATUS_SUCCESS;
|
|
buffer.out_bytes = 0;
|
|
return ble_serial_ops.packet_write(&ble_serial_ops, buffer.out, bytes, NULL);
|
|
}
|
|
|
|
static dc_status_t ble_serial_flush_read(void)
|
|
{
|
|
buffer.in_bytes = buffer.in_pos = 0;
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t ble_serial_close(dc_custom_io_t *io)
|
|
{
|
|
ble_serial_flush_write();
|
|
io->userdata = NULL;
|
|
return qt_ble_close(&ble_serial_ops);
|
|
}
|
|
|
|
static dc_status_t ble_serial_read(dc_custom_io_t *io, void* data, size_t size, size_t *actual)
|
|
{
|
|
Q_UNUSED(io)
|
|
size_t len;
|
|
|
|
if (buffer.in_pos >= buffer.in_bytes) {
|
|
dc_status_t rc;
|
|
size_t received;
|
|
|
|
ble_serial_flush_write();
|
|
rc = ble_serial_ops.packet_read(&ble_serial_ops, buffer.in, sizeof(buffer.in), &received);
|
|
if (rc != DC_STATUS_SUCCESS)
|
|
return rc;
|
|
if (!received)
|
|
return DC_STATUS_IO;
|
|
buffer.in_pos = 0;
|
|
buffer.in_bytes = received;
|
|
}
|
|
|
|
len = buffer.in_bytes - buffer.in_pos;
|
|
if (len > size)
|
|
len = size;
|
|
|
|
memcpy(data, buffer.in + buffer.in_pos, len);
|
|
buffer.in_pos += len;
|
|
if (actual)
|
|
*actual = len;
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t ble_serial_write(dc_custom_io_t *io, const void* data, size_t size, size_t *actual)
|
|
{
|
|
Q_UNUSED(io)
|
|
dc_status_t rc = DC_STATUS_SUCCESS;
|
|
size_t transferred = 0;
|
|
|
|
ble_serial_flush_read();
|
|
while (size) {
|
|
size_t len = sizeof(buffer.out) - buffer.out_bytes;
|
|
|
|
if (len > size)
|
|
len = size;
|
|
memcpy(buffer.out + buffer.out_bytes, data, len);
|
|
buffer.out_bytes += len;
|
|
|
|
if (buffer.out_bytes == sizeof(buffer.out)) {
|
|
rc = ble_serial_flush_write();
|
|
if (rc != DC_STATUS_SUCCESS)
|
|
break;
|
|
}
|
|
transferred += len;
|
|
data = (const void *) (len + (const char *)data);
|
|
size -= len;
|
|
}
|
|
if (actual)
|
|
*actual = transferred;
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t ble_serial_purge(dc_custom_io_t *io, dc_direction_t queue)
|
|
{
|
|
Q_UNUSED(io)
|
|
Q_UNUSED(queue)
|
|
/* Do we care? */
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t ble_serial_get_available(dc_custom_io_t *io, size_t *available)
|
|
{
|
|
Q_UNUSED(io)
|
|
*available = buffer.in_bytes - buffer.in_pos;
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t ble_serial_set_timeout(dc_custom_io_t *io, long timeout)
|
|
{
|
|
Q_UNUSED(io)
|
|
Q_UNUSED(timeout)
|
|
/* Do we care? */
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static dc_status_t qt_serial_open(dc_custom_io_t *io, dc_context_t *context, const char* devaddr)
|
|
{
|
|
#ifdef BLE_SUPPORT
|
|
if (!strncmp(devaddr, "LE:", 3))
|
|
return ble_serial_open(io, context, devaddr);
|
|
#endif
|
|
|
|
// Allocate memory.
|
|
qt_serial_t *serial_port = (qt_serial_t *) malloc (sizeof (qt_serial_t));
|
|
if (serial_port == NULL) {
|
|
return DC_STATUS_NOMEMORY;
|
|
}
|
|
|
|
// Default to blocking reads.
|
|
serial_port->timeout = -1;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
// Create a RFCOMM socket
|
|
serial_port->socket = ::socket(AF_BTH, SOCK_STREAM, BTHPROTO_RFCOMM);
|
|
|
|
if (serial_port->socket == INVALID_SOCKET) {
|
|
free(serial_port);
|
|
return DC_STATUS_IO;
|
|
}
|
|
|
|
SOCKADDR_BTH socketBthAddress;
|
|
int socketBthAddressBth = sizeof (socketBthAddress);
|
|
char *address = strdup(devaddr);
|
|
|
|
ZeroMemory(&socketBthAddress, socketBthAddressBth);
|
|
qDebug() << "Trying to connect to address " << devaddr;
|
|
|
|
if (WSAStringToAddressA(address,
|
|
AF_BTH,
|
|
NULL,
|
|
(LPSOCKADDR) &socketBthAddress,
|
|
&socketBthAddressBth
|
|
) != 0) {
|
|
qDebug() << "FAiled to convert the address " << address;
|
|
free(address);
|
|
|
|
return DC_STATUS_IO;
|
|
}
|
|
|
|
free(address);
|
|
|
|
socketBthAddress.addressFamily = AF_BTH;
|
|
socketBthAddress.port = BT_PORT_ANY;
|
|
memset(&socketBthAddress.serviceClassId, 0, sizeof(socketBthAddress.serviceClassId));
|
|
socketBthAddress.serviceClassId = SerialPortServiceClass_UUID;
|
|
|
|
// Try to connect to the device
|
|
if (::connect(serial_port->socket,
|
|
(struct sockaddr *) &socketBthAddress,
|
|
socketBthAddressBth
|
|
) != 0) {
|
|
qDebug() << "Failed to connect to device";
|
|
|
|
return DC_STATUS_NODEVICE;
|
|
}
|
|
|
|
qDebug() << "Succesfully connected to device";
|
|
#else
|
|
// Create a RFCOMM socket
|
|
serial_port->socket = new QBluetoothSocket(QBluetoothServiceInfo::RfcommProtocol);
|
|
|
|
// Wait until the connection succeeds or until an error occurs
|
|
QEventLoop loop;
|
|
loop.connect(serial_port->socket, SIGNAL(connected()), SLOT(quit()));
|
|
loop.connect(serial_port->socket, SIGNAL(error(QBluetoothSocket::SocketError)), SLOT(quit()));
|
|
|
|
// Create a timer. If the connection doesn't succeed after five seconds or no error occurs then stop the opening step
|
|
QTimer timer;
|
|
int msec = 5000;
|
|
timer.setSingleShot(true);
|
|
loop.connect(&timer, SIGNAL(timeout()), SLOT(quit()));
|
|
|
|
#if defined(Q_OS_LINUX) && !defined(Q_OS_ANDROID)
|
|
// First try to connect on RFCOMM channel 1. This is the default channel for most devices
|
|
QBluetoothAddress remoteDeviceAddress(devaddr);
|
|
serial_port->socket->connectToService(remoteDeviceAddress, 1, QIODevice::ReadWrite | QIODevice::Unbuffered);
|
|
timer.start(msec);
|
|
loop.exec();
|
|
|
|
if (serial_port->socket->state() == QBluetoothSocket::ConnectingState) {
|
|
// It seems that the connection on channel 1 took more than expected. Wait another 15 seconds
|
|
qDebug() << "The connection on RFCOMM channel number 1 took more than expected. Wait another 15 seconds.";
|
|
timer.start(3 * msec);
|
|
loop.exec();
|
|
} else if (serial_port->socket->state() == QBluetoothSocket::UnconnectedState) {
|
|
// Try to connect on channel number 5. Maybe this is a Shearwater Petrel2 device.
|
|
qDebug() << "Connection on channel 1 failed. Trying on channel number 5.";
|
|
serial_port->socket->connectToService(remoteDeviceAddress, 5, QIODevice::ReadWrite | QIODevice::Unbuffered);
|
|
timer.start(msec);
|
|
loop.exec();
|
|
|
|
if (serial_port->socket->state() == QBluetoothSocket::ConnectingState) {
|
|
// It seems that the connection on channel 5 took more than expected. Wait another 15 seconds
|
|
qDebug() << "The connection on RFCOMM channel number 5 took more than expected. Wait another 15 seconds.";
|
|
timer.start(3 * msec);
|
|
loop.exec();
|
|
}
|
|
}
|
|
#elif defined(Q_OS_ANDROID) || (QT_VERSION >= 0x050500 && defined(Q_OS_MAC))
|
|
// Try to connect to the device using the uuid of the Serial Port Profile service
|
|
QBluetoothAddress remoteDeviceAddress(devaddr);
|
|
#if defined(Q_OS_ANDROID)
|
|
QBluetoothUuid uuid = QBluetoothUuid(QUuid("{00001101-0000-1000-8000-00805f9b34fb}"));
|
|
qDebug() << "connecting to Uuid" << uuid;
|
|
serial_port->socket->setPreferredSecurityFlags(QBluetooth::NoSecurity);
|
|
serial_port->socket->connectToService(remoteDeviceAddress, uuid, QIODevice::ReadWrite | QIODevice::Unbuffered);
|
|
#else
|
|
serial_port->socket->connectToService(remoteDeviceAddress, 1, QIODevice::ReadWrite | QIODevice::Unbuffered);
|
|
#endif
|
|
timer.start(msec);
|
|
loop.exec();
|
|
|
|
if (serial_port->socket->state() == QBluetoothSocket::ConnectingState ||
|
|
serial_port->socket->state() == QBluetoothSocket::ServiceLookupState) {
|
|
// It seems that the connection step took more than expected. Wait another 20 seconds.
|
|
qDebug() << "The connection step took more than expected. Wait another 20 seconds";
|
|
timer.start(4 * msec);
|
|
loop.exec();
|
|
}
|
|
#endif
|
|
if (serial_port->socket->state() != QBluetoothSocket::ConnectedState) {
|
|
|
|
// Get the latest error and try to match it with one from libdivecomputer
|
|
QBluetoothSocket::SocketError err = serial_port->socket->error();
|
|
qDebug() << "Failed to connect to device " << devaddr << ". Device state " << serial_port->socket->state() << ". Error: " << err;
|
|
|
|
free (serial_port);
|
|
switch(err) {
|
|
case QBluetoothSocket::HostNotFoundError:
|
|
case QBluetoothSocket::ServiceNotFoundError:
|
|
return DC_STATUS_NODEVICE;
|
|
case QBluetoothSocket::UnsupportedProtocolError:
|
|
return DC_STATUS_PROTOCOL;
|
|
#if QT_VERSION >= 0x050400
|
|
case QBluetoothSocket::OperationError:
|
|
return DC_STATUS_UNSUPPORTED;
|
|
#endif
|
|
case QBluetoothSocket::NetworkError:
|
|
return DC_STATUS_IO;
|
|
default:
|
|
return DC_STATUS_IO;
|
|
}
|
|
}
|
|
#endif
|
|
io->userdata = serial_port;
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t qt_serial_close(dc_custom_io_t *io)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
if (device == NULL)
|
|
return DC_STATUS_SUCCESS;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
// Cleanup
|
|
closesocket(device->socket);
|
|
free(device);
|
|
#else
|
|
if (device->socket == NULL) {
|
|
free(device);
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
device->socket->close();
|
|
|
|
delete device->socket;
|
|
free(device);
|
|
#endif
|
|
|
|
io->userdata = NULL;
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t qt_serial_read(dc_custom_io_t *io, void* data, size_t size, size_t *actual)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
size_t nbytes = 0;
|
|
int rc;
|
|
|
|
while (nbytes < size) {
|
|
rc = recv (device->socket, (char *) data + nbytes, size - nbytes, 0);
|
|
|
|
if (rc < 0) {
|
|
return DC_STATUS_IO; // Error during recv call.
|
|
} else if (rc == 0) {
|
|
break; // EOF reached.
|
|
}
|
|
|
|
nbytes += rc;
|
|
}
|
|
#else
|
|
if (device == NULL || device->socket == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
size_t nbytes = 0;
|
|
int rc;
|
|
|
|
while(nbytes < size && device->socket->state() == QBluetoothSocket::ConnectedState)
|
|
{
|
|
rc = device->socket->read((char *) data + nbytes, size - nbytes);
|
|
|
|
if (rc < 0) {
|
|
if (errno == EINTR || errno == EAGAIN)
|
|
continue; // Retry.
|
|
|
|
return DC_STATUS_IO; // Something really bad happened :-(
|
|
} else if (rc == 0) {
|
|
// Wait until the device is available for read operations
|
|
QEventLoop loop;
|
|
QTimer timer;
|
|
timer.setSingleShot(true);
|
|
loop.connect(&timer, SIGNAL(timeout()), SLOT(quit()));
|
|
loop.connect(device->socket, SIGNAL(readyRead()), SLOT(quit()));
|
|
timer.start(device->timeout);
|
|
loop.exec();
|
|
|
|
if (!timer.isActive())
|
|
break;
|
|
}
|
|
|
|
nbytes += rc;
|
|
}
|
|
#endif
|
|
if (actual)
|
|
*actual = nbytes;
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t qt_serial_write(dc_custom_io_t *io, const void* data, size_t size, size_t *actual)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
size_t nbytes = 0;
|
|
int rc;
|
|
|
|
while (nbytes < size) {
|
|
rc = send(device->socket, (char *) data + nbytes, size - nbytes, 0);
|
|
|
|
if (rc < 0) {
|
|
return DC_STATUS_IO; // Error during send call.
|
|
}
|
|
|
|
nbytes += rc;
|
|
}
|
|
#else
|
|
if (device == NULL || device->socket == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
size_t nbytes = 0;
|
|
int rc;
|
|
|
|
while(nbytes < size && device->socket->state() == QBluetoothSocket::ConnectedState)
|
|
{
|
|
rc = device->socket->write((char *) data + nbytes, size - nbytes);
|
|
|
|
if (rc < 0) {
|
|
if (errno == EINTR || errno == EAGAIN)
|
|
continue; // Retry.
|
|
|
|
return DC_STATUS_IO; // Something really bad happened :-(
|
|
} else if (rc == 0) {
|
|
break;
|
|
}
|
|
|
|
nbytes += rc;
|
|
}
|
|
#endif
|
|
if (actual)
|
|
*actual = nbytes;
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t qt_serial_purge(dc_custom_io_t *io, dc_direction_t queue)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
(void)queue;
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
#if !defined(Q_OS_WIN)
|
|
if (device->socket == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
#endif
|
|
// TODO: add implementation
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
static dc_status_t qt_serial_get_available(dc_custom_io_t *io, size_t *available)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
#if defined(Q_OS_WIN)
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
// TODO use WSAIoctl to get the information
|
|
|
|
*available = 0;
|
|
#else
|
|
if (device == NULL || device->socket == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
*available = device->socket->bytesAvailable();
|
|
#endif
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
/* UNUSED! */
|
|
static int qt_serial_get_transmitted(qt_serial_t *device) __attribute__ ((unused));
|
|
|
|
static int qt_serial_get_transmitted(qt_serial_t *device)
|
|
{
|
|
#if defined(Q_OS_WIN)
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
// TODO add implementation
|
|
|
|
return 0;
|
|
#else
|
|
if (device == NULL || device->socket == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
return device->socket->bytesToWrite();
|
|
#endif
|
|
}
|
|
|
|
static dc_status_t qt_serial_set_timeout(dc_custom_io_t *io, long timeout)
|
|
{
|
|
qt_serial_t *device = (qt_serial_t*) io->userdata;
|
|
|
|
if (device == NULL)
|
|
return DC_STATUS_INVALIDARGS;
|
|
|
|
device->timeout = timeout;
|
|
|
|
return DC_STATUS_SUCCESS;
|
|
}
|
|
|
|
dc_custom_io_t qt_serial_ops = {
|
|
.userdata = NULL,
|
|
.user_device = NULL,
|
|
.serial_open = qt_serial_open,
|
|
.serial_close = qt_serial_close,
|
|
.serial_read = qt_serial_read,
|
|
.serial_write = qt_serial_write,
|
|
.serial_purge = qt_serial_purge,
|
|
.serial_get_available = qt_serial_get_available,
|
|
.serial_set_timeout = qt_serial_set_timeout,
|
|
// These doesn't make sense over bluetooth
|
|
// NULL means NOP
|
|
.serial_configure = NULL,
|
|
.serial_set_dtr = NULL,
|
|
.serial_set_rts = NULL,
|
|
.serial_set_halfduplex = NULL,
|
|
.serial_set_break = NULL,
|
|
|
|
#ifdef BLE_SUPPORT
|
|
.packet_size = 20,
|
|
.packet_open = qt_ble_open,
|
|
.packet_close = qt_ble_close,
|
|
.packet_read = qt_ble_read,
|
|
.packet_write = qt_ble_write,
|
|
#endif
|
|
};
|
|
|
|
dc_custom_io_t* get_qt_serial_ops() {
|
|
return (dc_custom_io_t*) &qt_serial_ops;
|
|
}
|
|
|
|
}
|
|
#endif
|