diff options
author | Berthold Stoeger <bstoeger@mail.tuwien.ac.at> | 2018-10-11 20:46:44 +0200 |
---|---|---|
committer | Dirk Hohndel <dirk@hohndel.org> | 2018-10-11 16:25:02 -0700 |
commit | 04ad9c885da672b476d3abf4bc33ba172868eb61 (patch) | |
tree | 3f0da41213aae9b7492b53fa4807f80294800d8f | |
parent | cb2f09a8e3673e420970d1d2c0f5c1688a0e1fbf (diff) | |
download | subsurface-04ad9c885da672b476d3abf4bc33ba172868eb61.tar.gz |
Geo lookup: make the reply a managed-pointer
By making reply a std::unique_ptr<>, the function can be quit
from any point and the reply will be freed. This is valid according
to Qt's documentation as we're not deleting during signal processing.
This commit fixes a leak: reply was overwritten with the address of
a new object without freeing the old object.
Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
-rw-r--r-- | core/divesitehelpers.cpp | 32 |
1 files changed, 17 insertions, 15 deletions
diff --git a/core/divesitehelpers.cpp b/core/divesitehelpers.cpp index 7a7176624..264d6ae74 100644 --- a/core/divesitehelpers.cpp +++ b/core/divesitehelpers.cpp @@ -19,6 +19,7 @@ #include <QUrlQuery> #include <QEventLoop> #include <QTimer> +#include <memory> void reverseGeoLookup() { @@ -41,9 +42,13 @@ void reverseGeoLookup() // first check the findNearbyPlaces API from geonames - that should give us country, state, city request.setUrl(geonamesURL.arg(uiLanguage(NULL).section(QRegExp("[-_ ]"), 0, 0)).arg(ds->latitude.udeg / 1000000.0).arg(ds->longitude.udeg / 1000000.0)); - QNetworkReply *reply = rgl.get(request); + // By using a std::unique_ptr<>, we can exit from the function at any point + // and the reply will be freed. Likewise, when overwriting the pointer with + // a new value. According to Qt's documentation it is fine the delete the + // reply as long as we're not in a slot connected to error() or finish(). + std::unique_ptr<QNetworkReply> reply(rgl.get(request)); timer.setSingleShot(true); - QObject::connect(reply, SIGNAL(finished()), &loop, SLOT(quit())); + QObject::connect(&*reply, SIGNAL(finished()), &loop, SLOT(quit())); timer.start(5000); // 5 secs. timeout loop.exec(); @@ -51,17 +56,17 @@ void reverseGeoLookup() timer.stop(); if (reply->error() > 0) { report_error("got error accessing geonames.org: %s", qPrintable(reply->errorString())); - goto clear_reply; + return; } int v = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt(); if (v < 200 || v >= 300) - goto clear_reply; + return; QByteArray fullReply = reply->readAll(); QJsonParseError errorObject; QJsonDocument jsonDoc = QJsonDocument::fromJson(fullReply, &errorObject); if (errorObject.error != QJsonParseError::NoError) { report_error("error parsing geonames.org response: %s", qPrintable(errorObject.errorString())); - goto clear_reply; + return; } QJsonObject obj = jsonDoc.object(); QVariant geoNamesObject = obj.value("geonames").toVariant(); @@ -112,30 +117,30 @@ void reverseGeoLookup() } } else { report_error("timeout accessing geonames.org"); - QObject::disconnect(reply, SIGNAL(finished()), &loop, SLOT(quit())); + QObject::disconnect(&*reply, SIGNAL(finished()), &loop, SLOT(quit())); reply->abort(); } // next check the oceans API to figure out the body of water request.setUrl(geonamesOceanURL.arg(uiLanguage(NULL).section(QRegExp("[-_ ]"), 0, 0)).arg(ds->latitude.udeg / 1000000.0).arg(ds->longitude.udeg / 1000000.0)); - reply = rgl.get(request); - QObject::connect(reply, SIGNAL(finished()), &loop, SLOT(quit())); + reply.reset(rgl.get(request)); // Note: frees old reply. + QObject::connect(&*reply, SIGNAL(finished()), &loop, SLOT(quit())); timer.start(5000); // 5 secs. timeout loop.exec(); if (timer.isActive()) { timer.stop(); if (reply->error() > 0) { report_error("got error accessing oceans API of geonames.org: %s", qPrintable(reply->errorString())); - goto clear_reply; + return; } int v = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt(); if (v < 200 || v >= 300) - goto clear_reply; + return; QByteArray fullReply = reply->readAll(); QJsonParseError errorObject; QJsonDocument jsonDoc = QJsonDocument::fromJson(fullReply, &errorObject); if (errorObject.error != QJsonParseError::NoError) { report_error("error parsing geonames.org response: %s", qPrintable(errorObject.errorString())); - goto clear_reply; + return; } QJsonObject obj = jsonDoc.object(); QVariant oceanObject = obj.value("ocean").toVariant(); @@ -158,10 +163,7 @@ void reverseGeoLookup() } } else { report_error("timeout accessing geonames.org"); - QObject::disconnect(reply, SIGNAL(finished()), &loop, SLOT(quit())); + QObject::disconnect(&*reply, SIGNAL(finished()), &loop, SLOT(quit())); reply->abort(); } - -clear_reply: - reply->deleteLater(); } |