summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Berthold Stoeger <bstoeger@mail.tuwien.ac.at>2018-10-11 20:46:44 +0200
committerGravatar Dirk Hohndel <dirk@hohndel.org>2018-10-11 16:25:02 -0700
commit04ad9c885da672b476d3abf4bc33ba172868eb61 (patch)
tree3f0da41213aae9b7492b53fa4807f80294800d8f
parentcb2f09a8e3673e420970d1d2c0f5c1688a0e1fbf (diff)
downloadsubsurface-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.cpp32
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();
}