f193616646
OBS-URL: https://build.opensuse.org/request/show/478584 OBS-URL: https://build.opensuse.org/package/show/openSUSE:Factory/pidgin?expand=0&rev=116
981 lines
24 KiB
Diff
981 lines
24 KiB
Diff
--- a/libpurple/plugins/ssl/ssl-gnutls.c
|
|
+++ b/libpurple/plugins/ssl/ssl-gnutls.c
|
|
@@ -435,8 +435,6 @@ ssl_gnutls_close(PurpleSslConnection *gs
|
|
if (gnutls_data->handshake_timer)
|
|
purple_timeout_remove(gnutls_data->handshake_timer);
|
|
|
|
- gnutls_bye(gnutls_data->session, GNUTLS_SHUT_RDWR);
|
|
-
|
|
gnutls_deinit(gnutls_data->session);
|
|
|
|
g_free(gnutls_data);
|
|
@@ -524,6 +522,22 @@ ssl_gnutls_read(PurpleSslConnection *gsc
|
|
}
|
|
|
|
static size_t
|
|
+ssl_gnutls_read_nonblock(PurpleSslConnection *gsc, void *data, size_t len)
|
|
+{
|
|
+ PurpleSslGnutlsData *gnutls_data = PURPLE_SSL_GNUTLS_DATA(gsc);
|
|
+ int s;
|
|
+
|
|
+ s = gnutls_record_recv(gnutls_data->session, data, len);
|
|
+
|
|
+ if (s == GNUTLS_E_AGAIN)
|
|
+ s = PURPLE_SSL_IO_AGAIN;
|
|
+ else if (s == GNUTLS_E_INTERRUPTED)
|
|
+ s = PURPLE_SSL_IO_INTERRUPTED;
|
|
+
|
|
+ return s;
|
|
+}
|
|
+
|
|
+static size_t
|
|
ssl_gnutls_write(PurpleSslConnection *gsc, const void *data, size_t len)
|
|
{
|
|
PurpleSslGnutlsData *gnutls_data = PURPLE_SSL_GNUTLS_DATA(gsc);
|
|
@@ -1296,6 +1310,23 @@ static PurpleCertificateScheme x509_gnut
|
|
x509_compare_pubkeys, /* Compare public keys */
|
|
};
|
|
|
|
+static size_t
|
|
+ssl_gnutls_write_nonblock(PurpleSslConnection *gsc, const void *data, size_t len)
|
|
+{
|
|
+ PurpleSslGnutlsData *gnutls_data = PURPLE_SSL_GNUTLS_DATA(gsc);
|
|
+ size_t s = 0;
|
|
+
|
|
+ if(gnutls_data)
|
|
+ s = gnutls_record_send(gnutls_data->session, data, len);
|
|
+
|
|
+ if (s == GNUTLS_E_AGAIN)
|
|
+ s = PURPLE_SSL_IO_AGAIN;
|
|
+ else if (s == GNUTLS_E_INTERRUPTED)
|
|
+ s = PURPLE_SSL_IO_INTERRUPTED;
|
|
+
|
|
+ return s;
|
|
+}
|
|
+
|
|
static PurpleSslOps ssl_ops =
|
|
{
|
|
ssl_gnutls_init,
|
|
@@ -1303,7 +1334,9 @@ static PurpleSslOps ssl_ops =
|
|
ssl_gnutls_connect,
|
|
ssl_gnutls_close,
|
|
ssl_gnutls_read,
|
|
+ ssl_gnutls_read_nonblock,
|
|
ssl_gnutls_write,
|
|
+ ssl_gnutls_write_nonblock,
|
|
ssl_gnutls_get_peer_certificates,
|
|
|
|
/* padding */
|
|
--- a/libpurple/plugins/ssl/ssl-nss.c
|
|
+++ b/libpurple/plugins/ssl/ssl-nss.c
|
|
@@ -1235,6 +1235,8 @@ static PurpleSslOps ssl_ops =
|
|
ssl_nss_connect,
|
|
ssl_nss_close,
|
|
ssl_nss_read,
|
|
+ ssl_nss_read,
|
|
+ ssl_nss_write,
|
|
ssl_nss_write,
|
|
ssl_nss_peer_certs,
|
|
|
|
--- /dev/null
|
|
+++ b/libpurple/purple-fifo.c
|
|
@@ -0,0 +1,232 @@
|
|
+/*
|
|
+ * @file purple-fifo.c A FIFO byte queue.
|
|
+ * @ingroup core
|
|
+ *
|
|
+ * Pidgin is the legal property of its developers, whose names are too numerous
|
|
+ * to list here. Please refer to the COPYRIGHT file distributed with this
|
|
+ * source distribution.
|
|
+ *
|
|
+ * This program is free software; you can redistribute it and/or modify
|
|
+ * it under the terms of the GNU General Public License as published by
|
|
+ * the Free Software Foundation; either version 2 of the License, or
|
|
+ * (at your option) any later version.
|
|
+ *
|
|
+ * This program is distributed in the hope that it will be useful,
|
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
+ * GNU General Public License for more details.
|
|
+ *
|
|
+ * You should have received a copy of the GNU General Public License
|
|
+ * along with this program; if not, write to the Free Software
|
|
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
+ */
|
|
+#include "internal.h"
|
|
+
|
|
+#define BUFFER_SIZE 256
|
|
+
|
|
+typedef struct
|
|
+{
|
|
+ GList *buffer_list;
|
|
+ GList *enqueue_buffer;
|
|
+ gint enqueue_pos;
|
|
+ GList *dequeue_buffer;
|
|
+ gint dequeue_pos;
|
|
+
|
|
+ gint total_used;
|
|
+}
|
|
+PurpleFifo;
|
|
+
|
|
+PurpleFifo *
|
|
+purple_fifo_new (void)
|
|
+{
|
|
+ return g_new0 (PurpleFifo, 1);
|
|
+}
|
|
+
|
|
+void
|
|
+purple_fifo_destroy (PurpleFifo *fifo)
|
|
+{
|
|
+ GList *l;
|
|
+
|
|
+ for (l = fifo->buffer_list; l; l = g_list_next (l))
|
|
+ g_free (l->data);
|
|
+
|
|
+ g_list_free (fifo->buffer_list);
|
|
+}
|
|
+
|
|
+gint
|
|
+purple_fifo_get_used (PurpleFifo *fifo)
|
|
+{
|
|
+ return fifo->total_used;
|
|
+}
|
|
+
|
|
+static void
|
|
+ensure_buffers (PurpleFifo *fifo)
|
|
+{
|
|
+ if (!fifo->buffer_list)
|
|
+ {
|
|
+ fifo->buffer_list = g_list_append (NULL, g_malloc (BUFFER_SIZE));
|
|
+ fifo->enqueue_buffer = fifo->buffer_list;
|
|
+ fifo->dequeue_buffer = fifo->buffer_list;
|
|
+
|
|
+ fifo->buffer_list->data = g_malloc (BUFFER_SIZE);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+enqueue_next_buffer (PurpleFifo *fifo)
|
|
+{
|
|
+ g_list_append (fifo->enqueue_buffer, g_malloc (BUFFER_SIZE));
|
|
+ fifo->enqueue_buffer = g_list_next (fifo->enqueue_buffer);
|
|
+ fifo->enqueue_pos = 0;
|
|
+}
|
|
+
|
|
+static void
|
|
+dequeue_next_buffer (PurpleFifo *fifo)
|
|
+{
|
|
+ g_free (fifo->dequeue_buffer->data);
|
|
+ fifo->dequeue_buffer = g_list_delete_link (fifo->dequeue_buffer, fifo->dequeue_buffer);
|
|
+
|
|
+ fifo->buffer_list = fifo->dequeue_buffer;
|
|
+ if (!fifo->buffer_list)
|
|
+ fifo->enqueue_buffer = NULL;
|
|
+
|
|
+ fifo->dequeue_pos = 0;
|
|
+}
|
|
+
|
|
+void
|
|
+purple_fifo_enqueue (PurpleFifo *fifo, gconstpointer libpurple, gint n)
|
|
+{
|
|
+ gint in_pos;
|
|
+
|
|
+ ensure_buffers (fifo);
|
|
+
|
|
+ for (in_pos = 0; in_pos < n; )
|
|
+ {
|
|
+ gint n_copied = MIN (BUFFER_SIZE - fifo->enqueue_pos, n - in_pos);
|
|
+
|
|
+ memcpy (((guint8 *) fifo->enqueue_buffer->data) + fifo->enqueue_pos, ((const guint8 *) libpurple) + in_pos, n_copied);
|
|
+
|
|
+ fifo->enqueue_pos += n_copied;
|
|
+ in_pos += n_copied;
|
|
+
|
|
+ if (fifo->enqueue_pos == BUFFER_SIZE)
|
|
+ enqueue_next_buffer (fifo);
|
|
+ }
|
|
+
|
|
+ fifo->total_used += n;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_fifo_dequeue (PurpleFifo *fifo, gpointer dest, gint n)
|
|
+{
|
|
+ gint out_pos;
|
|
+
|
|
+ if (n > fifo->total_used)
|
|
+ return FALSE;
|
|
+
|
|
+ for (out_pos = 0; out_pos < n; )
|
|
+ {
|
|
+ gint n_copied = MIN (BUFFER_SIZE - fifo->dequeue_pos, n - out_pos);
|
|
+
|
|
+ if (dest)
|
|
+ memcpy (((guint8 *) dest) + out_pos,
|
|
+ ((guint8 *) fifo->dequeue_buffer->data) + fifo->dequeue_pos,
|
|
+ n_copied);
|
|
+
|
|
+ fifo->dequeue_pos += n_copied;
|
|
+ out_pos += n_copied;
|
|
+
|
|
+ if (fifo->dequeue_pos == BUFFER_SIZE)
|
|
+ dequeue_next_buffer (fifo);
|
|
+ }
|
|
+
|
|
+ fifo->total_used -= n;
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_fifo_dequeue_line (PurpleFifo *fifo, gchar **dest)
|
|
+{
|
|
+ gchar *line;
|
|
+ gboolean have_cr = FALSE;
|
|
+ gboolean have_crlf = FALSE;
|
|
+ GList *buffer_l;
|
|
+ gint pos;
|
|
+ gint n;
|
|
+
|
|
+ buffer_l = fifo->dequeue_buffer;
|
|
+ n = 0;
|
|
+
|
|
+ for (pos = fifo->dequeue_pos; n < fifo->total_used && !have_crlf; pos++, n++)
|
|
+ {
|
|
+ guint8 *p = ((guint8 *) buffer_l->data) + fifo->dequeue_pos;
|
|
+ guint8 c = *p;
|
|
+
|
|
+ if (c == '\r')
|
|
+ {
|
|
+ have_cr = TRUE;
|
|
+ have_crlf = FALSE;
|
|
+ }
|
|
+ else if (c == '\n' && have_cr)
|
|
+ {
|
|
+ have_crlf = TRUE;
|
|
+ }
|
|
+
|
|
+ if (pos == BUFFER_SIZE)
|
|
+ {
|
|
+ buffer_l = g_list_next (buffer_l);
|
|
+ pos = 0;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (!have_crlf)
|
|
+ return FALSE;
|
|
+
|
|
+ line = g_malloc (n + 1);
|
|
+ if (!purple_fifo_dequeue (fifo, line, n))
|
|
+ {
|
|
+ /* Paranoid check; can't happen in practice */
|
|
+ g_free (line);
|
|
+ return FALSE;
|
|
+ }
|
|
+
|
|
+ *(line + n) = '\0';
|
|
+ *dest = line;
|
|
+
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_fifo_peek (PurpleFifo *fifo, gpointer dest, gint n)
|
|
+{
|
|
+ GList *dequeue_buffer;
|
|
+ gint dequeue_pos;
|
|
+ gint out_pos;
|
|
+
|
|
+ if (n > fifo->total_used)
|
|
+ return FALSE;
|
|
+
|
|
+ dequeue_buffer = fifo->dequeue_buffer;
|
|
+ dequeue_pos = fifo->dequeue_pos;
|
|
+
|
|
+ for (out_pos = 0; out_pos < n; )
|
|
+ {
|
|
+ gint n_copied = MIN (BUFFER_SIZE - dequeue_pos, n - out_pos);
|
|
+
|
|
+ if (dest)
|
|
+ memcpy (((guint8 *) dest) + out_pos,
|
|
+ ((guint8 *) dequeue_buffer->data) + dequeue_pos,
|
|
+ n_copied);
|
|
+
|
|
+ dequeue_pos += n_copied;
|
|
+ out_pos += n_copied;
|
|
+
|
|
+ if (dequeue_pos == BUFFER_SIZE)
|
|
+ {
|
|
+ dequeue_buffer = g_list_next (dequeue_buffer);
|
|
+ dequeue_pos = 0;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ return TRUE;
|
|
+}
|
|
--- /dev/null
|
|
+++ b/libpurple/purple-fifo.h
|
|
@@ -0,0 +1,54 @@
|
|
+/**
|
|
+ * @file purple-fifo.h A FIFO byte queue.
|
|
+ * @ingroup core
|
|
+ *
|
|
+ * Pidgin is the legal property of its developers, whose names are too numerous
|
|
+ * to list here. Please refer to the COPYRIGHT file distributed with this
|
|
+ * source distribution.
|
|
+ *
|
|
+ * This program is free software; you can redistribute it and/or modify
|
|
+ * it under the terms of the GNU General Public License as published by
|
|
+ * the Free Software Foundation; either version 2 of the License, or
|
|
+ * (at your option) any later version.
|
|
+ *
|
|
+ * This program is distributed in the hope that it will be useful,
|
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
+ * GNU General Public License for more details.
|
|
+ *
|
|
+ * You should have received a copy of the GNU General Public License
|
|
+ * along with this program; if not, write to the Free Software
|
|
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
+ */
|
|
+
|
|
+#ifndef _purple_fifo_H_
|
|
+#define _purple_fifo_H_
|
|
+
|
|
+#include <glib.h>
|
|
+
|
|
+G_BEGIN_DECLS
|
|
+
|
|
+typedef struct
|
|
+{
|
|
+ GList *buffer_list;
|
|
+ GList *enqueue_buffer;
|
|
+ gint enqueue_pos;
|
|
+ GList *dequeue_buffer;
|
|
+ gint dequeue_pos;
|
|
+}
|
|
+PurpleFifo;
|
|
+
|
|
+PurpleFifo *purple_fifo_new (void);
|
|
+void purple_fifo_destroy (PurpleFifo *fifo);
|
|
+
|
|
+gint purple_fifo_get_used (PurpleFifo *fifo);
|
|
+
|
|
+void purple_fifo_enqueue (PurpleFifo *fifo, gconstpointer libpurple, gint n);
|
|
+gboolean purple_fifo_dequeue (PurpleFifo *fifo, gpointer dest, gint n);
|
|
+gboolean purple_fifo_dequeue_line (PurpleFifo *fifo, gchar **dest);
|
|
+
|
|
+gboolean purple_fifo_peek (PurpleFifo *fifo, gpointer dest, gint n);
|
|
+
|
|
+G_END_DECLS
|
|
+
|
|
+#endif /* _purple_fifo_H_ */
|
|
--- /dev/null
|
|
+++ b/libpurple/purple-io.c
|
|
@@ -0,0 +1,391 @@
|
|
+/*
|
|
+ * @file purple-io.c Buffering I/O layer.
|
|
+ * @ingroup core
|
|
+ *
|
|
+ * Pidgin is the legal property of its developers, whose names are too numerous
|
|
+ * to list here. Please refer to the COPYRIGHT file distributed with this
|
|
+ * source distribution.
|
|
+ *
|
|
+ * This program is free software; you can redistribute it and/or modify
|
|
+ * it under the terms of the GNU General Public License as published by
|
|
+ * the Free Software Foundation; either version 2 of the License, or
|
|
+ * (at your option) any later version.
|
|
+ *
|
|
+ * This program is distributed in the hope that it will be useful,
|
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
+ * GNU General Public License for more details.
|
|
+ *
|
|
+ * You should have received a copy of the GNU General Public License
|
|
+ * along with this program; if not, write to the Free Software
|
|
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
+ */
|
|
+
|
|
+#include "internal.h"
|
|
+#include "eventloop.h"
|
|
+#include "purple-io.h"
|
|
+
|
|
+#define OPERATION_MAX_BYTES 4096
|
|
+#define WRITE_LOW_WATER 512
|
|
+
|
|
+static void update_watches (PurpleIO *io);
|
|
+
|
|
+static gboolean
|
|
+can_read (PurpleIO *io)
|
|
+{
|
|
+ fd_set rfds;
|
|
+ struct timeval tv;
|
|
+ int retval;
|
|
+
|
|
+ FD_ZERO(&rfds);
|
|
+ FD_SET(io->fd, &rfds);
|
|
+
|
|
+ tv.tv_sec = 0;
|
|
+ tv.tv_usec = 1;
|
|
+
|
|
+ retval = select (io->fd + 1, &rfds, NULL, NULL, &tv);
|
|
+
|
|
+ return retval ? TRUE : FALSE;
|
|
+}
|
|
+
|
|
+static gint
|
|
+try_read (PurpleIO *io, gint max_read, gint *read_errno)
|
|
+{
|
|
+ guint8 buf [OPERATION_MAX_BYTES];
|
|
+ glong flags;
|
|
+ gint n_read;
|
|
+
|
|
+ /* Do a non-blocking read */
|
|
+ flags = fcntl (io->fd, F_GETFL);
|
|
+ fcntl (io->fd, F_SETFL, flags | O_NONBLOCK);
|
|
+
|
|
+ if (io->gsc) {
|
|
+ n_read = purple_ssl_read_nonblock (io->gsc, buf, MIN (max_read, OPERATION_MAX_BYTES));
|
|
+
|
|
+ if (read_errno) {
|
|
+ if (n_read == PURPLE_SSL_IO_AGAIN)
|
|
+ *read_errno = EAGAIN;
|
|
+ else if (n_read == PURPLE_SSL_IO_INTERRUPTED)
|
|
+ *read_errno = EINTR;
|
|
+ else
|
|
+ *read_errno = 0;
|
|
+ }
|
|
+
|
|
+ if (n_read < 0)
|
|
+ n_read = 0;
|
|
+ } else {
|
|
+ n_read = read (io->fd, buf, MIN (max_read, OPERATION_MAX_BYTES));
|
|
+ if (read_errno)
|
|
+ *read_errno = errno;
|
|
+ }
|
|
+
|
|
+ fcntl (io->fd, F_SETFL, flags);
|
|
+
|
|
+ if (n_read > 0) {
|
|
+ /* Got some data */
|
|
+ purple_fifo_enqueue (io->read_fifo, buf, n_read);
|
|
+ }
|
|
+
|
|
+ return n_read;
|
|
+}
|
|
+
|
|
+static void
|
|
+read_nonblock (PurpleIO *io, gint max_read)
|
|
+{
|
|
+ gint read_errno;
|
|
+
|
|
+ if (io->got_eof)
|
|
+ return;
|
|
+
|
|
+ if (max_read <= purple_fifo_get_used (io->read_fifo))
|
|
+ return;
|
|
+
|
|
+ max_read -= purple_fifo_get_used (io->read_fifo);
|
|
+
|
|
+ if (can_read (io) && try_read (io, max_read, &read_errno) < 1 && read_errno != EINTR)
|
|
+ {
|
|
+ /* Disconnected */
|
|
+ io->got_eof = TRUE;
|
|
+ update_watches (io);
|
|
+
|
|
+ if (io->eof_func)
|
|
+ io->eof_func (io, io->eof_func_data);
|
|
+ }
|
|
+}
|
|
+
|
|
+static void
|
|
+read_cb (gpointer data, gint source, PurpleInputCondition condition)
|
|
+{
|
|
+ PurpleIO *io = data;
|
|
+ gint n_read;
|
|
+ gint read_errno;
|
|
+
|
|
+ n_read = try_read (io, OPERATION_MAX_BYTES, &read_errno);
|
|
+
|
|
+ /* If interrupted by signal, just retry later */
|
|
+ if (n_read < 0 && read_errno == EINTR)
|
|
+ return;
|
|
+
|
|
+ /* If we got 0 bytes or an error that's not EINTR, the connection was closed */
|
|
+ if (n_read < 1) {
|
|
+ io->got_eof = TRUE;
|
|
+ update_watches (io);
|
|
+
|
|
+ if (io->eof_func)
|
|
+ io->eof_func (io, io->eof_func_data);
|
|
+
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ if (io->read_func)
|
|
+ io->read_func (io, io->read_func_data);
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+static void
|
|
+write_cb (gpointer data, gint source, PurpleInputCondition condition)
|
|
+{
|
|
+ PurpleIO *io = data;
|
|
+ guint8 buf [OPERATION_MAX_BYTES];
|
|
+ gint n_written;
|
|
+ glong flags;
|
|
+
|
|
+ n_written = purple_fifo_get_used (io->write_fifo);
|
|
+ n_written = MIN (OPERATION_MAX_BYTES, n_written);
|
|
+
|
|
+ /* Paranoid: Shouldn't happen */
|
|
+ if (n_written < 1)
|
|
+ return;
|
|
+
|
|
+ purple_fifo_peek (io->write_fifo, buf, n_written);
|
|
+
|
|
+ flags = fcntl (io->fd, F_GETFL);
|
|
+ fcntl (io->fd, F_SETFL, flags | O_NONBLOCK);
|
|
+
|
|
+ if (io->gsc) {
|
|
+ n_written = purple_ssl_write_nonblock (io->gsc, buf, n_written);
|
|
+ } else {
|
|
+ n_written = write (io->fd, buf, n_written);
|
|
+ }
|
|
+
|
|
+ fcntl (io->fd, F_SETFL, flags);
|
|
+
|
|
+ if (n_written < 0)
|
|
+ io->got_eof = TRUE;
|
|
+
|
|
+ if (n_written < 1)
|
|
+ return;
|
|
+
|
|
+ /* Drop written data from buffer */
|
|
+ purple_fifo_dequeue (io->write_fifo, NULL, n_written);
|
|
+
|
|
+ if (io->flushed_func &&
|
|
+ purple_fifo_get_used (io->write_fifo) == 0)
|
|
+ io->flushed_func (io, io->flushed_func_data);
|
|
+
|
|
+ if (io->write_func &&
|
|
+ purple_fifo_get_used (io->write_fifo) < WRITE_LOW_WATER)
|
|
+ io->write_func (io, io->write_func_data);
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+static void
|
|
+update_watches (PurpleIO *io)
|
|
+{
|
|
+ gboolean want_read_watch = FALSE;
|
|
+ gboolean want_write_watch = FALSE;
|
|
+
|
|
+ if (!io->got_eof) {
|
|
+ if (io->read_func)
|
|
+ want_read_watch = TRUE;
|
|
+
|
|
+ if (purple_fifo_get_used (io->write_fifo))
|
|
+ want_write_watch = TRUE;
|
|
+ }
|
|
+
|
|
+ if (want_read_watch && !io->read_watch_id) {
|
|
+ io->read_watch_id = purple_input_add (io->fd, PURPLE_INPUT_READ, read_cb, io);
|
|
+ } else if (!want_read_watch && io->read_watch_id) {
|
|
+ purple_input_remove (io->read_watch_id);
|
|
+ io->read_watch_id = 0;
|
|
+ }
|
|
+
|
|
+ if (want_write_watch && !io->write_watch_id) {
|
|
+ io->write_watch_id = purple_input_add (io->fd, PURPLE_INPUT_WRITE, write_cb, io);
|
|
+ } else if (!want_write_watch && io->write_watch_id) {
|
|
+ purple_input_remove (io->write_watch_id);
|
|
+ io->write_watch_id = 0;
|
|
+ }
|
|
+}
|
|
+
|
|
+PurpleIO *
|
|
+purple_io_new (gint fd)
|
|
+{
|
|
+ PurpleIO *io;
|
|
+
|
|
+ io = g_new0 (PurpleIO, 1);
|
|
+
|
|
+ io->fd = fd;
|
|
+ io->read_fifo = purple_fifo_new ();
|
|
+ io->write_fifo = purple_fifo_new ();
|
|
+
|
|
+ return io;
|
|
+}
|
|
+
|
|
+PurpleIO *
|
|
+purple_io_new_ssl (PurpleSslConnection *gsc)
|
|
+{
|
|
+ PurpleIO *io;
|
|
+
|
|
+ io = g_new0 (PurpleIO, 1);
|
|
+
|
|
+ io->gsc = gsc;
|
|
+ io->fd = gsc->fd;
|
|
+ io->read_fifo = purple_fifo_new ();
|
|
+ io->write_fifo = purple_fifo_new ();
|
|
+
|
|
+ return io;
|
|
+}
|
|
+
|
|
+void
|
|
+purple_io_destroy (PurpleIO *io)
|
|
+{
|
|
+#if 0
|
|
+ if (purple_fifo_get_used (io->write_fifo) > 0)
|
|
+ g_print ("Throwing away outbound data!\n");
|
|
+#endif
|
|
+
|
|
+ if (io->read_watch_id)
|
|
+ purple_input_remove (io->read_watch_id);
|
|
+ if (io->write_watch_id)
|
|
+ purple_input_remove (io->write_watch_id);
|
|
+
|
|
+ purple_fifo_destroy (io->read_fifo);
|
|
+ purple_fifo_destroy (io->write_fifo);
|
|
+
|
|
+ if (io->gsc)
|
|
+ purple_ssl_close (io->gsc);
|
|
+ else
|
|
+ close (io->fd);
|
|
+
|
|
+ g_free (io);
|
|
+}
|
|
+
|
|
+gint
|
|
+purple_io_get_fd (PurpleIO *io)
|
|
+{
|
|
+ return io->fd;
|
|
+}
|
|
+
|
|
+PurpleSslConnection *
|
|
+purple_io_get_ssl_connection (PurpleIO *io)
|
|
+{
|
|
+ return io->gsc;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_io_is_connected (PurpleIO *io)
|
|
+{
|
|
+ return !io->got_eof;
|
|
+}
|
|
+
|
|
+void
|
|
+purple_io_set_read_func (PurpleIO *io, PurpleIOFunc func, gpointer data)
|
|
+{
|
|
+ io->read_func = func;
|
|
+ io->read_func_data = data;
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+void
|
|
+purple_io_set_write_func (PurpleIO *io, PurpleIOFunc func, gpointer data)
|
|
+{
|
|
+ io->write_func = func;
|
|
+ io->write_func_data = data;
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+void
|
|
+purple_io_set_eof_func (PurpleIO *io, PurpleIOFunc func, gpointer data)
|
|
+{
|
|
+ io->eof_func = func;
|
|
+ io->eof_func_data = data;
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+void
|
|
+purple_io_set_flushed_func (PurpleIO *io, PurpleIOFunc func, gpointer data)
|
|
+{
|
|
+ io->flushed_func = func;
|
|
+ io->flushed_func_data = data;
|
|
+
|
|
+ update_watches (io);
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_io_write_bytes (PurpleIO *io, gconstpointer libpurple, gint n_bytes)
|
|
+{
|
|
+ if (io->got_eof)
|
|
+ return FALSE;
|
|
+
|
|
+ purple_fifo_enqueue (io->write_fifo, libpurple, n_bytes);
|
|
+
|
|
+ update_watches (io);
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_io_write_line (PurpleIO *io, const gchar *line)
|
|
+{
|
|
+ if (io->got_eof)
|
|
+ return FALSE;
|
|
+
|
|
+ purple_fifo_enqueue (io->write_fifo, line, strlen (line));
|
|
+ purple_fifo_enqueue (io->write_fifo, "\r\n", 2);
|
|
+
|
|
+ update_watches (io);
|
|
+ return TRUE;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_io_read_bytes (PurpleIO *io, gpointer dest, gint n_bytes)
|
|
+{
|
|
+ gboolean result;
|
|
+
|
|
+ read_nonblock (io, n_bytes);
|
|
+
|
|
+ result = purple_fifo_dequeue (io->read_fifo, dest, n_bytes);
|
|
+
|
|
+ update_watches (io);
|
|
+ return result;
|
|
+}
|
|
+
|
|
+gboolean
|
|
+purple_io_read_line (PurpleIO *io, gchar **line)
|
|
+{
|
|
+ gboolean result;
|
|
+
|
|
+ read_nonblock (io, 65536);
|
|
+
|
|
+ result = purple_fifo_dequeue_line (io->read_fifo, line);
|
|
+
|
|
+ update_watches (io);
|
|
+ return result;
|
|
+}
|
|
+
|
|
+gint
|
|
+purple_io_get_pending_read (PurpleIO *io)
|
|
+{
|
|
+ return purple_fifo_get_used (io->read_fifo);
|
|
+}
|
|
+
|
|
+gint
|
|
+purple_io_get_pending_write (PurpleIO *io)
|
|
+{
|
|
+ return purple_fifo_get_used (io->write_fifo);
|
|
+}
|
|
--- /dev/null
|
|
+++ b/libpurple/purple-io.h
|
|
@@ -0,0 +1,87 @@
|
|
+/**
|
|
+ * @file purple-io.h Buffering I/O layer.
|
|
+ * @ingroup core
|
|
+ *
|
|
+ * Pidgin is the legal property of its developers, whose names are too numerous
|
|
+ * to list here. Please refer to the COPYRIGHT file distributed with this
|
|
+ * source distribution.
|
|
+ *
|
|
+ * This program is free software; you can redistribute it and/or modify
|
|
+ * it under the terms of the GNU General Public License as published by
|
|
+ * the Free Software Foundation; either version 2 of the License, or
|
|
+ * (at your option) any later version.
|
|
+ *
|
|
+ * This program is distributed in the hope that it will be useful,
|
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
+ * GNU General Public License for more details.
|
|
+ *
|
|
+ * You should have received a copy of the GNU General Public License
|
|
+ * along with this program; if not, write to the Free Software
|
|
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
+ */
|
|
+
|
|
+#ifndef _PURPLE_IO_H_
|
|
+#define _PURPLE_IO_H_
|
|
+
|
|
+#include "purple-fifo.h"
|
|
+#include "sslconn.h"
|
|
+
|
|
+G_BEGIN_DECLS
|
|
+
|
|
+typedef struct PurpleIO PurpleIO;
|
|
+
|
|
+typedef void (*PurpleIOFunc) (PurpleIO *io, gpointer data);
|
|
+
|
|
+struct PurpleIO
|
|
+{
|
|
+ gint fd;
|
|
+ PurpleSslConnection *gsc;
|
|
+
|
|
+ guint got_eof : 1;
|
|
+
|
|
+ PurpleFifo *read_fifo;
|
|
+ PurpleFifo *write_fifo;
|
|
+
|
|
+ guint read_watch_id;
|
|
+ guint write_watch_id;
|
|
+
|
|
+ /* Callback invoked when input buffer grows */
|
|
+ PurpleIOFunc read_func;
|
|
+ gpointer read_func_data;
|
|
+
|
|
+ /* Callback invoked when output buffer shrinks (sufficiently) */
|
|
+ PurpleIOFunc write_func;
|
|
+ gpointer write_func_data;
|
|
+
|
|
+ /* Callback invoked when connection drops */
|
|
+ PurpleIOFunc eof_func;
|
|
+ gpointer eof_func_data;
|
|
+
|
|
+ /* Callback invoked when output buffer empty */
|
|
+ PurpleIOFunc flushed_func;
|
|
+ gpointer flushed_func_data;
|
|
+};
|
|
+
|
|
+PurpleIO *purple_io_new (gint fd);
|
|
+PurpleIO *purple_io_new_ssl (PurpleSslConnection *gsc);
|
|
+void purple_io_destroy (PurpleIO *io);
|
|
+
|
|
+gint purple_io_get_fd (PurpleIO *io);
|
|
+PurpleSslConnection *purple_io_get_ssl_connection (PurpleIO *io);
|
|
+gboolean purple_io_is_connected (PurpleIO *io);
|
|
+
|
|
+void purple_io_set_read_func (PurpleIO *io, PurpleIOFunc func, gpointer data);
|
|
+void purple_io_set_write_func (PurpleIO *io, PurpleIOFunc func, gpointer data);
|
|
+void purple_io_set_eof_func (PurpleIO *io, PurpleIOFunc func, gpointer data);
|
|
+void purple_io_set_flushed_func (PurpleIO *io, PurpleIOFunc func, gpointer data);
|
|
+
|
|
+gboolean purple_io_write_bytes (PurpleIO *io, gconstpointer libpurple, gint n_bytes);
|
|
+gboolean purple_io_write_line (PurpleIO *io, const gchar *line);
|
|
+
|
|
+gboolean purple_io_read_bytes (PurpleIO *io, gpointer dest, gint n_bytes);
|
|
+gboolean purple_io_read_line (PurpleIO *io, gchar **line);
|
|
+
|
|
+G_END_DECLS
|
|
+
|
|
+#endif /* _PURPLE_IO_H_ */
|
|
--- a/libpurple/sslconn.c
|
|
+++ b/libpurple/sslconn.c
|
|
@@ -273,6 +273,27 @@ purple_ssl_read(PurpleSslConnection *gsc
|
|
}
|
|
|
|
size_t
|
|
+purple_ssl_read_nonblock(PurpleSslConnection *gsc, void *data, size_t len)
|
|
+{
|
|
+ PurpleSslOps *ops;
|
|
+
|
|
+ g_return_val_if_fail(gsc != NULL, 0);
|
|
+ g_return_val_if_fail(data != NULL, 0);
|
|
+ g_return_val_if_fail(len > 0, 0);
|
|
+
|
|
+ ops = purple_ssl_get_ops();
|
|
+
|
|
+ if (ops != NULL) {
|
|
+ if (ops->read_nonblock != NULL)
|
|
+ return (ops->read_nonblock)(gsc, data, len);
|
|
+ else if (ops->read != NULL)
|
|
+ return (ops->read)(gsc, data, len);
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+size_t
|
|
purple_ssl_write(PurpleSslConnection *gsc, const void *data, size_t len)
|
|
{
|
|
PurpleSslOps *ops;
|
|
@@ -296,6 +317,27 @@ purple_ssl_get_peer_certificates(PurpleS
|
|
return (ops->get_peer_certificates)(gsc);
|
|
}
|
|
|
|
+size_t
|
|
+purple_ssl_write_nonblock(PurpleSslConnection *gsc, const void *data, size_t len)
|
|
+{
|
|
+ PurpleSslOps *ops;
|
|
+
|
|
+ g_return_val_if_fail(gsc != NULL, 0);
|
|
+ g_return_val_if_fail(data != NULL, 0);
|
|
+ g_return_val_if_fail(len > 0, 0);
|
|
+
|
|
+ ops = purple_ssl_get_ops();
|
|
+
|
|
+ if (ops != NULL) {
|
|
+ if (ops->write_nonblock != NULL)
|
|
+ return (ops->write_nonblock)(gsc, data, len);
|
|
+ else if (ops->write != NULL)
|
|
+ return (ops->write)(gsc, data, len);
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
void
|
|
purple_ssl_set_ops(PurpleSslOps *ops)
|
|
{
|
|
--- a/libpurple/sslconn.h
|
|
+++ b/libpurple/sslconn.h
|
|
@@ -34,6 +34,12 @@ typedef enum
|
|
PURPLE_SSL_CERTIFICATE_INVALID = 3
|
|
} PurpleSslErrorType;
|
|
|
|
+typedef enum
|
|
+{
|
|
+ PURPLE_SSL_IO_INTERRUPTED = -1,
|
|
+ PURPLE_SSL_IO_AGAIN = -2
|
|
+} PurpleSslIOErrorType;
|
|
+
|
|
#include "certificate.h"
|
|
#include "proxy.h"
|
|
|
|
@@ -115,6 +121,7 @@ typedef struct
|
|
* @see purple_ssl_read
|
|
*/
|
|
size_t (*read)(PurpleSslConnection *gsc, void *data, size_t len);
|
|
+ size_t (*read_nonblock)(PurpleSslConnection *gsc, void *data, size_t len);
|
|
/** Writes data to a connection (like POSIX send())
|
|
* @param gsc Connection context
|
|
* @param data Data buffer to send data from
|
|
@@ -124,6 +131,7 @@ typedef struct
|
|
* @see purple_ssl_write
|
|
*/
|
|
size_t (*write)(PurpleSslConnection *gsc, const void *data, size_t len);
|
|
+ size_t (*write_nonblock)(PurpleSslConnection *gsc, const void *data, size_t len);
|
|
/** Obtains the certificate chain provided by the peer
|
|
*
|
|
* @param gsc Connection context
|
|
@@ -281,6 +289,17 @@ void purple_ssl_close(PurpleSslConnectio
|
|
size_t purple_ssl_read(PurpleSslConnection *gsc, void *buffer, size_t len);
|
|
|
|
/**
|
|
+ * Reads data from an SSL connection without blocking.
|
|
+ *
|
|
+ * @param gsc The SSL connection handle.
|
|
+ * @param buffer The destination buffer.
|
|
+ * @param len The maximum number of bytes to read.
|
|
+ *
|
|
+ * @return The number of bytes read.
|
|
+ */
|
|
+size_t purple_ssl_read_nonblock(PurpleSslConnection *gsc, void *buffer, size_t len);
|
|
+
|
|
+/**
|
|
* Writes data to an SSL connection.
|
|
*
|
|
* @param gsc The SSL connection handle.
|
|
@@ -303,6 +322,17 @@ size_t purple_ssl_write(PurpleSslConnect
|
|
*/
|
|
GList * purple_ssl_get_peer_certificates(PurpleSslConnection *gsc);
|
|
|
|
+/**
|
|
+ * Writes data to an SSL connection without blocking.
|
|
+ *
|
|
+ * @param gsc The SSL connection handle.
|
|
+ * @param buffer The buffer to write.
|
|
+ * @param len The length of the data to write.
|
|
+ *
|
|
+ * @return The number of bytes written.
|
|
+ */
|
|
+size_t purple_ssl_write_nonblock(PurpleSslConnection *gsc, const void *buffer, size_t len);
|
|
+
|
|
/*@}*/
|
|
|
|
/**************************************************************************/
|