summaryrefslogtreecommitdiffstats
path: root/core
diff options
context:
space:
mode:
authorGravatar Linus Torvalds <torvalds@linux-foundation.org>2020-03-13 10:06:06 -0700
committerGravatar Dirk Hohndel <dirk@hohndel.org>2020-03-14 11:34:46 -0700
commit4ee4bbdb58d7dabae1be1fa6e62622caa5a4fcbc (patch)
treefa0d0a6e2812232414ea4557bb91da3f114b00ee /core
parent0acfa28d873ded8b87a5b64cc8e13aca5c6b34a3 (diff)
downloadsubsurface-4ee4bbdb58d7dabae1be1fa6e62622caa5a4fcbc.tar.gz
iostream: fix incorrect rfcomm error case when reading
We had two independent bugs here, both of which needed to fire for this to cause a problem. This fixes both of them. The first bug was that our rfcomm code would return DC_STATUS_SUCCESS with a zero-sized read when a timeout happened, or when the rfcomm socket had disconnected. That makes absolutely no sense. We should return DC_STATUS_TIMEOUT on timeout, and DC_STATUS_IO if the socket has disconnected without any data. The fix to this is to make the whole rfcomm iostream read logic much simpler: there's no need to loop at all for partial results, because the libdivecomputer iostream side will do the loop for us (and handle partial results much better: it knows if the target backend can handle those partial results or not). The second bug was in our libdivecomputer iostream read() function, which reacted very badly to this bad return value. This updates our libdivecomputer branch to one that is more careful about things. Reported-by: linuxcrash <albin@mrty.ch> Debugged-by: Jef Driesen <jef@libdivecomputer.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'core')
-rw-r--r--core/qtserialbluetooth.cpp51
1 files changed, 24 insertions, 27 deletions
diff --git a/core/qtserialbluetooth.cpp b/core/qtserialbluetooth.cpp
index 6b47524de..f6bbcafdf 100644
--- a/core/qtserialbluetooth.cpp
+++ b/core/qtserialbluetooth.cpp
@@ -133,42 +133,39 @@ static dc_status_t qt_serial_read(void *io, void* data, size_t size, size_t *act
{
qt_serial_t *device = (qt_serial_t*) io;
- if (device == NULL || device->socket == NULL)
+ if (device == NULL || device->socket == NULL || !actual)
return DC_STATUS_INVALIDARGS;
- size_t nbytes = 0;
- int rc;
+ *actual = 0;
+ for (;;) {
+ int rc;
- while(nbytes < size && device->socket->state() == QBluetoothSocket::ConnectedState)
- {
- rc = device->socket->read((char *) data + nbytes, size - nbytes);
+ if (device->socket->state() != QBluetoothSocket::ConnectedState)
+ return DC_STATUS_IO;
+ rc = device->socket->read((char *) data, size);
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;
+ continue;
+ return DC_STATUS_IO;
}
- nbytes += rc;
- }
-
- if (actual)
- *actual = nbytes;
+ *actual = rc;
+ if (rc > 0 || !size)
+ return DC_STATUS_SUCCESS;
+
+ // Timeout handling
+ 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();
- return DC_STATUS_SUCCESS;
+ if (!timer.isActive())
+ return DC_STATUS_TIMEOUT;
+ }
}
static dc_status_t qt_serial_write(void *io, const void* data, size_t size, size_t *actual)