Line data Source code
1 : // SPDX-FileCopyrightText: 2022-2023 Paul Colby <git@colby.id.au>
2 : // SPDX-License-Identifier: LGPL-3.0-or-later
3 :
4 : #include "devicecommand.h"
5 :
6 : #include <qtpokit/abstractpokitservice.h>
7 : #include <qtpokit/pokitdevice.h>
8 : #include <qtpokit/pokitdiscoveryagent.h>
9 : #include "qtpokit/pokitmeter.h"
10 : #include "qtpokit/pokitpro.h"
11 :
12 : /*!
13 : * \class DeviceCommand
14 : *
15 : * The AbstractCommand class extends AbstractCommand to add a PokitDevice instance.
16 : */
17 :
18 : /*!
19 : * Construct a new DeviceCommand object with \a parent.
20 : */
21 6810 : DeviceCommand::DeviceCommand(QObject * const parent) : AbstractCommand(parent)
22 : {
23 :
24 6810 : }
25 :
26 : /*!
27 : * Begins scanning for the Pokit device.
28 : */
29 36 : bool DeviceCommand::start()
30 : {
31 72 : qCInfo(lc).noquote() << ((deviceToScanFor.isNull())
32 64 : ? tr("Looking for first available Pokit device...")
33 76 : : tr(R"(Looking for device "%1"...)").arg(deviceToScanFor));
34 36 : discoveryAgent->start();
35 36 : return true;
36 : }
37 :
38 : /*!
39 : * Disconnects the underlying Pokit device, and sets \a exitCode to be return to the OS once the
40 : * disconnection has taken place.
41 : */
42 474 : void DeviceCommand::disconnect(int exitCode)
43 : {
44 474 : qCDebug(lc).noquote() << tr("Disconnecting Pokit device...");
45 : Q_ASSERT(device);
46 : Q_ASSERT(device->controller());
47 474 : exitCodeOnDisconnect = exitCode;
48 474 : device->controller()->disconnectFromDevice();
49 474 : }
50 :
51 : /*!
52 : * \fn virtual AbstractPokitService * DeviceCommand::getService() = 0
53 : *
54 : * Returns a Pokit service object for the derived command class. This will be called by
55 : * deviceDiscovered() when the requested Pokit device has been found, after which
56 : * deviceDiscovered() will connect the returned service's common signals, and kick off the
57 : * device's connection process.
58 : */
59 :
60 : #define DOKIT_CLI_IF_LESS_THAN_RETURN(value, ns, label) \
61 : if (value <= ns::maxValue(label).toUInt()) { \
62 : return label; \
63 : }
64 :
65 : /**
66 : * \fn template<typename T> static T DeviceCommand::minRange(const quint32 maxValue)
67 : *
68 : * Returns the lowest \a T range that can measure at least up to \a maxValue, or AutoRange if no such range is
69 : * available.
70 : *
71 : * \tparam T Range enumerator to evaluate ranges for. Must be one of:
72 : * * PokitMeter::CurrentRange
73 : * * PokitMeter::ResistanceRange
74 : * * PokitMeter::VoltageRange
75 : * * PokitPro::CapacitanceRange
76 : * * PokitPro::CurrentRange
77 : * * PokitPro::ResistanceRange
78 : * * PokitPro::VoltageRange
79 : *
80 : * \cond Doxygen has "only very limited support for member specialization at the moment", so hide these from Doxygen.
81 : * Specifically, if we don't hide them, then Doxygen (at least the current version: v1.9.6) sees the following
82 : * specialisations as new, public, non-static members.
83 : */
84 :
85 : /*!
86 : * Returns the lowest PokitMeter::CurrentRange value that can measure at least up to \a maxValue, or AutoRange if
87 : * the Pokit Meter cannot measure as high as \a maxValue.
88 : */
89 0 : template<> PokitMeter::CurrentRange DeviceCommand::minRange<>(const quint32 maxValue)
90 : {
91 0 : if (maxValue == 0) return PokitMeter::CurrentRange::AutoRange;
92 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::CurrentRange::_10mA)
93 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::CurrentRange::_30mA)
94 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::CurrentRange::_150mA)
95 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::CurrentRange::_300mA)
96 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::CurrentRange::_2A)
97 : return PokitMeter::CurrentRange::AutoRange;
98 : }
99 :
100 : /*!
101 : * Returns the lowest PokitMeter::ResistanceRange value that can measure at least up to \a maxValue, or AutoRange if
102 : * the Pokit Meter cannot measure as high as \a maxValue.
103 : */
104 0 : template<> PokitMeter::ResistanceRange DeviceCommand::minRange(const quint32 maxValue)
105 : {
106 0 : if (maxValue == 0) return PokitMeter::ResistanceRange::AutoRange;
107 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_160)
108 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_330)
109 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_890)
110 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_1K5)
111 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_10K)
112 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_100K)
113 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_470K)
114 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::ResistanceRange::_1M)
115 : return PokitMeter::ResistanceRange::AutoRange;
116 : }
117 :
118 : /*!
119 : * Returns the lowest PokitMeter::VoltageRange value that can measure at least up to \a maxValue, or AutoRange if
120 : * the Pokit Meter cannot measure as high as \a maxValue.
121 : */
122 0 : template<> PokitMeter::VoltageRange DeviceCommand::minRange(const quint32 maxValue)
123 : {
124 0 : if (maxValue == 0) return PokitMeter::VoltageRange::AutoRange;
125 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_300mV)
126 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_2V)
127 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_6V)
128 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_12V)
129 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_30V)
130 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitMeter, PokitMeter::VoltageRange::_60V)
131 : return PokitMeter::VoltageRange::AutoRange;
132 : }
133 :
134 : /*!
135 : * Returns the lowest PokitPro::CapacitanceRange value that can measure at least up to \a maxValue, or AutoRange if
136 : * the Pokit Pro cannot measure as high as \a maxValue.
137 : */
138 0 : template<> PokitPro::CapacitanceRange DeviceCommand::minRange(const quint32 maxValue)
139 : {
140 0 : if (maxValue == 0) return PokitPro::CapacitanceRange::AutoRange;
141 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CapacitanceRange::_100nF)
142 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CapacitanceRange::_10uF)
143 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CapacitanceRange::_1mF)
144 : return PokitPro::CapacitanceRange::AutoRange;
145 : }
146 :
147 : /*!
148 : * Returns the lowest PokitPro::CurrentRange value that can measure at least up to \a maxValue, or AutoRange if
149 : * the Pokit Pro cannot measure as high as \a maxValue.
150 : */
151 0 : template<> PokitPro::CurrentRange DeviceCommand::minRange(const quint32 maxValue)
152 : {
153 0 : if (maxValue == 0) return PokitPro::CurrentRange::AutoRange;
154 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_500uA)
155 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_2mA)
156 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_10mA)
157 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_125mA)
158 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_300mA)
159 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_3A)
160 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::CurrentRange::_10A)
161 : return PokitPro::CurrentRange::AutoRange;
162 : }
163 :
164 : /*!
165 : * Returns the lowest PokitPro::ResistanceRange value that can measure at least up to \a maxValue, or AutoRange if
166 : * the Pokit Pro cannot measure as high as \a maxValue.
167 : */
168 0 : template<> PokitPro::ResistanceRange DeviceCommand::minRange(const quint32 maxValue)
169 : {
170 0 : if (maxValue == 0) return PokitPro::ResistanceRange::AutoRange;
171 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_30)
172 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_75)
173 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_400)
174 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_5K)
175 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_10K)
176 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_15K)
177 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_40K)
178 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_500K)
179 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_700K)
180 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_1M)
181 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::ResistanceRange::_3M)
182 : return PokitPro::ResistanceRange::AutoRange;
183 : }
184 :
185 : /*!
186 : * Returns the lowest PokitPro::VoltageRange value that can measure at least up to \a maxValue, or AutoRange if
187 : * the Pokit Pro cannot measure as high as \a maxValue.
188 : */
189 0 : template<> PokitPro::VoltageRange DeviceCommand::minRange(const quint32 maxValue)
190 : {
191 0 : if (maxValue == 0) return PokitPro::VoltageRange::AutoRange;
192 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_250mV)
193 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_2V)
194 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_10V)
195 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_30V)
196 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_60V)
197 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_125V)
198 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_400V)
199 0 : DOKIT_CLI_IF_LESS_THAN_RETURN(maxValue, PokitPro, PokitPro::VoltageRange::_600V)
200 : return PokitPro::VoltageRange::AutoRange;
201 : }
202 :
203 : /// \endcond
204 :
205 : /*!
206 : * Returns the \a product's lowest capacitance range that can measure at least up to \a maxValue, or AutoRange if no
207 : * such range is available.
208 : *
209 : * \note Since Pokit Meters do not support capacitance measurement, \a product should not be PokitProduct::PokitMeter.
210 : *
211 : * \see minRange<PokitPro::CapacitanceRange>
212 : */
213 0 : quint8 DeviceCommand::minCapacitanceRange(const PokitProduct product, const quint32 maxValue)
214 : {
215 0 : switch (product) {
216 : case PokitProduct::PokitMeter:
217 : Q_ASSERT_X(false, "DeviceCommand::minCapacitanceRange", "Pokit Meter has no capacitance support.");
218 : return 255;
219 0 : case PokitProduct::PokitPro:
220 0 : return +minRange<PokitPro::CapacitanceRange>(maxValue);
221 : }
222 : Q_ASSERT_X(false, "DeviceCommand::minCapacitanceRange", "Unknown PokitProduct enum value");
223 : return 255;
224 : }
225 :
226 : /*!
227 : * Returns the \a product's lowest current range that can measure at least up to \a maxValue, or AutoRange if no
228 : * such range is available.
229 : *
230 : * \see DeviceCommand::minRange<PokitMeter::CurrentRange>(const quint32 maxValue)
231 : * \see minRange<PokitPro::CurrentRange>(const quint32 maxValue)
232 : */
233 0 : quint8 DeviceCommand::minCurrentRange(const PokitProduct product, const quint32 maxValue)
234 : {
235 0 : switch (product) {
236 0 : case PokitProduct::PokitMeter:
237 0 : return +minRange<PokitMeter::CurrentRange>(maxValue);
238 0 : case PokitProduct::PokitPro:
239 0 : return +minRange<PokitPro::CurrentRange>(maxValue);
240 : }
241 : Q_ASSERT_X(false, "DeviceCommand::minCurrentRange", "Unknown PokitProduct enum value");
242 : return 255;
243 : }
244 :
245 : /*!
246 : * Returns the \a product's lowest resistance range that can measure at least up to \a maxValue, or AutoRange if no
247 : * such range is available.
248 : *
249 : * \see DeviceCommand::minRange<PokitMeter::ResistanceRange>(const quint32 maxValue)
250 : * \see minRange<PokitPro::ResistanceRange>(const quint32 maxValue)
251 : */
252 0 : quint8 DeviceCommand::minResistanceRange(const PokitProduct product, const quint32 maxValue)
253 : {
254 0 : switch (product) {
255 0 : case PokitProduct::PokitMeter:
256 0 : return +minRange<PokitMeter::ResistanceRange>(maxValue);
257 0 : case PokitProduct::PokitPro:
258 0 : return +minRange<PokitPro::ResistanceRange>(maxValue);
259 : }
260 : Q_ASSERT_X(false, "DeviceCommand::minResistanceRange", "Unknown PokitProduct enum value");
261 : return 255;
262 : }
263 :
264 : /*!
265 : t
266 : * Returns the \a product's lowest voltage range that can measure at least up to \a maxValue, or AutoRange if no
267 : * such range is available.
268 : *
269 : * \see DeviceCommand::minRange<PokitMeter::VoltageRange>(const quint32 maxValue)
270 : * \see minRange<PokitPro::VoltageRange>(const quint32 maxValue)
271 : */
272 0 : quint8 DeviceCommand::minVoltageRange(const PokitProduct product, const quint32 maxValue)
273 : {
274 0 : switch (product) {
275 0 : case PokitProduct::PokitMeter:
276 0 : return +minRange<PokitMeter::VoltageRange>(maxValue);
277 0 : case PokitProduct::PokitPro:
278 0 : return +minRange<PokitPro::VoltageRange>(maxValue);
279 : }
280 : Q_ASSERT_X(false, "DeviceCommand::minVoltageRange", "Unknown PokitProduct enum value");
281 : return 255;
282 : }
283 :
284 : #undef DOKIT_CLI_IF_LESS_THAN_RETURN
285 :
286 :
287 : /*!
288 : * Handles controller error events. This base implementation simply logs \a error and then exits
289 : * with `EXIT_FAILURE`. Derived classes may override this slot to implement their own error
290 : * handing if desired.
291 : */
292 36 : void DeviceCommand::controllerError(QLowEnergyController::Error error)
293 : {
294 108 : qCWarning(lc).noquote() << tr("Bluetooth controller error:") << error;
295 36 : QCoreApplication::exit(EXIT_FAILURE);
296 36 : }
297 :
298 : /*!
299 : * Handles devics disconnection events. This base implementation simply logs and exits the
300 : * application (via QCoreApplication::exit) with the current exitCodeOnDisconnect value, which is
301 : * initialise to `EXIT_FAILURE` in the constructor, but should be set to `EXIT_SUCESS` if/when
302 : * the derived command class has completed its actions and requested the disconnection (as opposed
303 : * to a spontaneous disconnection on error).
304 : */
305 18 : void DeviceCommand::deviceDisconnected()
306 : {
307 18 : qCDebug(lc).noquote() << tr("Pokit device disconnected. Exiting with code %1.")
308 0 : .arg(exitCodeOnDisconnect);
309 18 : QCoreApplication::exit(exitCodeOnDisconnect);
310 18 : }
311 :
312 : /*!
313 : * Handles service error events. This base implementation simply logs \a error and then exits
314 : * with `EXIT_FAILURE`. Derived classes may override this slot to implement their own error
315 : * handing if desired.
316 : *
317 : * \note As this base class does not construct services (derived classed do), its up to the derived
318 : * classed to connect this slot to the relevant service's error signal if desired.
319 : */
320 36 : void DeviceCommand::serviceError(const QLowEnergyService::ServiceError error)
321 : {
322 108 : qCWarning(lc).noquote() << tr("Bluetooth service error:") << error;
323 36 : QCoreApplication::exit(EXIT_FAILURE);
324 36 : }
325 :
326 : /*!
327 : * Handles service detail discovery events. This base implementation simply logs the event, and
328 : * nothing more. Derived classes may (usually do) override this slot to provide their own processing
329 : * when a services details have been discovered.
330 : */
331 348 : void DeviceCommand::serviceDetailsDiscovered()
332 : {
333 348 : qCDebug(lc).noquote() << tr("Service details discovered.");
334 348 : }
335 :
336 : /*!
337 : * Checks if \a info is the device (if any) we're looking for, and if so, create a contoller and
338 : * service, and begins connecting to the device.
339 : */
340 18 : void DeviceCommand::deviceDiscovered(const QBluetoothDeviceInfo &info)
341 : {
342 : Q_ASSERT(isPokitProduct(info));
343 :
344 18 : if (device) {
345 0 : qCDebug(lc).noquote() << tr(R"(Ignoring additional Pokit device "%1" (%2) at (%3).)")
346 0 : .arg(info.name(), info.deviceUuid().toString(), info.address().toString());
347 0 : return;
348 : }
349 :
350 54 : if ((deviceToScanFor.isEmpty()) || (deviceToScanFor == info.name()) ||
351 72 : ((!info.address().isNull()) && (info.address() == QBluetoothAddress(deviceToScanFor))) ||
352 36 : ((!info.deviceUuid().isNull()) && (info.deviceUuid() == QBluetoothUuid(deviceToScanFor))))
353 : {
354 0 : qCDebug(lc).noquote() << tr(R"(Found Pokit device "%1" (%2) at (%3).)")
355 0 : .arg(info.name(), info.deviceUuid().toString(), info.address().toString());
356 0 : discoveryAgent->stop();
357 :
358 0 : device = new PokitDevice(info, this);
359 0 : connect(device->controller(), &QLowEnergyController::disconnected,
360 : this, &DeviceCommand::deviceDisconnected);
361 0 : connect(device->controller(),
362 : #if (QT_VERSION < QT_VERSION_CHECK(6, 2, 0))
363 : QOverload<QLowEnergyController::Error>::of(&QLowEnergyController::error),
364 : #else
365 : &QLowEnergyController::errorOccurred,
366 : #endif
367 : this, &DeviceCommand::controllerError, Qt::QueuedConnection);
368 :
369 0 : AbstractPokitService * const service = getService();
370 0 : service->setPokitProduct(pokitProduct(info));
371 :
372 : Q_ASSERT(service);
373 0 : connect(service, &AbstractPokitService::serviceDetailsDiscovered,
374 : this, &DeviceCommand::serviceDetailsDiscovered);
375 0 : connect(service, &AbstractPokitService::serviceErrorOccurred,
376 : this, &DeviceCommand::serviceError);
377 :
378 0 : qCDebug(lc).noquote() << tr(R"(Connecting to %1 device "%2" (%3) at (%4).)").arg(
379 0 : toString(*service->pokitProduct()), info.name(), info.deviceUuid().toString(), info.address().toString());
380 0 : device->controller()->connectToDevice();
381 0 : return;
382 : }
383 :
384 18 : qCDebug(lc).noquote() << tr(R"(Ignoring non-matching Pokit device "%1" (%2) at (%3).)")
385 0 : .arg(info.name(), info.deviceUuid().toString(), info.address().toString());
386 3 : return;
387 : }
388 :
389 : /*!
390 : * Checks that the requested device was discovered, and if not, reports and error and exits.
391 : */
392 36 : void DeviceCommand::deviceDiscoveryFinished()
393 : {
394 36 : if (!device) {
395 72 : qCWarning(lc).noquote() << ((deviceToScanFor.isNull())
396 64 : ? tr("Failed to find any Pokit device.")
397 76 : : tr(R"(Failed to find device "%1".)").arg(deviceToScanFor));
398 36 : QCoreApplication::exit(EXIT_FAILURE);
399 : }
400 36 : }
|