summaryrefslogtreecommitdiffstats
path: root/lib/libtdekrb/src/tdekrbclientsocket.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'lib/libtdekrb/src/tdekrbclientsocket.cpp')
-rw-r--r--lib/libtdekrb/src/tdekrbclientsocket.cpp330
1 files changed, 200 insertions, 130 deletions
diff --git a/lib/libtdekrb/src/tdekrbclientsocket.cpp b/lib/libtdekrb/src/tdekrbclientsocket.cpp
index 2736f5a..24acb29 100644
--- a/lib/libtdekrb/src/tdekrbclientsocket.cpp
+++ b/lib/libtdekrb/src/tdekrbclientsocket.cpp
@@ -23,6 +23,8 @@
#include <tqapplication.h>
#include <tqbuffer.h>
+#include <tqeventloop.h>
+#include <tqtimer.h>
#include <sasl.h>
#include <saslplug.h>
@@ -33,17 +35,17 @@
#define NET_SEC_BUF_SIZE (2048)
// When control comes back from processEvents() my object may be completely gone! This attempts to mitigate the risk
-#define SAFELY_PROCESS_EVENTS if (!m_canary) { \
- m_canary = new bool; \
- *m_canary = false; \
- } \
- bool* canary = m_canary; \
- tqApp->processEvents(); \
- if (*canary == true) { \
- delete canary; \
- return -1; \
- } \
- delete m_canary; \
+#define SAFELY_PROCESS_EVENTS if (!m_canary) { \
+ m_canary = new bool; \
+ *m_canary = false; \
+ } \
+ bool* canary = m_canary; \
+ tqApp->eventLoop()->processEvents(TQEventLoop::ExcludeUserInput); \
+ if (*canary == true) { \
+ delete canary; \
+ return -1; \
+ } \
+ delete m_canary; \
m_canary = NULL;
/* exception handling */
@@ -93,7 +95,7 @@ static int logSASLMessages(void *context __attribute__((unused)), int priority,
return SASL_OK;
}
-TDEKerberosClientSocket::TDEKerberosClientSocket(TQObject *parent, const char *name) : TQSocket(parent, name), m_kerberosRequested(false), m_criticalSection(0), m_bufferLength(0), m_canary(NULL), m_negotiatedMaxBufferSize(NET_SEC_BUF_SIZE) {
+TDEKerberosClientSocket::TDEKerberosClientSocket(TQObject *parent, const char *name) : TQSocket(parent, name), m_kerberosRequested(false), m_criticalSection(0), m_bufferLength(0), m_krbInitRunning(false), m_krbInitState(-1), m_canary(NULL), m_negotiatedMaxBufferSize(NET_SEC_BUF_SIZE) {
saslData = new SASLDataPrivate;
saslData->m_krbConnection = NULL;
m_buffer = new TQBuffer();
@@ -122,7 +124,7 @@ void TDEKerberosClientSocket::close() {
}
void TDEKerberosClientSocket::flush() {
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
// FIXME
// If a write buffer is implemented, it will need to be flushed before the following call is made
TQSocket::flush();
@@ -135,7 +137,7 @@ void TDEKerberosClientSocket::flush() {
TQIODevice::Offset TDEKerberosClientSocket::size() const {
TQIODevice::Offset ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
ret = m_bufferLength;
}
else {
@@ -153,7 +155,7 @@ bool TDEKerberosClientSocket::at(TQIODevice::Offset off) {
long i;
bool ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
if (off > 0) {
// Prevent overflow
if (off > (unsigned long)m_bufferLength) {
@@ -184,7 +186,7 @@ bool TDEKerberosClientSocket::at(TQIODevice::Offset off) {
bool TDEKerberosClientSocket::atEnd() const {
bool ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
ret = TQSocket::atEnd();
}
else {
@@ -205,7 +207,7 @@ int TDEKerberosClientSocket::setUsingKerberos(bool krbactive) {
if (krbactive) {
m_kerberosRequested = true;
if ((!saslData->m_krbConnection) && (state() == TQSocket::Connected)) {
- ret = initializeKerberosInterface();
+ initializeKerberosInterface();
}
}
else {
@@ -230,7 +232,7 @@ TQ_LONG TDEKerberosClientSocket::readBlock(char *data, TQ_ULONG maxlen) {
long i;
TQ_LONG ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
int reclen;
int wrlen;
char* buf = (char*)malloc(m_negotiatedMaxBufferSize);
@@ -277,7 +279,7 @@ TQ_LONG TDEKerberosClientSocket::readBlock(char *data, TQ_ULONG maxlen) {
TQ_LONG TDEKerberosClientSocket::writeBlock(const char *data, TQ_ULONG len) {
TQ_LONG ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
ret = transmitEncryptedData(socket(), data, len);
}
else {
@@ -291,7 +293,7 @@ TQ_LONG TDEKerberosClientSocket::readLine(char *data, TQ_ULONG maxlen) {
long i;
TQ_LONG ret;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
int reclen;
int wrlen;
char* buf = (char*)malloc(m_negotiatedMaxBufferSize);
@@ -340,7 +342,7 @@ TQString TDEKerberosClientSocket::readLine() {
TQString ret;
long maxlen;
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
int reclen;
int wrlen;
int readlen;
@@ -393,7 +395,7 @@ TQString TDEKerberosClientSocket::readLine() {
}
void TDEKerberosClientSocket::writeLine(TQString str) {
- if (m_kerberosRequested) {
+ if (kerberosStatus() == KerberosInUse) {
transmitEncryptedData(socket(), str.ascii(), str.length());
}
else {
@@ -561,6 +563,168 @@ int TDEKerberosClientSocket::receiveEncryptedData(char *buf, unsigned int truncl
return recv_len;
}
+TDEKerberosClientSocket::KerberosStatus TDEKerberosClientSocket::kerberosStatus() const {
+ if (!m_kerberosRequested) {
+ return KerberosNotRequested;
+ }
+ if (m_krbInitRunning) {
+ return KerberosInitializing;
+ }
+ if (m_krbInitState < 0) {
+ return KerberosFailure;
+ }
+ return KerberosInUse;
+}
+
+void TDEKerberosClientSocket::continueKerberosInitialization() {
+ int slen;
+ char buf[NET_SEC_BUF_SIZE];
+ unsigned int len;
+ const char *data;
+ const char *chosenmech;
+ sasl_ssf_t *ssf;
+
+ if (m_krbInitRunning) {
+ switch (m_krbInitState) {
+ case 0:
+ if (state() == TQSocket::Connected) {
+ if (canReadLine()) {
+ printf("[DEBUG] Waiting for mechanism list from server...\n\r");
+ slen = getSASLDataFromNetwork(buf, NET_SEC_BUF_SIZE);
+ if (slen < 0) {
+ m_krbInitState = -2;
+ m_krbInitRunning = false;
+ return;
+ }
+ len = slen;
+
+ printf("Choosing best mechanism from: %s\n", buf);
+
+ m_krbInitResult = sasl_client_start(saslData->m_krbConnection, buf, NULL, &data, &len, &chosenmech);
+ if (m_krbInitResult != SASL_OK && m_krbInitResult != SASL_CONTINUE) {
+ printf("[ERROR] Starting SASL negotiation returned %s (%d)\n\r", sasl_errstring(m_krbInitResult, NULL, NULL), m_krbInitResult);
+ freeKerberosConnection();
+ m_krbInitState = -1;
+ m_krbInitRunning = false;
+ return;
+ }
+
+ printf("[DEBUG] Using mechanism %s\n\r", chosenmech);
+ strcpy(buf, chosenmech);
+ if (data) {
+ if (NET_SEC_BUF_SIZE - strlen(buf) - 1 < len) {
+ printf("[ERROR] Insufficient buffer space to construct initial response!\n\r");
+ freeKerberosConnection();
+ m_krbInitState = -1;
+ m_krbInitRunning = false;
+ return;
+ }
+ printf("[DEBUG] Preparing initial response...\n\r");
+ memcpy(buf + strlen(buf) + 1, data, len);
+ len += (unsigned) strlen(buf) + 1;
+ data = NULL;
+ }
+ else {
+ len = (unsigned) strlen(buf);
+ }
+
+ printf("[DEBUG] Sending initial response...\n\r");
+ sendSASLDataToNetwork(buf, len, socket());
+
+ m_krbInitState = 1;
+ }
+ }
+ else {
+ m_krbInitState = -3;
+ m_krbInitRunning = false;
+ }
+ break;
+ case 1:
+ if (state() == TQSocket::Connected) {
+ if (m_krbInitResult == SASL_CONTINUE) {
+ if (canReadLine()) {
+ printf("[DEBUG] Waiting for server reply...\n\r");
+ slen = getSASLDataFromNetwork(buf, NET_SEC_BUF_SIZE);
+ if (slen < 0) {
+ m_krbInitState = -2;
+ m_krbInitRunning = false;
+ return;
+ }
+ len = slen;
+ m_krbInitResult = sasl_client_step(saslData->m_krbConnection, buf, len, NULL, &data, &len);
+ if (m_krbInitResult != SASL_OK && m_krbInitResult != SASL_CONTINUE) {
+ printf("[ERROR] Performing SASL negotiation returned %s (%d)\n\r", sasl_errstring(m_krbInitResult, NULL, NULL), m_krbInitResult);
+ freeKerberosConnection();
+ m_krbInitState = -1;
+ m_krbInitRunning = false;
+ return;
+ }
+ if (data && len) {
+ printf("[DEBUG] Sending response...\n\r");
+ sendSASLDataToNetwork(data, len, socket());
+ }
+ else if (m_krbInitResult != SASL_OK || !m_krbInitServerLast) {
+ sendSASLDataToNetwork("", 0, socket());
+ }
+ }
+ }
+ else {
+ printf("[DEBUG] Negotiation complete!\n\r");
+ m_krbInitState = 2;
+ }
+ }
+ else {
+ m_krbInitState = -3;
+ m_krbInitRunning = false;
+ }
+ break;
+ case 2:
+ if (state() == TQSocket::Connected) {
+ m_krbInitResult = sasl_getprop(saslData->m_krbConnection, SASL_USERNAME, (const void **)&data);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[WARNING] Unable to determine authenticated username!\n\r");
+ }
+ else {
+ printf("[DEBUG] Authenticated username: %s\n\r", data ? data : "(NULL)");
+ }
+
+ m_krbInitResult = sasl_getprop(saslData->m_krbConnection, SASL_DEFUSERREALM, (const void **)&data);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[WARNING] Unable to determine authenticated realm!\n\r");
+ }
+ else {
+ printf("[DEBUG] Authenticated realm: %s\n\r", data ? data : "(NULL)");
+ }
+
+ m_krbInitResult = sasl_getprop(saslData->m_krbConnection, SASL_SSF, (const void **)&ssf);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[WARNING] Unable to determine SSF!\n\r");
+ }
+ else {
+ printf("[DEBUG] Authenticated SSF: %d\n", *ssf);
+ }
+
+ m_krbInitResult = sasl_getprop(saslData->m_krbConnection, SASL_MAXOUTBUF, (const void **)&m_negotiatedMaxBufferSize);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[WARNING] Unable to determine maximum buffer size!\n\r");
+ m_negotiatedMaxBufferSize = NET_SEC_BUF_SIZE;
+ }
+ else {
+ printf("[DEBUG] Maximum buffer size: %d\n", m_negotiatedMaxBufferSize);
+ }
+ m_krbInitState = 3;
+ m_krbInitRunning = false;
+ }
+ else {
+ m_krbInitState = -3;
+ m_krbInitRunning = false;
+ }
+ break;
+ }
+ TQTimer::singleShot(0, this, SLOT(continueKerberosInitialization()));
+ }
+}
+
int TDEKerberosClientSocket::initializeKerberosInterface() {
if (state() != TQSocket::Connected) {
saslData->m_krbConnection = false;
@@ -568,15 +732,9 @@ int TDEKerberosClientSocket::initializeKerberosInterface() {
}
sasl_callback_t *callback;
- char buf[NET_SEC_BUF_SIZE];
- int result = 0;
- int serverlast = 0;
+ m_krbInitResult = 0;
+ m_krbInitServerLast = 0;
sasl_security_properties_t secprops;
- const char *chosenmech;
- unsigned int len;
- int slen;
- const char *data;
- sasl_ssf_t *ssf;
char *iplocal = NULL;
char *ipremote = NULL;
const char *service = m_serviceName.ascii();
@@ -601,116 +759,28 @@ int TDEKerberosClientSocket::initializeKerberosInterface() {
secprops.maxbufsize = NET_SEC_BUF_SIZE;
secprops.max_ssf = UINT_MAX;
- result = sasl_client_init(saslData->m_callbacks);
- if (result != SASL_OK) {
- printf("[ERROR] Initializing libsasl returned %s (%d)\n\r", sasl_errstring(result, NULL, NULL), result);
- return -1;
- }
-
- result = sasl_client_new(service, fqdn, iplocal, ipremote, NULL, serverlast, &saslData->m_krbConnection);
- if (result != SASL_OK) {
- printf("[ERROR] Allocating sasl connection state returned %s (%d)\n\r", sasl_errstring(result, NULL, NULL), result);
+ m_krbInitResult = sasl_client_init(saslData->m_callbacks);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[ERROR] Initializing libsasl returned %s (%d)\n\r", sasl_errstring(m_krbInitResult, NULL, NULL), m_krbInitResult);
return -1;
}
- result = sasl_setprop(saslData->m_krbConnection, SASL_SEC_PROPS, &secprops);
- if (result != SASL_OK) {
- printf("[ERROR] Setting security properties returned %s (%d)\n\r", sasl_errstring(result, NULL, NULL), result);
- freeKerberosConnection();
+ m_krbInitResult = sasl_client_new(service, fqdn, iplocal, ipremote, NULL, m_krbInitServerLast, &saslData->m_krbConnection);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[ERROR] Allocating sasl connection state returned %s (%d)\n\r", sasl_errstring(m_krbInitResult, NULL, NULL), m_krbInitResult);
return -1;
}
- printf("[DEBUG] Waiting for mechanism list from server...\n\r");
- slen = getSASLDataFromNetwork(buf, NET_SEC_BUF_SIZE);
- if (slen < 0) {
- return -2;
- }
- len = slen;
-
- printf("Choosing best mechanism from: %s\n", buf);
-
- result = sasl_client_start(saslData->m_krbConnection, buf, NULL, &data, &len, &chosenmech);
- if (result != SASL_OK && result != SASL_CONTINUE) {
- printf("[ERROR] Starting SASL negotiation returned %s (%d)\n\r", sasl_errstring(result, NULL, NULL), result);
+ m_krbInitResult = sasl_setprop(saslData->m_krbConnection, SASL_SEC_PROPS, &secprops);
+ if (m_krbInitResult != SASL_OK) {
+ printf("[ERROR] Setting security properties returned %s (%d)\n\r", sasl_errstring(m_krbInitResult, NULL, NULL), m_krbInitResult);
freeKerberosConnection();
return -1;
}
- printf("[DEBUG] Using mechanism %s\n\r", chosenmech);
- strcpy(buf, chosenmech);
- if (data) {
- if (NET_SEC_BUF_SIZE - strlen(buf) - 1 < len) {
- printf("[ERROR] Insufficient buffer space to construct initial response!\n\r");
- freeKerberosConnection();
- return -1;
- }
- printf("[DEBUG] Preparing initial response...\n\r");
- memcpy(buf + strlen(buf) + 1, data, len);
- len += (unsigned) strlen(buf) + 1;
- data = NULL;
- }
- else {
- len = (unsigned) strlen(buf);
- }
-
- printf("[DEBUG] Sending initial response...\n\r");
- sendSASLDataToNetwork(buf, len, socket());
-
- while (result == SASL_CONTINUE) {
- printf("[DEBUG] Waiting for server reply...\n\r");
- slen = getSASLDataFromNetwork(buf, NET_SEC_BUF_SIZE);
- if (slen < 0) {
- return -2;
- }
- len = slen;
- result = sasl_client_step(saslData->m_krbConnection, buf, len, NULL, &data, &len);
- if (result != SASL_OK && result != SASL_CONTINUE) {
- printf("[ERROR] Performing SASL negotiation returned %s (%d)\n\r", sasl_errstring(result, NULL, NULL), result);
- freeKerberosConnection();
- return -1;
- }
- if (data && len) {
- printf("[DEBUG] Sending response...\n\r");
- sendSASLDataToNetwork(data, len, socket());
- }
- else if (result != SASL_OK || !serverlast) {
- sendSASLDataToNetwork("", 0, socket());
- }
- }
- printf("[DEBUG] Negotiation complete!\n\r");
-
- result = sasl_getprop(saslData->m_krbConnection, SASL_USERNAME, (const void **)&data);
- if (result != SASL_OK) {
- printf("[WARNING] Unable to determine authenticated username!\n\r");
- }
- else {
- printf("[DEBUG] Authenticated username: %s\n\r", data ? data : "(NULL)");
- }
-
- result = sasl_getprop(saslData->m_krbConnection, SASL_DEFUSERREALM, (const void **)&data);
- if (result != SASL_OK) {
- printf("[WARNING] Unable to determine authenticated realm!\n\r");
- }
- else {
- printf("[DEBUG] Authenticated realm: %s\n\r", data ? data : "(NULL)");
- }
-
- result = sasl_getprop(saslData->m_krbConnection, SASL_SSF, (const void **)&ssf);
- if (result != SASL_OK) {
- printf("[WARNING] Unable to determine SSF!\n\r");
- }
- else {
- printf("[DEBUG] Authenticated SSF: %d\n", *ssf);
- }
-
- result = sasl_getprop(saslData->m_krbConnection, SASL_MAXOUTBUF, (const void **)&m_negotiatedMaxBufferSize);
- if (result != SASL_OK) {
- printf("[WARNING] Unable to determine maximum buffer size!\n\r");
- m_negotiatedMaxBufferSize = NET_SEC_BUF_SIZE;
- }
- else {
- printf("[DEBUG] Maximum buffer size: %d\n", m_negotiatedMaxBufferSize);
- }
+ m_krbInitRunning = true;
+ m_krbInitState = 0;
+ TQTimer::singleShot(0, this, SLOT(continueKerberosInitialization()));
return 0;
} \ No newline at end of file