summaryrefslogtreecommitdiffstats
path: root/src/network
diff options
context:
space:
mode:
authorShane Kearns <shane.kearns@accenture.com>2011-01-18 14:04:05 (GMT)
committerShane Kearns <shane.kearns@accenture.com>2011-01-18 17:04:52 (GMT)
commit4d4bc3f3729a057e428208b4037dddb49b14b1ed (patch)
tree2d1b90c61df4c72d373f295aad36b23700801728 /src/network
parent066ba528045435519917d37c781d04cea4ca4cfc (diff)
downloadQt-4d4bc3f3729a057e428208b4037dddb49b14b1ed.zip
Qt-4d4bc3f3729a057e428208b4037dddb49b14b1ed.tar.gz
Qt-4d4bc3f3729a057e428208b4037dddb49b14b1ed.tar.bz2
Fix some issues with socket handle allocation
QAbstractSocket requires a handle of -1 to be returned when accept fails. This is a common case, as it calls accept() in a loop to get all incoming connections. Also the blank socket wasn't being closed properly on failure so that is also fixed. Fixed a possible race condition in QSymbianSocketEngine::close - the socket descriptor is now deregistered before closing the symbian socket to avoid another thread getting the symbian handle reused in open and asserting when trying to register it. This patch also adds debug around socket handle allocation when socket engine debug is enabled. Reviewed-by: Markus Goetz
Diffstat (limited to 'src/network')
-rw-r--r--src/network/socket/qsymbiansocketengine.cpp35
1 files changed, 31 insertions, 4 deletions
diff --git a/src/network/socket/qsymbiansocketengine.cpp b/src/network/socket/qsymbiansocketengine.cpp
index 455bf5e..cf5f33e 100644
--- a/src/network/socket/qsymbiansocketengine.cpp
+++ b/src/network/socket/qsymbiansocketengine.cpp
@@ -208,8 +208,13 @@ bool QSymbianSocketEnginePrivate::createNewSocket(QAbstractSocket::SocketType so
return false;
}
-
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << "QSymbianSocketEnginePrivate::createNewSocket - created" << nativeSocket.SubSessionHandle();
+#endif
socketDescriptor = QSymbianSocketManager::instance().addSocket(nativeSocket);
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << " - allocated socket descriptor" << socketDescriptor;
+#endif
return true;
}
@@ -343,10 +348,15 @@ bool QSymbianSocketEngine::initialize(int socketDescriptor, QAbstractSocket::Soc
close();
if (!QSymbianSocketManager::instance().lookupSocket(socketDescriptor, d->nativeSocket)) {
+ qWarning("QSymbianSocketEngine::initialize - socket descriptor not found");
d->setError(QAbstractSocket::SocketResourceError,
QSymbianSocketEnginePrivate::InvalidSocketErrorString);
return false;
}
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << "QSymbianSocketEngine::initialize - attached to" << d->nativeSocket.SubSessionHandle() << socketDescriptor;
+#endif
+ Q_ASSERT(d->socketDescriptor == socketDescriptor || d->socketDescriptor == -1);
d->socketDescriptor = socketDescriptor;
// determine socket type and protocol
@@ -712,16 +722,28 @@ bool QSymbianSocketEngine::listen()
int QSymbianSocketEngine::accept()
{
Q_D(QSymbianSocketEngine);
+ Q_CHECK_VALID_SOCKETLAYER(QSymbianSocketEngine::accept(), -1);
+ Q_CHECK_STATE(QSymbianSocketEngine::accept(), QAbstractSocket::ListeningState, false);
+ Q_CHECK_TYPE(QSymbianSocketEngine::accept(), QAbstractSocket::TcpSocket, false);
RSocket blankSocket;
blankSocket.Open(d->socketServer);
TRequestStatus status;
d->nativeSocket.Accept(blankSocket, status);
User::WaitForRequest(status);
if (status.Int()) {
+ blankSocket.Close();
qWarning("QSymbianSocketEnginePrivate::nativeAccept() - error %d", status.Int());
- return 0;
+ return -1;
}
- return QSymbianSocketManager::instance().addSocket(blankSocket);
+
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << "QSymbianSocketEnginePrivate::accept - created" << blankSocket.SubSessionHandle();
+#endif
+ int fd = QSymbianSocketManager::instance().addSocket(blankSocket);
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << " - allocated socket descriptor" << fd;
+#endif
+ return fd;
}
qint64 QSymbianSocketEngine::bytesAvailable() const
@@ -930,8 +952,13 @@ void QSymbianSocketEngine::close()
d->nativeSocket.Shutdown(RSocket::EImmediate, stat);
User::WaitForRequest(stat);
}
- d->nativeSocket.Close();
+#ifdef QNATIVESOCKETENGINE_DEBUG
+ qDebug() << "QSymbianSocketEngine::close - closing socket" << d->nativeSocket.SubSessionHandle() << d->socketDescriptor;
+#endif
+ //remove must come before close to avoid a race where another thread gets the old subsession handle
+ //reused & asserts when calling QSymbianSocketManager::instance->addSocket
QSymbianSocketManager::instance().removeSocket(d->nativeSocket);
+ d->nativeSocket.Close();
d->socketDescriptor = -1;
d->socketState = QAbstractSocket::UnconnectedState;