// SPDX-License-Identifier: GPL-2.0 #include "btdiscovery.h" #include "downloadfromdcthread.h" #include "core/libdivecomputer.h" #include extern QMap descriptorLookup; BTDiscovery *BTDiscovery::m_instance = NULL; ConnectionListModel::ConnectionListModel(QObject *parent) : QAbstractListModel(parent) { } QHash ConnectionListModel::roleNames() const { QHash roles; roles[AddressRole] = "address"; return roles; } QVariant ConnectionListModel::data(const QModelIndex &index, int role) const { if (index.row() < 0 || index.row() >= m_addresses.count()) return QVariant(); if (role != AddressRole) return QVariant(); return m_addresses[index.row()]; } QString ConnectionListModel::address(int idx) const { if (idx < 0 || idx >> m_addresses.count()) return QString(); return m_addresses[idx]; } int ConnectionListModel::rowCount(const QModelIndex &parent) const { Q_UNUSED(parent) return m_addresses.count(); } void ConnectionListModel::addAddress(const QString address) { beginInsertRows(QModelIndex(), rowCount(), rowCount()); m_addresses.append(address); endInsertRows(); } static dc_descriptor_t *getDeviceType(QString btName) // central function to convert a BT name to a Subsurface known vendor/model pair { QString vendor, product; if (btName.startsWith("OSTC")) { vendor = "Heinrichs Weikamp"; if (btName.mid(4,2) == "3#") product = "OSTC 3"; else if (btName.mid(4,2) == "3+") product = "OSTC 3+"; else if (btName.mid(4,2) == "s#") product = "OSTC Sport"; else if (btName.mid(4,2) == "s ") product = "OSTC Sport"; else if (btName.mid(4,2) == "4-") product = "OSTC 4"; else if (btName.mid(4,2) == "2-") product = "OSTC 2N"; // all OSTCs are HW_FAMILY_OSTC_3, so when we do not know, // just try this else product = "OSTC 3"; // all OSTCs are HW_FAMILY_OSTC_3 } if (btName.startsWith("Petrel") || btName.startsWith("Perdix")) { vendor = "Shearwater"; if (btName.startsWith("Petrel")) product = "Petrel"; // or petrel 2? if (btName.startsWith("Perdix")) product = "Perdix"; } if (btName.startsWith("EON Steel")) { vendor = "Suunto"; product = "EON Steel"; } if (btName.startsWith("G2")) { vendor = "Scubapro"; product = "G2"; } if (!vendor.isEmpty() && !product.isEmpty()) return(descriptorLookup[vendor + product]); return(NULL); } BTDiscovery::BTDiscovery(QObject *parent) { Q_UNUSED(parent) if (m_instance) { qDebug() << "trying to create an additional BTDiscovery object"; return; } m_instance = this; #if defined(BT_SUPPORT) if (localBtDevice.isValid() && localBtDevice.hostMode() == QBluetoothLocalDevice::HostConnectable) { btPairedDevices.clear(); qDebug() << "localDevice " + localBtDevice.name() + " is valid, starting discovery"; m_btValid = true; #if defined(Q_OS_LINUX) && !defined(Q_OS_ANDROID) discoveryAgent = new QBluetoothDeviceDiscoveryAgent(this); connect(discoveryAgent, &QBluetoothDeviceDiscoveryAgent::deviceDiscovered, this, &BTDiscovery::btDeviceDiscovered); discoveryAgent->start(); #endif #if defined(Q_OS_ANDROID) && defined(BT_SUPPORT) getBluetoothDevices(); // and add the paired devices to the internal data // So behaviour is same on Linux/Bluez stack and // Android/Java stack with respect to discovery for (int i = 0; i < btPairedDevices.length(); i++) { btDeviceDiscoveredMain(btPairedDevices[i]); } #endif for (int i = 0; i < btPairedDevices.length(); i++) { qDebug() << "Paired =" << btPairedDevices[i].name << btPairedDevices[i].address; } #if defined(Q_OS_LINUX) && !defined(Q_OS_ANDROID) discoveryAgent->stop(); #endif } else { qDebug() << "localBtDevice isn't valid"; m_btValid = false; } #endif } BTDiscovery::~BTDiscovery() { m_instance = NULL; #if defined(BT_SUPPORT) free(discoveryAgent); #endif } BTDiscovery *BTDiscovery::instance() { if (!m_instance) m_instance = new BTDiscovery(); return m_instance; } #if defined(BT_SUPPORT) #if defined(SSRF_CUSTOM_IO) extern void addBtUuid(QBluetoothUuid uuid); #endif extern QHash productList; extern QStringList vendorList; QString markBLEAddress(const QBluetoothDeviceInfo *device) { QBluetoothDeviceInfo::CoreConfigurations flags; QString prefix = ""; flags = device->coreConfigurations(); if (flags == QBluetoothDeviceInfo::LowEnergyCoreConfiguration) prefix = "LE:"; return prefix + device->address().toString(); } void BTDiscovery::btDeviceDiscovered(const QBluetoothDeviceInfo &device) { #if defined(SSRF_CUSTOM_IO) btPairedDevice this_d; this_d.address = markBLEAddress(&device); this_d.name = device.name(); btPairedDevices.append(this_d); QList serviceUuids = device.serviceUuids(); foreach (QBluetoothUuid id, serviceUuids) { addBtUuid(id); qDebug() << id.toByteArray(); } btDeviceDiscoveredMain(this_d); #endif } void BTDiscovery::btDeviceDiscoveredMain(const btPairedDevice &device) { btVendorProduct btVP; QString newDevice; dc_descriptor_t *newDC = getDeviceType(device.name); if (newDC) newDevice = dc_descriptor_get_product(newDC); else newDevice = device.name; qDebug() << "Found new device:" << newDevice << device.address; QString vendor; if (newDC) foreach (vendor, productList.keys()) { if (productList[vendor].contains(newDevice)) { qDebug() << "this could be a " + vendor + " " + (newDevice == "OSTC 3" ? "OSTC family" : newDevice); btVP.btpdi = device; btVP.dcDescriptor = newDC; btVP.vendorIdx = vendorList.indexOf(vendor); btVP.productIdx = productList[vendor].indexOf(newDevice); qDebug() << "adding new btDCs entry (detected DC)" << newDevice << btVP.vendorIdx << btVP.productIdx << btVP.btpdi.address;; btDCs << btVP; productList[QObject::tr("Paired Bluetooth Devices")].append(device.name + " (" + device.address + ")"); connectionListModel.addAddress(device.address + " (" + device.name + ")"); return; } } connectionListModel.addAddress(device.address); qDebug() << "Not recognized as dive computer"; } QList BTDiscovery::getBtDcs() { return btDCs; } bool BTDiscovery::btAvailable() const { return m_btValid; } // Android: As Qt is not able to pull the pairing data from a device, i // a lengthy discovery process is needed to see what devices are paired. On // https://forum.qt.io/topic/46075/solved-bluetooth-list-paired-devices // user s.frings74 does, however, present a solution to this using JNI. // Currently, this code is taken "as is". #if defined(Q_OS_ANDROID) void BTDiscovery::getBluetoothDevices() { struct BTDiscovery::btPairedDevice result; // Query via Android Java API. // returns a BluetoothAdapter QAndroidJniObject adapter=QAndroidJniObject::callStaticObjectMethod("android/bluetooth/BluetoothAdapter","getDefaultAdapter","()Landroid/bluetooth/BluetoothAdapter;"); if (checkException("BluetoothAdapter.getDefaultAdapter()", &adapter)) { return; } // returns a Set QAndroidJniObject pairedDevicesSet=adapter.callObjectMethod("getBondedDevices","()Ljava/util/Set;"); if (checkException("BluetoothAdapter.getBondedDevices()", &pairedDevicesSet)) { return; } jint size=pairedDevicesSet.callMethod("size"); checkException("Set.size()", &pairedDevicesSet); if (size > 0) { // returns an Iterator QAndroidJniObject iterator=pairedDevicesSet.callObjectMethod("iterator","()Ljava/util/Iterator;"); if (checkException("Set.iterator()", &iterator)) { return; } for (int i = 0; i < size; i++) { // returns a BluetoothDevice QAndroidJniObject dev=iterator.callObjectMethod("next","()Ljava/lang/Object;"); if (checkException("Iterator.next()", &dev)) { continue; } jint btType = dev.callMethod("getType", "()I"); result.address = dev.callObjectMethod("getAddress","()Ljava/lang/String;").toString(); if (btType == 2) // DEVICE_TYPE_LE result.address = QString("LE:%1").arg(result.address); result.name = dev.callObjectMethod("getName", "()Ljava/lang/String;").toString(); qDebug() << "paired Device type" << btType << "with address" << result.address; btPairedDevices.append(result); if (btType == 3) { // DEVICE_TYPE_DUAL result.address = QString("LE:%1").arg(result.address); qDebug() << "paired Device type" << btType << "with address" << result.address; btPairedDevices.append(result); } } } } bool BTDiscovery::checkException(const char* method, const QAndroidJniObject *obj) { static QAndroidJniEnvironment env; bool result = false; if (env->ExceptionCheck()) { qCritical("Exception in %s", method); env->ExceptionDescribe(); env->ExceptionClear(); result=true; } if (!(obj == NULL || obj->isValid())) { qCritical("Invalid object returned by %s", method); result=true; } return result; } #endif // Q_OS_ANDROID #endif // BT_SUPPORT