mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2024-11-10 03:16:17 +01:00
23d80a04da
Any file handle created with FLAG_OVERLAPPED must have ReadFile()/WriteFile() called with an OVERLAPPED structure. Failing to do so will give unspecified results, invalid read/write or corruption. Without FLAG_OVERLAPPED, it is not possible to read and write concurrently, even with two seperate threads, created by 2 input and output gio streams. Also, only with FLAG_OVERLAPPED may an IO operation be asynchronous and thus be cancellable. We may want to call ReOpenFile() to make sure the FLAG is set, but this API is only available since Vista+. According to MSDN doc, adding the OVERLAPPED argument for IO operation on handles without FLAG_OVERLAPPED is allowed, and indeed the existing test still passes. v2: - update GetLastError() after _g_win32_overlap_wait_result () - split the unrelated ERROR_MORE_DATA handling https://bugzilla.gnome.org/show_bug.cgi?id=679288
383 lines
10 KiB
C
383 lines
10 KiB
C
/* 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 <alexl@redhat.com>
|
|
* Author: Tor Lillqvist <tml@iki.fi>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <windows.h>
|
|
|
|
#include <io.h>
|
|
|
|
#include <glib.h>
|
|
#include <glib/gstdio.h>
|
|
#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 <filename><gio/gwin32outputstream.h></filename> belongs
|
|
* to the Windows-specific GIO interfaces, thus you have to use the
|
|
* <filename>gio-windows-2.0.pc</filename> 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;
|
|
};
|
|
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* 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;
|
|
|
|
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;
|
|
}
|