/* GIO - GLib Input, Output and Streaming Library * * Copyright (C) 2006-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 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, write to the * Free Software Foundation, Inc., 59 Temple Place, Suite 330, * Boston, MA 02111-1307, USA. * * Author: Alexander Larsson * Author: Tor Lillqvist */ #include "config.h" #include #include #include #include #include "gioerror.h" #include "gwin32outputstream.h" #include "gcancellable.h" #include "gsimpleasyncresult.h" #include "gasynchelper.h" #include "glibintl.h" /** * SECTION:gwin32outputstream * @short_description: Streaming output operations for Windows file handles * @include: gio/gwin32outputstream.h * @see_also: #GOutputStream * * #GWin32OutputStream implements #GOutputStream for writing to a * Windows file handle. * * Note that <gio/gwin32outputstream.h> belongs * to the Windows-specific GIO interfaces, thus you have to use the * gio-windows-2.0.pc pkg-config file when using it. */ enum { PROP_0, PROP_HANDLE, PROP_CLOSE_HANDLE }; G_DEFINE_TYPE (GWin32OutputStream, g_win32_output_stream, G_TYPE_OUTPUT_STREAM); struct _GWin32OutputStreamPrivate { HANDLE handle; gboolean close_handle; gint fd; }; static void g_win32_output_stream_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void g_win32_output_stream_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static gssize g_win32_output_stream_write (GOutputStream *stream, const void *buffer, gsize count, GCancellable *cancellable, GError **error); static gboolean g_win32_output_stream_close (GOutputStream *stream, GCancellable *cancellable, GError **error); static void g_win32_output_stream_finalize (GObject *object) { G_OBJECT_CLASS (g_win32_output_stream_parent_class)->finalize (object); } static void g_win32_output_stream_class_init (GWin32OutputStreamClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GOutputStreamClass *stream_class = G_OUTPUT_STREAM_CLASS (klass); g_type_class_add_private (klass, sizeof (GWin32OutputStreamPrivate)); gobject_class->get_property = g_win32_output_stream_get_property; gobject_class->set_property = g_win32_output_stream_set_property; gobject_class->finalize = g_win32_output_stream_finalize; stream_class->write_fn = g_win32_output_stream_write; stream_class->close_fn = g_win32_output_stream_close; /** * GWin32OutputStream:handle: * * The file handle that the stream writes to. * * Since: 2.26 */ g_object_class_install_property (gobject_class, PROP_HANDLE, g_param_spec_pointer ("handle", P_("File handle"), P_("The file handle to write to"), G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB)); /** * GWin32OutputStream:close-handle: * * Whether to close the file handle when the stream is closed. * * Since: 2.26 */ g_object_class_install_property (gobject_class, PROP_CLOSE_HANDLE, g_param_spec_boolean ("close-handle", P_("Close file handle"), P_("Whether to close the file handle when the stream is closed"), TRUE, G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB)); } static void g_win32_output_stream_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GWin32OutputStream *win32_stream; win32_stream = G_WIN32_OUTPUT_STREAM (object); switch (prop_id) { case PROP_HANDLE: win32_stream->priv->handle = g_value_get_pointer (value); break; case PROP_CLOSE_HANDLE: win32_stream->priv->close_handle = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void g_win32_output_stream_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GWin32OutputStream *win32_stream; win32_stream = G_WIN32_OUTPUT_STREAM (object); switch (prop_id) { case PROP_HANDLE: g_value_set_pointer (value, win32_stream->priv->handle); break; case PROP_CLOSE_HANDLE: g_value_set_boolean (value, win32_stream->priv->close_handle); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); } } static void g_win32_output_stream_init (GWin32OutputStream *win32_stream) { win32_stream->priv = G_TYPE_INSTANCE_GET_PRIVATE (win32_stream, G_TYPE_WIN32_OUTPUT_STREAM, GWin32OutputStreamPrivate); win32_stream->priv->handle = NULL; win32_stream->priv->close_handle = TRUE; win32_stream->priv->fd = -1; } /** * g_win32_output_stream_new: * @handle: a Win32 file handle * @close_handle: %TRUE to close the handle when done * * Creates a new #GWin32OutputStream for the given @handle. * * If @close_handle, is %TRUE, the handle will be closed when the * output stream is destroyed. * * Returns: a new #GOutputStream * * Since: 2.26 **/ GOutputStream * g_win32_output_stream_new (void *handle, gboolean close_handle) { GWin32OutputStream *stream; g_return_val_if_fail (handle != NULL, NULL); stream = g_object_new (G_TYPE_WIN32_OUTPUT_STREAM, "handle", handle, "close-handle", close_handle, NULL); return G_OUTPUT_STREAM (stream); } /** * g_win32_output_stream_set_close_handle: * @stream: a #GWin32OutputStream * @close_handle: %TRUE to close the handle when done * * Sets whether the handle of @stream shall be closed when the stream * is closed. * * Since: 2.26 */ void g_win32_output_stream_set_close_handle (GWin32OutputStream *stream, gboolean close_handle) { g_return_if_fail (G_IS_WIN32_OUTPUT_STREAM (stream)); close_handle = close_handle != FALSE; if (stream->priv->close_handle != close_handle) { stream->priv->close_handle = close_handle; g_object_notify (G_OBJECT (stream), "close-handle"); } } /** * g_win32_output_stream_get_close_handle: * @stream: a #GWin32OutputStream * * Returns whether the handle of @stream will be closed when the * stream is closed. * * Return value: %TRUE if the handle is closed when done * * Since: 2.26 */ gboolean g_win32_output_stream_get_close_handle (GWin32OutputStream *stream) { g_return_val_if_fail (G_IS_WIN32_OUTPUT_STREAM (stream), FALSE); return stream->priv->close_handle; } /** * g_win32_output_stream_get_handle: * @stream: a #GWin32OutputStream * * Return the Windows handle that the stream writes to. * * Return value: The handle descriptor of @stream * * Since: 2.26 */ void * g_win32_output_stream_get_handle (GWin32OutputStream *stream) { g_return_val_if_fail (G_IS_WIN32_OUTPUT_STREAM (stream), NULL); return stream->priv->handle; } static gssize g_win32_output_stream_write (GOutputStream *stream, const void *buffer, gsize count, GCancellable *cancellable, GError **error) { GWin32OutputStream *win32_stream; BOOL res; DWORD nbytes, nwritten; OVERLAPPED overlap = { 0, }; gssize retval = -1; win32_stream = G_WIN32_OUTPUT_STREAM (stream); if (g_cancellable_set_error_if_cancelled (cancellable, error)) return -1; if (count > G_MAXINT) nbytes = G_MAXINT; else nbytes = count; overlap.hEvent = CreateEvent (NULL, FALSE, FALSE, NULL); g_return_val_if_fail (overlap.hEvent != NULL, -1); res = WriteFile (win32_stream->priv->handle, buffer, nbytes, &nwritten, &overlap); if (res) retval = nwritten; else { int errsv = GetLastError (); if (errsv == ERROR_IO_PENDING && _g_win32_overlap_wait_result (win32_stream->priv->handle, &overlap, &nwritten, cancellable)) { retval = nwritten; goto end; } if (g_cancellable_set_error_if_cancelled (cancellable, error)) goto end; errsv = GetLastError (); if (errsv == ERROR_HANDLE_EOF || errsv == ERROR_BROKEN_PIPE) { retval = 0; } else { gchar *emsg; emsg = g_win32_error_message (errsv); g_set_error (error, G_IO_ERROR, g_io_error_from_win32_error (errsv), _("Error writing to handle: %s"), emsg); g_free (emsg); } } end: CloseHandle (overlap.hEvent); return retval; } static gboolean g_win32_output_stream_close (GOutputStream *stream, GCancellable *cancellable, GError **error) { GWin32OutputStream *win32_stream; BOOL res; win32_stream = G_WIN32_OUTPUT_STREAM (stream); if (!win32_stream->priv->close_handle) return TRUE; if (win32_stream->priv->fd != -1) { int errsv = errno; if (close (win32_stream->priv->fd) < 0) { g_set_error_literal (error, G_IO_ERROR, g_io_error_from_errno (errsv), g_strerror (errsv)); return FALSE; } } else { res = CloseHandle (win32_stream->priv->handle); if (!res) { int errsv = GetLastError (); gchar *emsg = g_win32_error_message (errsv); g_set_error (error, G_IO_ERROR, g_io_error_from_win32_error (errsv), _("Error closing handle: %s"), emsg); g_free (emsg); return FALSE; } } return TRUE; } GOutputStream * g_win32_output_stream_new_from_fd (gint fd, gboolean close_fd) { GWin32OutputStream *win32_stream; win32_stream = G_WIN32_OUTPUT_STREAM (g_win32_output_stream_new ((HANDLE) _get_osfhandle (fd), close_fd)); win32_stream->priv->fd = fd; return (GOutputStream*)win32_stream; }