/* -*- 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);
}