/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* * GIO - GLib Input, Output and Streaming Library * * Copyright 2010 Red Hat, Inc * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General * Public License along with this library; if not, see * . * * In addition, when the library is used with OpenSSL, a special * exception applies. Refer to the LICENSE_EXCEPTION file for details. */ #include "config.h" #include "gtlsoutputstream.h" #include struct _GTlsOutputStream { GOutputStream parent_instance; GWeakRef weak_conn; }; static void g_tls_output_stream_pollable_iface_init (GPollableOutputStreamInterface *iface); G_DEFINE_FINAL_TYPE_WITH_CODE (GTlsOutputStream, g_tls_output_stream, G_TYPE_OUTPUT_STREAM, G_IMPLEMENT_INTERFACE (G_TYPE_POLLABLE_OUTPUT_STREAM, g_tls_output_stream_pollable_iface_init) ) static void g_tls_output_stream_dispose (GObject *object) { GTlsOutputStream *stream = G_TLS_OUTPUT_STREAM (object); g_weak_ref_set (&stream->weak_conn, NULL); G_OBJECT_CLASS (g_tls_output_stream_parent_class)->dispose (object); } static void g_tls_output_stream_finalize (GObject *object) { GTlsOutputStream *stream = G_TLS_OUTPUT_STREAM (object); g_weak_ref_clear (&stream->weak_conn); G_OBJECT_CLASS (g_tls_output_stream_parent_class)->finalize (object); } static gssize g_tls_output_stream_write (GOutputStream *stream, const void *buffer, gsize count, GCancellable *cancellable, GError **error) { GTlsOutputStream *tls_stream = G_TLS_OUTPUT_STREAM (stream); GTlsConnectionBase *conn; gssize ret; conn = g_weak_ref_get (&tls_stream->weak_conn); if (!conn) { g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_CLOSED, _("Connection is closed")); return -1; } ret = g_tls_connection_base_write (conn, buffer, count, -1 /* blocking */, cancellable, error); g_object_unref (conn); return ret; } static gboolean g_tls_output_stream_pollable_is_writable (GPollableOutputStream *pollable) { GTlsOutputStream *tls_stream = G_TLS_OUTPUT_STREAM (pollable); GTlsConnectionBase *conn; gboolean ret; conn = g_weak_ref_get (&tls_stream->weak_conn); if (!conn) return FALSE; ret = g_tls_connection_base_check (conn, G_IO_OUT); g_object_unref (conn); return ret; } static GSource * g_tls_output_stream_pollable_create_source (GPollableOutputStream *pollable, GCancellable *cancellable) { GTlsOutputStream *tls_stream = G_TLS_OUTPUT_STREAM (pollable); GTlsConnectionBase *conn; GSource *ret; conn = g_weak_ref_get (&tls_stream->weak_conn); if (!conn) { ret = g_idle_source_new (); g_source_set_name (ret, "[glib-networking] g_tls_output_stream_pollable_create_source dummy source"); return ret; } ret = g_tls_connection_base_create_source (conn, G_IO_OUT, cancellable); g_object_unref (conn); return ret; } static gssize g_tls_output_stream_pollable_write_nonblocking (GPollableOutputStream *pollable, const void *buffer, gsize size, GError **error) { GTlsOutputStream *tls_stream = G_TLS_OUTPUT_STREAM (pollable); GTlsConnectionBase *conn; gssize ret; conn = g_weak_ref_get (&tls_stream->weak_conn); if (!conn) { g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_CLOSED, _("Connection is closed")); return -1; } ret = g_tls_connection_base_write (conn, buffer, size, 0 /* non-blocking */, NULL, error); g_object_unref (conn); return ret; } static gboolean g_tls_output_stream_close (GOutputStream *stream, GCancellable *cancellable, GError **error) { GTlsOutputStream *tls_stream = G_TLS_OUTPUT_STREAM (stream); GIOStream *conn; gboolean ret; conn = g_weak_ref_get (&tls_stream->weak_conn); if (!conn) return TRUE; ret = g_tls_connection_base_close_internal (conn, G_TLS_DIRECTION_WRITE, -1, /* blocking */ cancellable, error); g_object_unref (conn); return ret; } /* We do async close as synchronous-in-a-thread so we don't need to * implement G_IO_IN/G_IO_OUT flip-flopping just for this one case * (since handshakes are also done synchronously now). */ static void close_thread (GTask *task, gpointer object, gpointer task_data, GCancellable *cancellable) { GTlsOutputStream *tls_stream = object; GError *error = NULL; GIOStream *conn; conn = g_weak_ref_get (&tls_stream->weak_conn); if (conn && !g_tls_connection_base_close_internal (conn, G_TLS_DIRECTION_WRITE, -1, /* blocking */ cancellable, &error)) g_task_return_error (task, error); else g_task_return_boolean (task, TRUE); if (conn) g_object_unref (conn); } static void g_tls_output_stream_close_async (GOutputStream *stream, int io_priority, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GTask *task; task = g_task_new (stream, cancellable, callback, user_data); g_task_set_source_tag (task, g_tls_output_stream_close_async); g_task_set_name (task, "[glib-networking] g_tls_output_stream_close_async"); g_task_set_priority (task, io_priority); g_task_run_in_thread (task, close_thread); g_object_unref (task); } static gboolean g_tls_output_stream_close_finish (GOutputStream *stream, GAsyncResult *result, GError **error) { g_return_val_if_fail (g_task_is_valid (result, stream), FALSE); g_return_val_if_fail (g_task_get_source_tag (G_TASK (result)) == g_tls_output_stream_close_async, FALSE); return g_task_propagate_boolean (G_TASK (result), error); } static void g_tls_output_stream_class_init (GTlsOutputStreamClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GOutputStreamClass *output_stream_class = G_OUTPUT_STREAM_CLASS (klass); gobject_class->dispose = g_tls_output_stream_dispose; gobject_class->finalize = g_tls_output_stream_finalize; output_stream_class->write_fn = g_tls_output_stream_write; output_stream_class->close_fn = g_tls_output_stream_close; output_stream_class->close_async = g_tls_output_stream_close_async; output_stream_class->close_finish = g_tls_output_stream_close_finish; } static void g_tls_output_stream_pollable_iface_init (GPollableOutputStreamInterface *iface) { iface->is_writable = g_tls_output_stream_pollable_is_writable; iface->create_source = g_tls_output_stream_pollable_create_source; iface->write_nonblocking = g_tls_output_stream_pollable_write_nonblocking; } static void g_tls_output_stream_init (GTlsOutputStream *stream) { } GOutputStream * g_tls_output_stream_new (GTlsConnectionBase *conn) { GTlsOutputStream *tls_stream; tls_stream = g_object_new (G_TYPE_TLS_OUTPUT_STREAM, NULL); g_weak_ref_init (&tls_stream->weak_conn, conn); return G_OUTPUT_STREAM (tls_stream); }