mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-19 00:20:09 +00:00
964affb1cc
The C standard does not specify whether the underlying type of an enum is signed or unsigned, and until C23 there was no way to control this explicitly. GCC appears to make enums unsigned unless there is a negative value among cases of the enum, in which case it becomes signed. MSCV appears to make enums signed by default. A bitfield of an enum type (which is not specificied in the C standard either) behaves as if it was an instance of a numeric type with a reduced value range. Specifically, a 'signed int val : 2;' bitfield will have the possible values of -2, -1, 0, and 1, with the usual wraparound behavior for the values that don't fit (although this too is implementation-defined). This causes the following issue, if we have: typedef enum { GTK_ZERO, GTK_ONE, GTK_TWO } GtkFoo; struct _GtkBar { GtkFoo foo : 2; }; and then assign bar.foo = GTK_TWO and read it back, it will have the expected value of 2 (aka GTK_TWO) on GCC, but a value of -2 (not matching any of the enum variants) on MSVC. There does not seem to be any way to influence signedness of an enum prior to C23, nor is there a 'unsigned GtkFoo foo : 2;' syntax. The only remaining options seems to be never using enums in bitfields, which is what this change implements. In practice, this fixes GdkPipeIOStream crashing with an assertion when trying to copy-paste in-app in MSVC builds on GTK. Signed-off-by: Sergey Bugaev <bugaevc@gmail.com>
478 lines
13 KiB
C
478 lines
13 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
*
|
|
* Copyright (C) 2017 Benjamin Otte <otte@gnome.org>
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gdkpipeiostreamprivate.h"
|
|
|
|
#include <string.h>
|
|
|
|
/* PIPE */
|
|
|
|
typedef enum {
|
|
GDK_IO_PIPE_EMPTY,
|
|
GDK_IO_PIPE_INPUT_BUFFER,
|
|
GDK_IO_PIPE_OUTPUT_BUFFER
|
|
} GdkIOPipeState;
|
|
|
|
typedef struct _GdkIOPipe GdkIOPipe;
|
|
|
|
struct _GdkIOPipe
|
|
{
|
|
int ref_count;
|
|
|
|
GMutex mutex;
|
|
GCond cond;
|
|
guchar *buffer;
|
|
gsize size;
|
|
guint state : 2; /* GdkIOPipeState */
|
|
guint input_closed : 1;
|
|
guint output_closed : 1;
|
|
};
|
|
|
|
static GdkIOPipe *
|
|
gdk_io_pipe_new (void)
|
|
{
|
|
GdkIOPipe *pipe;
|
|
|
|
pipe = g_new0 (GdkIOPipe, 1);
|
|
pipe->ref_count = 1;
|
|
|
|
g_mutex_init (&pipe->mutex);
|
|
g_cond_init (&pipe->cond);
|
|
|
|
return pipe;
|
|
}
|
|
|
|
static GdkIOPipe *
|
|
gdk_io_pipe_ref (GdkIOPipe *pipe)
|
|
{
|
|
g_atomic_int_inc (&pipe->ref_count);
|
|
|
|
return pipe;
|
|
}
|
|
|
|
static void
|
|
gdk_io_pipe_unref (GdkIOPipe *pipe)
|
|
{
|
|
if (!g_atomic_int_dec_and_test (&pipe->ref_count))
|
|
return;
|
|
|
|
g_cond_clear (&pipe->cond);
|
|
g_mutex_clear (&pipe->mutex);
|
|
|
|
g_free (pipe);
|
|
}
|
|
|
|
static void
|
|
gdk_io_pipe_lock (GdkIOPipe *pipe)
|
|
{
|
|
g_mutex_lock (&pipe->mutex);
|
|
}
|
|
|
|
static void
|
|
gdk_io_pipe_unlock (GdkIOPipe *pipe)
|
|
{
|
|
g_mutex_unlock (&pipe->mutex);
|
|
}
|
|
|
|
/* INPUT STREAM */
|
|
|
|
#define GDK_TYPE_PIPE_INPUT_STREAM (gdk_pipe_input_stream_get_type ())
|
|
#define GDK_PIPE_INPUT_STREAM(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GDK_TYPE_PIPE_INPUT_STREAM, GdkPipeInputStream))
|
|
#define GDK_IS_PIPE_INPUT_STREAM(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GDK_TYPE_PIPE_INPUT_STREAM))
|
|
#define GDK_PIPE_INPUT_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_PIPE_INPUT_STREAM, GdkPipeInputStreamClass))
|
|
#define GDK_IS_PIPE_INPUT_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_PIPE_INPUT_STREAM))
|
|
#define GDK_PIPE_INPUT_STREAM_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_PIPE_INPUT_STREAM, GdkPipeInputStreamClass))
|
|
|
|
typedef struct _GdkPipeInputStream GdkPipeInputStream;
|
|
typedef struct _GdkPipeInputStreamClass GdkPipeInputStreamClass;
|
|
|
|
struct _GdkPipeInputStream
|
|
{
|
|
GInputStream parent;
|
|
|
|
GdkIOPipe *pipe;
|
|
};
|
|
|
|
struct _GdkPipeInputStreamClass
|
|
{
|
|
GInputStreamClass parent_class;
|
|
};
|
|
|
|
GType gdk_pipe_input_stream_get_type (void) G_GNUC_CONST;
|
|
|
|
G_DEFINE_TYPE (GdkPipeInputStream, gdk_pipe_input_stream, G_TYPE_INPUT_STREAM)
|
|
|
|
static void
|
|
gdk_pipe_input_stream_finalize (GObject *object)
|
|
{
|
|
GdkPipeInputStream *pipe = GDK_PIPE_INPUT_STREAM (object);
|
|
|
|
g_clear_pointer (&pipe->pipe, gdk_io_pipe_unref);
|
|
|
|
G_OBJECT_CLASS (gdk_pipe_input_stream_parent_class)->finalize (object);
|
|
}
|
|
|
|
static gssize
|
|
gdk_pipe_input_stream_read (GInputStream *stream,
|
|
void *buffer,
|
|
gsize count,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
GdkPipeInputStream *pipe_stream = GDK_PIPE_INPUT_STREAM (stream);
|
|
GdkIOPipe *pipe = pipe_stream->pipe;
|
|
gsize amount;
|
|
|
|
gdk_io_pipe_lock (pipe);
|
|
|
|
switch (pipe->state)
|
|
{
|
|
case GDK_IO_PIPE_EMPTY:
|
|
if (pipe->output_closed)
|
|
{
|
|
amount = 0;
|
|
break;
|
|
}
|
|
pipe->buffer = buffer;
|
|
pipe->size = count;
|
|
pipe->state = GDK_IO_PIPE_INPUT_BUFFER;
|
|
do
|
|
g_cond_wait (&pipe->cond, &pipe->mutex);
|
|
while (pipe->size == count &&
|
|
pipe->state == GDK_IO_PIPE_INPUT_BUFFER &&
|
|
!pipe->output_closed);
|
|
if (pipe->state == GDK_IO_PIPE_INPUT_BUFFER)
|
|
{
|
|
amount = count - pipe->size;
|
|
pipe->state = GDK_IO_PIPE_EMPTY;
|
|
pipe->size = 0;
|
|
}
|
|
else
|
|
{
|
|
amount = count;
|
|
}
|
|
break;
|
|
|
|
case GDK_IO_PIPE_OUTPUT_BUFFER:
|
|
amount = MIN (count, pipe->size);
|
|
|
|
memcpy (buffer, pipe->buffer, amount);
|
|
pipe->size -= amount;
|
|
|
|
if (pipe->size == 0)
|
|
pipe->state = GDK_IO_PIPE_EMPTY;
|
|
else
|
|
pipe->buffer += amount;
|
|
break;
|
|
|
|
case GDK_IO_PIPE_INPUT_BUFFER:
|
|
default:
|
|
g_assert_not_reached ();
|
|
amount = 0;
|
|
break;
|
|
}
|
|
|
|
g_cond_broadcast (&pipe->cond);
|
|
gdk_io_pipe_unlock (pipe);
|
|
|
|
return amount;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pipe_input_stream_close (GInputStream *stream,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
GdkPipeInputStream *pipe_stream = GDK_PIPE_INPUT_STREAM (stream);
|
|
GdkIOPipe *pipe = pipe_stream->pipe;
|
|
|
|
gdk_io_pipe_lock (pipe);
|
|
|
|
pipe->input_closed = TRUE;
|
|
g_cond_broadcast (&pipe->cond);
|
|
|
|
gdk_io_pipe_unlock (pipe);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_input_stream_class_init (GdkPipeInputStreamClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
GInputStreamClass *input_stream_class = G_INPUT_STREAM_CLASS (class);
|
|
|
|
object_class->finalize = gdk_pipe_input_stream_finalize;
|
|
|
|
input_stream_class->read_fn = gdk_pipe_input_stream_read;
|
|
input_stream_class->close_fn = gdk_pipe_input_stream_close;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_input_stream_init (GdkPipeInputStream *pipe)
|
|
{
|
|
}
|
|
|
|
/* OUTPUT STREAM */
|
|
|
|
#define GDK_TYPE_PIPE_OUTPUT_STREAM (gdk_pipe_output_stream_get_type ())
|
|
#define GDK_PIPE_OUTPUT_STREAM(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GDK_TYPE_PIPE_OUTPUT_STREAM, GdkPipeOutputStream))
|
|
#define GDK_IS_PIPE_OUTPUT_STREAM(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GDK_TYPE_PIPE_OUTPUT_STREAM))
|
|
#define GDK_PIPE_OUTPUT_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_PIPE_OUTPUT_STREAM, GdkPipeOutputStreamClass))
|
|
#define GDK_IS_PIPE_OUTPUT_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_PIPE_OUTPUT_STREAM))
|
|
#define GDK_PIPE_OUTPUT_STREAM_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_PIPE_OUTPUT_STREAM, GdkPipeOutputStreamClass))
|
|
|
|
typedef struct _GdkPipeOutputStream GdkPipeOutputStream;
|
|
typedef struct _GdkPipeOutputStreamClass GdkPipeOutputStreamClass;
|
|
|
|
struct _GdkPipeOutputStream
|
|
{
|
|
GOutputStream parent;
|
|
|
|
GdkIOPipe *pipe;
|
|
};
|
|
|
|
struct _GdkPipeOutputStreamClass
|
|
{
|
|
GOutputStreamClass parent_class;
|
|
};
|
|
|
|
GType gdk_pipe_output_stream_get_type (void) G_GNUC_CONST;
|
|
|
|
G_DEFINE_TYPE (GdkPipeOutputStream, gdk_pipe_output_stream, G_TYPE_OUTPUT_STREAM)
|
|
|
|
static void
|
|
gdk_pipe_output_stream_finalize (GObject *object)
|
|
{
|
|
GdkPipeOutputStream *pipe = GDK_PIPE_OUTPUT_STREAM (object);
|
|
|
|
g_clear_pointer (&pipe->pipe, gdk_io_pipe_unref);
|
|
|
|
G_OBJECT_CLASS (gdk_pipe_output_stream_parent_class)->finalize (object);
|
|
}
|
|
|
|
static gssize
|
|
gdk_pipe_output_stream_write (GOutputStream *stream,
|
|
const void *buffer,
|
|
gsize count,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
GdkPipeOutputStream *pipe_stream = GDK_PIPE_OUTPUT_STREAM (stream);
|
|
GdkIOPipe *pipe = pipe_stream->pipe;
|
|
gsize amount;
|
|
|
|
gdk_io_pipe_lock (pipe);
|
|
|
|
switch (pipe->state)
|
|
{
|
|
case GDK_IO_PIPE_EMPTY:
|
|
pipe->buffer = (void *) buffer;
|
|
pipe->size = count;
|
|
pipe->state = GDK_IO_PIPE_OUTPUT_BUFFER;
|
|
while (pipe->size == count &&
|
|
pipe->state == GDK_IO_PIPE_OUTPUT_BUFFER &&
|
|
!pipe->input_closed)
|
|
g_cond_wait (&pipe->cond, &pipe->mutex);
|
|
if (pipe->state == GDK_IO_PIPE_OUTPUT_BUFFER)
|
|
{
|
|
amount = count - pipe->size;
|
|
pipe->state = GDK_IO_PIPE_EMPTY;
|
|
pipe->size = 0;
|
|
if (pipe->input_closed && amount == 0)
|
|
amount = count;
|
|
}
|
|
else
|
|
{
|
|
amount = count;
|
|
}
|
|
break;
|
|
|
|
case GDK_IO_PIPE_INPUT_BUFFER:
|
|
amount = MIN (count, pipe->size);
|
|
|
|
memcpy (pipe->buffer, buffer, amount);
|
|
pipe->size -= amount;
|
|
|
|
if (pipe->size == 0)
|
|
pipe->state = GDK_IO_PIPE_EMPTY;
|
|
else
|
|
pipe->buffer += amount;
|
|
break;
|
|
|
|
case GDK_IO_PIPE_OUTPUT_BUFFER:
|
|
default:
|
|
g_assert_not_reached ();
|
|
amount = 0;
|
|
break;
|
|
}
|
|
|
|
g_cond_broadcast (&pipe->cond);
|
|
gdk_io_pipe_unlock (pipe);
|
|
|
|
return amount;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pipe_output_stream_close (GOutputStream *stream,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
GdkPipeOutputStream *pipe_stream = GDK_PIPE_OUTPUT_STREAM (stream);
|
|
GdkIOPipe *pipe = pipe_stream->pipe;
|
|
|
|
gdk_io_pipe_lock (pipe);
|
|
|
|
pipe->output_closed = TRUE;
|
|
|
|
g_cond_broadcast (&pipe->cond);
|
|
gdk_io_pipe_unlock (pipe);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_output_stream_class_init (GdkPipeOutputStreamClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
GOutputStreamClass *output_stream_class = G_OUTPUT_STREAM_CLASS (class);
|
|
|
|
object_class->finalize = gdk_pipe_output_stream_finalize;
|
|
|
|
output_stream_class->write_fn = gdk_pipe_output_stream_write;
|
|
output_stream_class->close_fn = gdk_pipe_output_stream_close;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_output_stream_init (GdkPipeOutputStream *pipe)
|
|
{
|
|
}
|
|
|
|
/* IOSTREAM */
|
|
|
|
#define GDK_TYPE_PIPE_IO_STREAM (gdk_pipe_io_stream_get_type ())
|
|
#define GDK_PIPE_IO_STREAM(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GDK_TYPE_PIPE_IO_STREAM, GdkPipeIOStream))
|
|
#define GDK_IS_PIPE_IO_STREAM(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GDK_TYPE_PIPE_IO_STREAM))
|
|
#define GDK_PIPE_IO_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_PIPE_IO_STREAM, GdkPipeIOStreamClass))
|
|
#define GDK_IS_PIPE_IO_STREAM_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_PIPE_IO_STREAM))
|
|
#define GDK_PIPE_IO_STREAM_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_PIPE_IO_STREAM, GdkPipeIOStreamClass))
|
|
|
|
typedef struct _GdkPipeIOStream GdkPipeIOStream;
|
|
typedef struct _GdkPipeIOStreamClass GdkPipeIOStreamClass;
|
|
|
|
struct _GdkPipeIOStream
|
|
{
|
|
GIOStream parent;
|
|
|
|
GInputStream *input_stream;
|
|
GOutputStream *output_stream;
|
|
GdkIOPipe *pipe;
|
|
};
|
|
|
|
struct _GdkPipeIOStreamClass
|
|
{
|
|
GIOStreamClass parent_class;
|
|
};
|
|
|
|
GType gdk_pipe_io_stream_get_type (void) G_GNUC_CONST;
|
|
|
|
G_DEFINE_TYPE (GdkPipeIOStream, gdk_pipe_io_stream, G_TYPE_IO_STREAM)
|
|
|
|
static void
|
|
gdk_pipe_io_stream_finalize (GObject *object)
|
|
{
|
|
GdkPipeIOStream *pipe = GDK_PIPE_IO_STREAM (object);
|
|
|
|
g_clear_object (&pipe->input_stream);
|
|
g_clear_object (&pipe->output_stream);
|
|
g_clear_pointer (&pipe->pipe, gdk_io_pipe_unref);
|
|
|
|
G_OBJECT_CLASS (gdk_pipe_io_stream_parent_class)->finalize (object);
|
|
}
|
|
|
|
static GInputStream *
|
|
gdk_pipe_io_stream_get_input_stream (GIOStream *stream)
|
|
{
|
|
GdkPipeIOStream *pipe = GDK_PIPE_IO_STREAM (stream);
|
|
|
|
return pipe->input_stream;
|
|
}
|
|
|
|
static GOutputStream *
|
|
gdk_pipe_io_stream_get_output_stream (GIOStream *stream)
|
|
{
|
|
GdkPipeIOStream *pipe = GDK_PIPE_IO_STREAM (stream);
|
|
|
|
return pipe->output_stream;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_pipe_io_stream_close (GIOStream *stream,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
/* overwrite so we don't close the 2 streams */
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_io_stream_class_init (GdkPipeIOStreamClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
GIOStreamClass *io_class = G_IO_STREAM_CLASS (class);
|
|
|
|
object_class->finalize = gdk_pipe_io_stream_finalize;
|
|
|
|
io_class->get_input_stream = gdk_pipe_io_stream_get_input_stream;
|
|
io_class->get_output_stream = gdk_pipe_io_stream_get_output_stream;
|
|
io_class->close_fn = gdk_pipe_io_stream_close;
|
|
}
|
|
|
|
static void
|
|
gdk_pipe_io_stream_init (GdkPipeIOStream *pipe)
|
|
{
|
|
pipe->pipe = gdk_io_pipe_new ();
|
|
|
|
pipe->input_stream = g_object_new (GDK_TYPE_PIPE_INPUT_STREAM, NULL);
|
|
GDK_PIPE_INPUT_STREAM (pipe->input_stream)->pipe = gdk_io_pipe_ref (pipe->pipe);
|
|
|
|
pipe->output_stream = g_object_new (GDK_TYPE_PIPE_OUTPUT_STREAM, NULL);
|
|
GDK_PIPE_OUTPUT_STREAM (pipe->output_stream)->pipe = gdk_io_pipe_ref (pipe->pipe);
|
|
}
|
|
|
|
/**
|
|
* gdk_pipe_io_stream_new:
|
|
*
|
|
* Creates a `GIOStream` whose input- and output-stream behave like a pipe.
|
|
*
|
|
* Data written into the output stream becomes available for reading on
|
|
* the input stream.
|
|
*
|
|
* Note that this is data transfer in the opposite direction to
|
|
* g_output_stream_splice().
|
|
*
|
|
* Returns: a new `GIOStream`
|
|
*/
|
|
GIOStream *
|
|
gdk_pipe_io_stream_new (void)
|
|
{
|
|
return g_object_new (GDK_TYPE_PIPE_IO_STREAM, NULL);
|
|
}
|