summaryrefslogtreecommitdiffstats
path: root/src/network/kernel
diff options
context:
space:
mode:
authorShane Kearns <shane.kearns@accenture.com>2011-04-04 17:26:00 (GMT)
committerShane Kearns <shane.kearns@accenture.com>2011-04-05 10:28:39 (GMT)
commita0ca845cd32178d789f4eeff602b283a290842b8 (patch)
treef87bf1099387f2df499826db00d7e3fa482065c3 /src/network/kernel
parent3b395c6e45ed4f73503586e6b9a80bb11844d315 (diff)
downloadQt-a0ca845cd32178d789f4eeff602b283a290842b8.zip
Qt-a0ca845cd32178d789f4eeff602b283a290842b8.tar.gz
Qt-a0ca845cd32178d789f4eeff602b283a290842b8.tar.bz2
Thread safety for QHostInfo symbian implementation
Each thread needs at least one current request if it has any queued requests, this is to stop the queue stalling. When starting a queued request, start it in the same thread it belongs to When aborting a request from the wrong thread, just detach it (it will complete normally but the slot isn't connected, and then delete itself) Reviewed-by: Markus Goetz
Diffstat (limited to 'src/network/kernel')
-rw-r--r--src/network/kernel/qhostinfo_p.h2
-rw-r--r--src/network/kernel/qhostinfo_symbian.cpp54
2 files changed, 46 insertions, 10 deletions
diff --git a/src/network/kernel/qhostinfo_p.h b/src/network/kernel/qhostinfo_p.h
index bae6efa..8da0692 100644
--- a/src/network/kernel/qhostinfo_p.h
+++ b/src/network/kernel/qhostinfo_p.h
@@ -234,6 +234,7 @@ public:
~QSymbianHostResolver();
void requestHostLookup();
+ void abortHostLookup();
int id();
void returnResults();
@@ -301,6 +302,7 @@ public:
private:
void runNextLookup();
+ // this is true for single threaded use, with multiple threads the max is ((number of threads) + KMaxConcurrentLookups - 1)
static const int KMaxConcurrentLookups = 5;
QList<QSymbianHostResolver*> iCurrentLookups;
diff --git a/src/network/kernel/qhostinfo_symbian.cpp b/src/network/kernel/qhostinfo_symbian.cpp
index ded2416..dcfc490 100644
--- a/src/network/kernel/qhostinfo_symbian.cpp
+++ b/src/network/kernel/qhostinfo_symbian.cpp
@@ -311,6 +311,23 @@ void QSymbianHostResolver::requestHostLookup()
}
}
+void QSymbianHostResolver::abortHostLookup()
+{
+ if (resultEmitter.thread() == QThread::currentThread()) {
+#ifdef QHOSTINFO_DEBUG
+ qDebug("QSymbianHostResolver::abortHostLookup - deleting %d", id());
+#endif
+ //normal case, abort from same thread it was started
+ delete this; //will cancel outstanding request
+ } else {
+#ifdef QHOSTINFO_DEBUG
+ qDebug("QSymbianHostResolver::abortHostLookup - detaching %d", id());
+#endif
+ //abort from different thread, carry on but don't report the results
+ resultEmitter.disconnect();
+ }
+}
+
void QSymbianHostResolver::DoCancel()
{
#if defined(QHOSTINFO_DEBUG)
@@ -462,9 +479,10 @@ void QSymbianHostInfoLookupManager::clear()
#if defined(QHOSTINFO_DEBUG)
qDebug() << "QSymbianHostInfoLookupManager::clear" << QThread::currentThreadId();
#endif
- //TODO: these aren't deleted because of thread unsafety, but that is a behaviour difference
- //qDeleteAll(iCurrentLookups);
- //qDeleteAll(iScheduledLookups);
+ foreach (QSymbianHostResolver *hr, iCurrentLookups)
+ hr->abortHostLookup();
+ iCurrentLookups.clear();
+ qDeleteAll(iScheduledLookups);
cache.clear();
}
@@ -493,12 +511,17 @@ void QSymbianHostInfoLookupManager::runNextLookup()
qDebug() << "QSymbianHostInfoLookupManager::runNextLookup" << QThread::currentThreadId() << "current" << iCurrentLookups.count() << "queued" << iScheduledLookups.count();
#endif
// check to see if there are any scheduled lookups
- if (iScheduledLookups.count() > 0) {
- // if so, move one to the current lookups and run it
- // FIFO
- QSymbianHostResolver* hostResolver = iScheduledLookups.takeFirst();
- iCurrentLookups.append(hostResolver);
- hostResolver->requestHostLookup();
+ for (int i=0; i<iScheduledLookups.count(); i++) {
+ QSymbianHostResolver* hostResolver = iScheduledLookups.at(i);
+ if (hostResolver->resultEmitter.thread() == QThread::currentThread()) {
+ // if so, move one to the current lookups and run it
+ iCurrentLookups.append(hostResolver);
+ iScheduledLookups.removeAt(i);
+ hostResolver->requestHostLookup();
+ // if spare capacity, try to start another one
+ if (iCurrentLookups.count() >= KMaxConcurrentLookups)
+ break;
+ }
}
}
@@ -511,7 +534,18 @@ void QSymbianHostInfoLookupManager::scheduleLookup(QSymbianHostResolver* r)
qDebug() << "QSymbianHostInfoLookupManager::scheduleLookup" << QThread::currentThreadId() << r->id() << "current" << iCurrentLookups.count() << "queued" << iScheduledLookups.count();
#endif
// Check to see if we have space on the current lookups pool.
+ bool defer = false;
if (iCurrentLookups.count() >= KMaxConcurrentLookups) {
+ // busy, defer unless there are no request in this thread
+ // at least one active request per thread with queued requests is needed
+ for (int i=0; i < iCurrentLookups.count();i++) {
+ if (iCurrentLookups.at(i)->resultEmitter.thread() == QThread::currentThread()) {
+ defer = true;
+ break;
+ }
+ }
+ }
+ if (defer) {
// If no, schedule for later.
iScheduledLookups.append(r);
#if defined(QHOSTINFO_DEBUG)
@@ -541,7 +575,7 @@ void QSymbianHostInfoLookupManager::abortLookup(int id)
if (id == iCurrentLookups[i]->id()) {
QSymbianHostResolver* r = iCurrentLookups.at(i);
iCurrentLookups.removeAt(i);
- delete r; //cancels via destructor
+ r->abortHostLookup();
runNextLookup();
return;
}