Как сделать неблокирующее соединение OpenSSL?
Я хочу сделать неблокирующее соединение OpenSSL
При этом соединении - если данные недоступны для чтения, весь процесс выполнения программы останавливается на SSL_read(). Я хочу, чтобы, если нет данных, доступных для чтения, он дал мне возвращаемые значения, например WANT_READ, и я не знаю больше доступных данных.
char *sslRead (connection *c)
{
const int readSize = 1024;
char *rc = NULL;
int r;
int received, count = 0;
int ReallocSize = 0;
char buffer[1024];
if (c)
{
while (1)
{
if (!rc)
{
rc = malloc (readSize + 1);
if (rc == NULL)
printf("the major error have happen. leave program\n");
}
else
{
ReallocSize = (count + 1) * (readSize + 1);
rc = realloc (rc, ReallocSize);
}
// if i have no data available for read after reading data,
// this call will not return anything and wait for more data
// i want change this non blocking connections
received = SSL_read (c->sslHandle, buffer, readSize);
buffer[received] = '\0';
if (received <= 0)
{
printf(" received equal to or less than 0\n");
switch (SSL_get_error(c->sslHandle, r))
{
case SSL_ERROR_NONE:
printf("SSL_ERROR_NONE\n");
break;
case SSL_ERROR_ZERO_RETURN:
printf("SSL_ERROR_ZERO_RETURN\n");
break;
case SSL_ERROR_WANT_READ:
printf("SSL_ERROR_WANT_READ\n");
break;
default:
printf("error happens %i\n", r);
}
break;
}
count++;
}
}
return rc;
}
вот как я устанавливаю соединение
connection *sslConnect (void)
{
connection *c;
c = malloc (sizeof (connection));
c->sslHandle = NULL;
c->sslContext = NULL;
c->socket = tcpConnect ();
if (c->socket)
{
// Register the error strings for libcrypto & libssl
SSL_load_error_strings ();
// Register the available ciphers and digests
SSL_library_init ();
// New context saying we are a client, and using SSL 2 or 3
c->sslContext = SSL_CTX_new (SSLv23_client_method ());
if (c->sslContext == NULL)
ERR_print_errors_fp (stderr);
// Create an SSL struct for the connection
c->sslHandle = SSL_new (c->sslContext);
if (c->sslHandle == NULL)
ERR_print_errors_fp (stderr);
// Connect the SSL struct to our connection
if (!SSL_set_fd (c->sslHandle, c->socket))
ERR_print_errors_fp (stderr);
// Initiate SSL handshake
if (SSL_connect (c->sslHandle) != 1)
ERR_print_errors_fp (stderr);
}
else
{
perror ("Connect failed");
}
return c;
}
большое вам спасибо.
1 ответ
Создание неблокирующего сокета является необходимым условием для неблокирующего соединения...
Подводя итог следующим шагам: (см. Полное описание на сайте, ссылка ниже)
1) Вызовите API fcntl (), чтобы получить текущие настройки флага дескриптора сокета в локальную переменную.
2) В нашей локальной переменной установите флаг O_NONBLOCK (неблокирующий). (будьте осторожны, конечно, чтобы не вмешиваться в другие флаги)
3) Вызвать API fcntl (), чтобы установить для флагов дескриптора значение в нашей локальной переменной.
( подробнее о технике неблокирующих сокетов читайте здесь)
Предполагая существующий сокет, следующие реализуют шаги, описанные выше:
BOOL SetSocketBlockingEnabled(SOCKET fd, BOOL blocking)
{
if (fd < 0) return FALSE;
#ifdef WIN32
unsigned long mode = blocking ? 0 : 1;
return (ioctlsocket(fd, FIONBIO, &mode) == 0) ? TRUE : FALSE;
#else
int flags = fcntl(fd, F_GETFL, 0);
if (flags < 0) return false;
flags = blocking ? (flags&~O_NONBLOCK) : (flags|O_NONBLOCK);
return (fcntl(fd, F_SETFL, flags) == 0) ? TRUE : FALSE;
#endif
}
Если у вас есть неблокирующий сокет, посмотрите этот пост, объясняющий, как выполнить неблокирующее соединение.