LCOV - code coverage report
Current view: top level - src/lib - deviceinfoservice.cpp (source / functions) Coverage Total Hit
Project: Dokit Lines: 75.0 % 140 105
Version: Functions: 87.5 % 40 35

            Line data    Source code
       1              : // SPDX-FileCopyrightText: 2022-2025 Paul Colby <git@colby.id.au>
       2              : // SPDX-License-Identifier: LGPL-3.0-or-later
       3              : 
       4              : /*!
       5              :  * \file
       6              :  * Defines the DeviceInfoService and DeviceInfoServicePrivate classes.
       7              :  */
       8              : 
       9              : #include <qtpokit/deviceinfoservice.h>
      10              : #include "deviceinfoservice_p.h"
      11              : 
      12              : #include <QtEndian>
      13              : 
      14              : /*!
      15              :  * \class DeviceInfoService
      16              :  *
      17              :  * The DeviceInfoService class accesses the `Device Info` service of Pokit devices.
      18              :  */
      19              : 
      20              : /*!
      21              :  * Constructs a new Pokit service with \a parent.
      22              :  */
      23         1600 : DeviceInfoService::DeviceInfoService(QLowEnergyController * const controller, QObject * parent)
      24         2280 :     : AbstractPokitService(new DeviceInfoServicePrivate(controller, this), parent)
      25         1160 : {
      26              : 
      27         2760 : }
      28              : 
      29              : /*!
      30              :  * \cond internal
      31              :  * Constructs a new Pokit service with \a parent, and private implementation \a d.
      32              :  */
      33            0 : DeviceInfoService::DeviceInfoService(
      34            0 :     DeviceInfoServicePrivate * const d, QObject * const parent)
      35            0 :     : AbstractPokitService(d, parent)
      36            0 : {
      37              : 
      38            0 : }
      39              : /// \endcond
      40              : 
      41           40 : bool DeviceInfoService::readCharacteristics()
      42           47 : {
      43           64 :     const bool r1 = readFirmwareRevisionCharacteristic();
      44           64 :     const bool r2 = readHardwareRevisionCharacteristic();
      45           64 :     const bool r3 = readSoftwareRevisionCharacteristic();
      46           64 :     const bool r4 = readManufacturerCharacteristics();
      47           64 :     const bool r5 = readModelNumberCharacteristic();
      48           87 :     const bool r6 = ((service() != nullptr) && (service()->characteristic(CharacteristicUuids::serialNumber).isValid()))
      49           87 :         ? readSerialNumberCharacteristic() : true;
      50           87 :     return (r1 && r2 && r3 && r4 && r5 && r6);
      51           47 : }
      52              : 
      53              : /*!
      54              :  * Read the `Device Info` service's `Firmware Revision` characteristic.
      55              :  *
      56              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
      57              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
      58              :  * not yet been discovered).
      59              :  *
      60              :  * Emits firmwareRevisionRead() if/when the characteristic has been read successfully.
      61              :  */
      62           57 : bool DeviceInfoService::readFirmwareRevisionCharacteristic()
      63           94 : {
      64           94 :     Q_D(DeviceInfoService);
      65          174 :     return d->readCharacteristic(CharacteristicUuids::firmwareRevision);
      66           94 : }
      67              : 
      68              : /*!
      69              :  * Read the `Device Info` service's `Hardware Revision` characteristic.
      70              :  *
      71              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
      72              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
      73              :  * not yet been discovered).
      74              :  *
      75              :  * Emits hardwareRevisionRead() if/when the characteristic has been read successfully.
      76              :  */
      77           57 : bool DeviceInfoService::readHardwareRevisionCharacteristic()
      78           94 : {
      79           94 :     Q_D(DeviceInfoService);
      80          174 :     return d->readCharacteristic(CharacteristicUuids::hardwareRevision);
      81           94 : }
      82              : 
      83              : /*!
      84              :  * Read the `Device Info` service's `Manufacturer Name` characteristic.
      85              :  *
      86              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
      87              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
      88              :  * not yet been discovered).
      89              :  *
      90              :  * Emits manufacturerNameRead() if/when the characteristic has been read successfully.
      91              :  */
      92           57 : bool DeviceInfoService::readManufacturerCharacteristics()
      93           94 : {
      94           94 :     Q_D(DeviceInfoService);
      95          174 :     return d->readCharacteristic(CharacteristicUuids::manufacturerName);
      96           94 : }
      97              : 
      98              : /*!
      99              :  * Read the `Device Info` service's `Model Number` characteristic.
     100              :  *
     101              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
     102              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
     103              :  * not yet been discovered).
     104              :  *
     105              :  * Emits modelNumberRead() if/when the characteristic has been read successfully.
     106              :  */
     107           57 : bool DeviceInfoService::readModelNumberCharacteristic()
     108           94 : {
     109           94 :     Q_D(DeviceInfoService);
     110          174 :     return d->readCharacteristic(CharacteristicUuids::modelNumber);
     111           94 : }
     112              : 
     113              : /*!
     114              :  * Read the `Device Info` service's `Software Revision` characteristic.
     115              :  *
     116              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
     117              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
     118              :  * not yet been discovered).
     119              :  *
     120              :  * Emits softwareRevisionRead() if/when the characteristic has been read successfully.
     121              :  */
     122           57 : bool DeviceInfoService::readSoftwareRevisionCharacteristic()
     123           94 : {
     124           94 :     Q_D(DeviceInfoService);
     125          174 :     return d->readCharacteristic(CharacteristicUuids::softwareRevision);
     126           94 : }
     127              : 
     128              : /*!
     129              :  * Read the `Device Info` service's (undocumented) `Serial Number` characteristic.
     130              :  *
     131              :  * Returns `true` is the read request is succesfully queued, `false` otherwise (ie if the
     132              :  * underlying controller it not yet connected to the Pokit device, or the device's services have
     133              :  * not yet been discovered).
     134              :  *
     135              :  * Emits serialNumberRead() if/when the characteristic has been read successfully.
     136              :  */
     137           40 : bool DeviceInfoService::readSerialNumberCharacteristic()
     138           47 : {
     139           47 :     Q_D(DeviceInfoService);
     140           87 :     return d->readCharacteristic(CharacteristicUuids::serialNumber);
     141           47 : }
     142              : 
     143              : /*!
     144              :  * Returns the most recent value of the `Device Info` service's `Manufacturer Name` characteristic.
     145              :  *
     146              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     147              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     148              :  * null QString is returned.
     149              :  */
     150         1000 : QString DeviceInfoService::manufacturer() const
     151          455 : {
     152          455 :     Q_D(const DeviceInfoService);
     153          455 :     const QLowEnergyCharacteristic characteristic =
     154         1455 :         d->getCharacteristic(CharacteristicUuids::manufacturerName);
     155         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()) : QString();
     156         1455 : }
     157              : 
     158              : /*!
     159              :  * Returns the most recent value of the `Device Info` service's `Model Number` characteristic.
     160              :  *
     161              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     162              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     163              :  * null QString is returned.
     164              :  */
     165         1000 : QString DeviceInfoService::modelNumber() const
     166          455 : {
     167          455 :     Q_D(const DeviceInfoService);
     168          455 :     const QLowEnergyCharacteristic characteristic =
     169         1455 :         d->getCharacteristic(CharacteristicUuids::modelNumber);
     170         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()) : QString();
     171         1455 : }
     172              : 
     173              : /*!
     174              :  * Returns the most recent value of the `Device Info` service's `Hardware Revision` characteristic.
     175              :  *
     176              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     177              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     178              :  * null QString is returned.
     179              :  */
     180         1000 : QString DeviceInfoService::hardwareRevision() const
     181          455 : {
     182          455 :     Q_D(const DeviceInfoService);
     183          455 :     const QLowEnergyCharacteristic characteristic =
     184         1455 :         d->getCharacteristic(CharacteristicUuids::hardwareRevision);
     185         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()) : QString();
     186         1455 : }
     187              : 
     188              : /*!
     189              :  * Returns the most recent value of the `Device Info` service's `Firmware Revision` characteristic.
     190              :  *
     191              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     192              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     193              :  * null QString is returned.
     194              :  */
     195         1000 : QString DeviceInfoService::firmwareRevision() const
     196          455 : {
     197          455 :     Q_D(const DeviceInfoService);
     198          455 :     const QLowEnergyCharacteristic characteristic =
     199         1455 :         d->getCharacteristic(CharacteristicUuids::firmwareRevision);
     200         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()) : QString();
     201         1455 : }
     202              : 
     203              : /*!
     204              :  * Returns the most recent value of the `Device Info` service's `Software Revision` characteristic.
     205              :  *
     206              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     207              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     208              :  * null QString is returned.
     209              :  */
     210         1000 : QString DeviceInfoService::softwareRevision() const
     211          455 : {
     212          455 :     Q_D(const DeviceInfoService);
     213          455 :     const QLowEnergyCharacteristic characteristic =
     214         1455 :         d->getCharacteristic(CharacteristicUuids::softwareRevision);
     215         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()) : QString();
     216         1455 : }
     217              : 
     218              : /*!
     219              :  * Returns the most recent value of the `Device Info` service's (undocumented) `Serial Number`
     220              :  * characteristic.
     221              :  *
     222              :  * The returned value, if any, is from the underlying Bluetooth stack's cache. If no such value is
     223              :  * currently available (ie the serviceDetailsDiscovered signal has not been emitted yet), then a
     224              :  * null QString is returned.
     225              :  */
     226         1000 : QString DeviceInfoService::serialNumber() const
     227          455 : {
     228          455 :     Q_D(const DeviceInfoService);
     229          455 :     const QLowEnergyCharacteristic characteristic =
     230         1455 :         d->getCharacteristic(CharacteristicUuids::serialNumber);
     231              :     /*!
     232              :      * \cond internal
     233              :      * \pokitApi Unlike other string characteristics, Pokit (Pro) devices always appear to add a trailing
     234              :      * `null` byte to serial number strings. So here we strip any that are present.
     235              :      * \endcond
     236              :      */
     237         2455 :     return (characteristic.isValid()) ? QString::fromUtf8(characteristic.value()).remove(QLatin1Char('\0')) : QString();
     238         1455 : }
     239              : 
     240              : /*!
     241              :  * \fn DeviceInfoService::manufacturerRead
     242              :  *
     243              :  * This signal is emitted when the `Manufacturer Name` characteristic has been read successfully.
     244              :  *
     245              :  * \see readManufacturerCharacteristic
     246              :  * \see manufacturer
     247              :  */
     248              : 
     249              : /*!
     250              :  * \fn DeviceInfoService::modelNumberRead
     251              :  *
     252              :  * This signal is emitted when the `Model Number` characteristic has been read successfully.
     253              :  *
     254              :  * \see readModelNumberCharacteristic
     255              :  * \see modelNumber
     256              :  */
     257              : 
     258              : /*!
     259              :  * \fn DeviceInfoService::hardwareRevisionRead
     260              :  *
     261              :  * This signal is emitted when the `Hardware Revision` characteristic has been read successfully.
     262              :  *
     263              :  * \see readHardwareRevisionCharacteristic
     264              :  * \see hardwareRevision
     265              :  */
     266              : 
     267              : /*!
     268              :  * \fn DeviceInfoService::firmwareRevisionRead
     269              :  *
     270              :  * This signal is emitted when the `Firmware Revision` characteristic has been read successfully.
     271              :  *
     272              :  * \see readFirmwareRevisionCharacteristic
     273              :  * \see firmwareRevision
     274              :  */
     275              : 
     276              : /*!
     277              :  * \fn DeviceInfoService::softwareRevisionRead
     278              :  *
     279              :  * This signal is emitted when the `Software Revision` characteristic has been read successfully.
     280              :  *
     281              :  * \see readSoftwareRevisionCharacteristic
     282              :  * \see softwareRevision
     283              :  */
     284              : 
     285              : /*!
     286              :  * \fn DeviceInfoService::serialNumberRead
     287              :  *
     288              :  * This signal is emitted when the `Serial Number` characteristic has been read successfully.
     289              :  *
     290              :  * \see readSerialNumberCharacteristic
     291              :  * \see serialNumber
     292              :  */
     293              : 
     294              : /*!
     295              :  * \cond internal
     296              :  * \class DeviceInfoServicePrivate
     297              :  *
     298              :  * The DeviceInfoServicePrivate class provides private implementation for DeviceInfoService.
     299              :  */
     300              : 
     301              : /*!
     302              :  * \internal
     303              :  * Constructs a new DeviceInfoServicePrivate object with public implementation \a q.
     304              :  */
     305          680 : DeviceInfoServicePrivate::DeviceInfoServicePrivate(
     306         1600 :     QLowEnergyController * controller, DeviceInfoService * const q)
     307         2280 :     : AbstractPokitServicePrivate(DeviceInfoService::serviceUuid, controller, q)
     308         1160 : {
     309              : 
     310         1840 : }
     311              : 
     312              : /*!
     313              :  * Implements AbstractPokitServicePrivate::characteristicRead to parse \a value, then emit a
     314              :  * specialised signal, for each supported \a characteristic.
     315              :  */
     316           40 : void DeviceInfoServicePrivate::characteristicRead(const QLowEnergyCharacteristic &characteristic,
     317              :                                               const QByteArray &value)
     318           47 : {
     319           87 :     AbstractPokitServicePrivate::characteristicRead(characteristic, value);
     320              : 
     321           47 :     Q_Q(DeviceInfoService);
     322           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::manufacturerName) {
     323            0 :         const QString name = QString::fromUtf8(value);
     324            0 :         qCDebug(lc).noquote() << tr(R"(Manufacturer name: "%1")").arg(name);
     325            0 :         Q_EMIT q->manufacturerRead(name);
     326            0 :         return;
     327            0 :     }
     328              : 
     329           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::modelNumber) {
     330            0 :         const QString model = QString::fromUtf8(value);
     331            0 :         qCDebug(lc).noquote() << tr(R"(Model number: "%1")").arg(model);
     332            0 :         Q_EMIT q->modelNumberRead(model);
     333            0 :         return;
     334            0 :     }
     335              : 
     336           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::hardwareRevision) {
     337            0 :         const QString revision = QString::fromUtf8(value);
     338            0 :         qCDebug(lc).noquote() << tr(R"(Hardware revision: "%1")").arg(revision);
     339            0 :         Q_EMIT q->hardwareRevisionRead(revision);
     340            0 :         return;
     341            0 :     }
     342              : 
     343           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::firmwareRevision) {
     344            0 :         const QString revision = QString::fromUtf8(value);
     345            0 :         qCDebug(lc).noquote() << tr(R"(Firmware revision: "%1")").arg(revision);
     346            0 :         Q_EMIT q->firmwareRevisionRead(revision);
     347            0 :         return;
     348            0 :     }
     349              : 
     350           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::softwareRevision) {
     351            0 :         const QString revision = QString::fromUtf8(value);
     352            0 :         qCDebug(lc).noquote() << tr(R"(Software revision: "%1")").arg(revision);
     353            0 :         Q_EMIT q->softwareRevisionRead(revision);
     354            0 :         return;
     355            0 :     }
     356              : 
     357           87 :     if (characteristic.uuid() == DeviceInfoService::CharacteristicUuids::serialNumber) {
     358            0 :         const QString serialNumber = QString::fromUtf8(value);
     359            0 :         qCDebug(lc).noquote() << tr(R"(Serial number: "%1")").arg(serialNumber);
     360            0 :         Q_EMIT q->serialNumberRead(serialNumber);
     361            0 :         return;
     362            0 :     }
     363              : 
     364          192 :     qCWarning(lc).noquote() << tr("Unknown characteristic read for Device Info service")
     365          144 :         << serviceUuid << characteristic.name() << characteristic.uuid();
     366           47 : }
     367              : 
     368              : /// \endcond
        

Generated by: LCOV version 2.2-1