X-Git-Url: http://review.tizen.org/git/?a=blobdiff_plain;f=gio%2Fgmemoryinputstream.c;h=d8225885a11c704b45fe4573f5462dd9d5415ceb;hb=2e5bd8cf47f9e1559ccc44823a2f321b8ff8c1ea;hp=47f4873703000f2e9e496aa5668193b3b4b6189e;hpb=a5876e5fc1b1e192e8cde664780957798b8a2607;p=platform%2Fupstream%2Fglib.git diff --git a/gio/gmemoryinputstream.c b/gio/gmemoryinputstream.c index 47f4873..d822588 100644 --- a/gio/gmemoryinputstream.c +++ b/gio/gmemoryinputstream.c @@ -13,19 +13,18 @@ * 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. + * Public License along with this library; if not, see . * * Author: Christian Kellner */ #include "config.h" #include "gmemoryinputstream.h" +#include "gpollableinputstream.h" #include "ginputstream.h" #include "gseekable.h" #include "string.h" -#include "gsimpleasyncresult.h" +#include "gtask.h" #include "gioerror.h" #include "glibintl.h" @@ -39,16 +38,10 @@ * #GMemoryInputStream is a class for using arbitrary * memory chunks as input for GIO streaming input operations. * + * As of GLib 2.34, #GMemoryInputStream implements + * #GPollableInputStream. */ -typedef struct _Chunk Chunk; - -struct _Chunk { - guint8 *data; - gsize len; - GDestroyNotify destroy; -}; - struct _GMemoryInputStreamPrivate { GSList *chunks; gsize len; @@ -67,16 +60,6 @@ static gssize g_memory_input_stream_skip (GInputStream *stream static gboolean g_memory_input_stream_close (GInputStream *stream, GCancellable *cancellable, GError **error); -static void g_memory_input_stream_read_async (GInputStream *stream, - void *buffer, - gsize count, - int io_priority, - GCancellable *cancellable, - GAsyncReadyCallback callback, - gpointer user_data); -static gssize g_memory_input_stream_read_finish (GInputStream *stream, - GAsyncResult *result, - GError **error); static void g_memory_input_stream_skip_async (GInputStream *stream, gsize count, int io_priority, @@ -108,11 +91,21 @@ static gboolean g_memory_input_stream_truncate (GSeekable *seek goffset offset, GCancellable *cancellable, GError **error); + +static void g_memory_input_stream_pollable_iface_init (GPollableInputStreamInterface *iface); +static gboolean g_memory_input_stream_is_readable (GPollableInputStream *stream); +static GSource *g_memory_input_stream_create_source (GPollableInputStream *stream, + GCancellable *cancellable); + static void g_memory_input_stream_finalize (GObject *object); G_DEFINE_TYPE_WITH_CODE (GMemoryInputStream, g_memory_input_stream, G_TYPE_INPUT_STREAM, + G_ADD_PRIVATE (GMemoryInputStream) G_IMPLEMENT_INTERFACE (G_TYPE_SEEKABLE, - g_memory_input_stream_seekable_iface_init)) + g_memory_input_stream_seekable_iface_init); + G_IMPLEMENT_INTERFACE (G_TYPE_POLLABLE_INPUT_STREAM, + g_memory_input_stream_pollable_iface_init); + ) static void @@ -121,8 +114,6 @@ g_memory_input_stream_class_init (GMemoryInputStreamClass *klass) GObjectClass *object_class; GInputStreamClass *istream_class; - g_type_class_add_private (klass, sizeof (GMemoryInputStreamPrivate)); - object_class = G_OBJECT_CLASS (klass); object_class->finalize = g_memory_input_stream_finalize; @@ -131,8 +122,6 @@ g_memory_input_stream_class_init (GMemoryInputStreamClass *klass) istream_class->skip = g_memory_input_stream_skip; istream_class->close_fn = g_memory_input_stream_close; - istream_class->read_async = g_memory_input_stream_read_async; - istream_class->read_finish = g_memory_input_stream_read_finish; istream_class->skip_async = g_memory_input_stream_skip_async; istream_class->skip_finish = g_memory_input_stream_skip_finish; istream_class->close_async = g_memory_input_stream_close_async; @@ -140,18 +129,6 @@ g_memory_input_stream_class_init (GMemoryInputStreamClass *klass) } static void -free_chunk (gpointer data, - gpointer user_data) -{ - Chunk *chunk = data; - - if (chunk->destroy) - chunk->destroy (chunk->data); - - g_slice_free (Chunk, chunk); -} - -static void g_memory_input_stream_finalize (GObject *object) { GMemoryInputStream *stream; @@ -160,8 +137,7 @@ g_memory_input_stream_finalize (GObject *object) stream = G_MEMORY_INPUT_STREAM (object); priv = stream->priv; - g_slist_foreach (priv->chunks, free_chunk, NULL); - g_slist_free (priv->chunks); + g_slist_free_full (priv->chunks, (GDestroyNotify)g_bytes_unref); G_OBJECT_CLASS (g_memory_input_stream_parent_class)->finalize (object); } @@ -177,11 +153,16 @@ g_memory_input_stream_seekable_iface_init (GSeekableIface *iface) } static void +g_memory_input_stream_pollable_iface_init (GPollableInputStreamInterface *iface) +{ + iface->is_readable = g_memory_input_stream_is_readable; + iface->create_source = g_memory_input_stream_create_source; +} + +static void g_memory_input_stream_init (GMemoryInputStream *stream) { - stream->priv = G_TYPE_INSTANCE_GET_PRIVATE (stream, - G_TYPE_MEMORY_INPUT_STREAM, - GMemoryInputStreamPrivate); + stream->priv = g_memory_input_stream_get_instance_private (stream); } /** @@ -203,7 +184,7 @@ g_memory_input_stream_new (void) /** * g_memory_input_stream_new_from_data: - * @data: (array length=len) (element-type guint8): input data + * @data: (array length=len) (element-type guint8) (transfer full): input data * @len: length of the data, may be -1 if @data is a nul-terminated string * @destroy: (allow-none): function that is called to free @data, or %NULL * @@ -227,9 +208,33 @@ g_memory_input_stream_new_from_data (const void *data, } /** + * g_memory_input_stream_new_from_bytes: + * @bytes: a #GBytes + * + * Creates a new #GMemoryInputStream with data from the given @bytes. + * + * Returns: new #GInputStream read from @bytes + * + * Since: 2.34 + **/ +GInputStream * +g_memory_input_stream_new_from_bytes (GBytes *bytes) +{ + + GInputStream *stream; + + stream = g_memory_input_stream_new (); + + g_memory_input_stream_add_bytes (G_MEMORY_INPUT_STREAM (stream), + bytes); + + return stream; +} + +/** * g_memory_input_stream_add_data: * @stream: a #GMemoryInputStream - * @data: (array length=len) (element-type guint8): input data + * @data: (array length=len) (element-type guint8) (transfer full): input data * @len: length of the data, may be -1 if @data is a nul-terminated string * @destroy: (allow-none): function that is called to free @data, or %NULL * @@ -241,24 +246,43 @@ g_memory_input_stream_add_data (GMemoryInputStream *stream, gssize len, GDestroyNotify destroy) { + GBytes *bytes; + + if (len == -1) + len = strlen (data); + + /* It's safe to discard the const here because we're chaining the + * destroy callback. + */ + bytes = g_bytes_new_with_free_func (data, len, destroy, (void*)data); + + g_memory_input_stream_add_bytes (stream, bytes); + + g_bytes_unref (bytes); +} + +/** + * g_memory_input_stream_add_bytes: + * @stream: a #GMemoryInputStream + * @bytes: input data + * + * Appends @bytes to data that can be read from the input stream. + * + * Since: 2.34 + */ +void +g_memory_input_stream_add_bytes (GMemoryInputStream *stream, + GBytes *bytes) +{ GMemoryInputStreamPrivate *priv; - Chunk *chunk; g_return_if_fail (G_IS_MEMORY_INPUT_STREAM (stream)); - g_return_if_fail (data != NULL); + g_return_if_fail (bytes != NULL); priv = stream->priv; - if (len == -1) - len = strlen (data); - - chunk = g_slice_new (Chunk); - chunk->data = (guint8 *)data; - chunk->len = len; - chunk->destroy = destroy; - - priv->chunks = g_slist_append (priv->chunks, chunk); - priv->len += chunk->len; + priv->chunks = g_slist_append (priv->chunks, g_bytes_ref (bytes)); + priv->len += g_bytes_get_size (bytes); } static gssize @@ -271,7 +295,8 @@ g_memory_input_stream_read (GInputStream *stream, GMemoryInputStream *memory_stream; GMemoryInputStreamPrivate *priv; GSList *l; - Chunk *chunk; + GBytes *chunk; + gsize len; gsize offset, start, rest, size; memory_stream = G_MEMORY_INPUT_STREAM (stream); @@ -282,12 +307,13 @@ g_memory_input_stream_read (GInputStream *stream, offset = 0; for (l = priv->chunks; l; l = l->next) { - chunk = (Chunk *)l->data; + chunk = (GBytes *)l->data; + len = g_bytes_get_size (chunk); - if (offset + chunk->len > priv->pos) + if (offset + len > priv->pos) break; - offset += chunk->len; + offset += len; } start = priv->pos - offset; @@ -295,10 +321,14 @@ g_memory_input_stream_read (GInputStream *stream, for (; l && rest > 0; l = l->next) { - chunk = (Chunk *)l->data; - size = MIN (rest, chunk->len - start); + const guint8* chunk_data; + chunk = (GBytes *)l->data; - memcpy ((guint8 *)buffer + (count - rest), chunk->data + start, size); + chunk_data = g_bytes_get_data (chunk, &len); + + size = MIN (rest, len - start); + + memcpy ((guint8 *)buffer + (count - rest), chunk_data + start, size); rest -= size; start = 0; @@ -336,43 +366,6 @@ g_memory_input_stream_close (GInputStream *stream, } static void -g_memory_input_stream_read_async (GInputStream *stream, - void *buffer, - gsize count, - int io_priority, - GCancellable *cancellable, - GAsyncReadyCallback callback, - gpointer user_data) -{ - GSimpleAsyncResult *simple; - gssize nread; - - nread = g_input_stream_read (stream, buffer, count, cancellable, NULL); - simple = g_simple_async_result_new (G_OBJECT (stream), - callback, - user_data, - g_memory_input_stream_read_async); - g_simple_async_result_set_op_res_gssize (simple, nread); - g_simple_async_result_complete_in_idle (simple); - g_object_unref (simple); -} - -static gssize -g_memory_input_stream_read_finish (GInputStream *stream, - GAsyncResult *result, - GError **error) -{ - GSimpleAsyncResult *simple; - gssize nread; - - simple = G_SIMPLE_ASYNC_RESULT (result); - g_warn_if_fail (g_simple_async_result_get_source_tag (simple) == g_memory_input_stream_read_async); - - nread = g_simple_async_result_get_op_res_gssize (simple); - return nread; -} - -static void g_memory_input_stream_skip_async (GInputStream *stream, gsize count, int io_priority, @@ -380,17 +373,17 @@ g_memory_input_stream_skip_async (GInputStream *stream, GAsyncReadyCallback callback, gpointer user_data) { - GSimpleAsyncResult *simple; + GTask *task; gssize nskipped; - - nskipped = g_input_stream_skip (stream, count, cancellable, NULL); - simple = g_simple_async_result_new (G_OBJECT (stream), - callback, - user_data, - g_memory_input_stream_skip_async); - g_simple_async_result_set_op_res_gssize (simple, nskipped); - g_simple_async_result_complete_in_idle (simple); - g_object_unref (simple); + GError *error = NULL; + + nskipped = G_INPUT_STREAM_GET_CLASS (stream)->skip (stream, count, cancellable, &error); + task = g_task_new (stream, cancellable, callback, user_data); + if (error) + g_task_return_error (task, error); + else + g_task_return_int (task, nskipped); + g_object_unref (task); } static gssize @@ -398,14 +391,9 @@ g_memory_input_stream_skip_finish (GInputStream *stream, GAsyncResult *result, GError **error) { - GSimpleAsyncResult *simple; - gssize nskipped; + g_return_val_if_fail (g_task_is_valid (result, stream), -1); - simple = G_SIMPLE_ASYNC_RESULT (result); - g_warn_if_fail (g_simple_async_result_get_source_tag (simple) == g_memory_input_stream_skip_async); - - nskipped = g_simple_async_result_get_op_res_gssize (simple); - return nskipped; + return g_task_propagate_int (G_TASK (result), error); } static void @@ -415,14 +403,11 @@ g_memory_input_stream_close_async (GInputStream *stream, GAsyncReadyCallback callback, gpointer user_data) { - GSimpleAsyncResult *simple; - - simple = g_simple_async_result_new (G_OBJECT (stream), - callback, - user_data, - g_memory_input_stream_close_async); - g_simple_async_result_complete_in_idle (simple); - g_object_unref (simple); + GTask *task; + + task = g_task_new (stream, cancellable, callback, user_data); + g_task_return_boolean (task, TRUE); + g_object_unref (task); } static gboolean @@ -520,3 +505,23 @@ g_memory_input_stream_truncate (GSeekable *seekable, _("Cannot truncate GMemoryInputStream")); return FALSE; } + +static gboolean +g_memory_input_stream_is_readable (GPollableInputStream *stream) +{ + return TRUE; +} + +static GSource * +g_memory_input_stream_create_source (GPollableInputStream *stream, + GCancellable *cancellable) +{ + GSource *base_source, *pollable_source; + + base_source = g_timeout_source_new (0); + pollable_source = g_pollable_source_new_full (stream, base_source, + cancellable); + g_source_unref (base_source); + + return pollable_source; +}