
MetaSelectionSourceMemory currently uses GBytes for its underlying data. This can cause memory overhead when large items, such as HD images, are stored in the clipboard. This commit changes the underlying data structure to a MetaAnonymousFile object, which writes to memfd instead of heap. When reading, MetaSelectionSourceMemory will create a Gio.UnixInputStream from the file descriptor generated by MetaAnonymousFile. We subclass the UnixInputStream as MetaUnixInputStream, to override the stream's close_fn function so that it invokes meta_anonymous_file_close_fd when the stream terminates. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3551>
213 lines
6.5 KiB
C
213 lines
6.5 KiB
C
/*
|
|
* Copyright (C) 2018 Red Hat
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program 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
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Carlos Garnacho <carlosg@gnome.org>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "meta/meta-selection-source-memory.h"
|
|
|
|
#include <gio/gunixinputstream.h>
|
|
|
|
#include "core/meta-anonymous-file.h"
|
|
|
|
struct _MetaSelectionSourceMemory
|
|
{
|
|
MetaSelectionSource parent_instance;
|
|
char *mimetype;
|
|
MetaAnonymousFile *content;
|
|
};
|
|
|
|
G_DEFINE_TYPE (MetaSelectionSourceMemory,
|
|
meta_selection_source_memory,
|
|
META_TYPE_SELECTION_SOURCE)
|
|
|
|
struct _MetaUnixInputStream
|
|
{
|
|
GUnixInputStream parent_instance;
|
|
};
|
|
|
|
#define META_TYPE_UNIX_INPUT_STREAM (meta_unix_input_stream_get_type ())
|
|
G_DECLARE_FINAL_TYPE (MetaUnixInputStream,
|
|
meta_unix_input_stream,
|
|
META, UNIX_INPUT_STREAM,
|
|
GUnixInputStream)
|
|
|
|
G_DEFINE_FINAL_TYPE (MetaUnixInputStream,
|
|
meta_unix_input_stream,
|
|
G_TYPE_UNIX_INPUT_STREAM)
|
|
|
|
static gboolean
|
|
meta_unix_input_stream_close_fn (GInputStream *input_stream,
|
|
GCancellable *cancellable,
|
|
GError **error)
|
|
{
|
|
GUnixInputStream *stream = G_UNIX_INPUT_STREAM (input_stream);
|
|
int fd;
|
|
|
|
if (!g_unix_input_stream_get_close_fd (stream))
|
|
return TRUE;
|
|
|
|
fd = g_unix_input_stream_get_fd (stream);
|
|
meta_anonymous_file_close_fd (fd);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_unix_input_stream_class_init (MetaUnixInputStreamClass *klass)
|
|
{
|
|
GInputStreamClass *input_stream_class = G_INPUT_STREAM_CLASS (klass);
|
|
|
|
input_stream_class->close_fn = meta_unix_input_stream_close_fn;
|
|
}
|
|
|
|
static void
|
|
meta_unix_input_stream_init (MetaUnixInputStream *stream)
|
|
{
|
|
}
|
|
|
|
static GInputStream *
|
|
meta_unix_input_stream_new (int fd)
|
|
{
|
|
return G_INPUT_STREAM (g_object_new (META_TYPE_UNIX_INPUT_STREAM,
|
|
"fd", fd,
|
|
"close-fd", true,
|
|
NULL));
|
|
}
|
|
|
|
static void
|
|
meta_selection_source_memory_read_async (MetaSelectionSource *source,
|
|
const char *mimetype,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
MetaSelectionSourceMemory *source_mem = META_SELECTION_SOURCE_MEMORY (source);
|
|
GInputStream *stream;
|
|
g_autoptr (GTask) task = NULL;
|
|
int fd;
|
|
|
|
if (g_strcmp0 (mimetype, source_mem->mimetype) != 0)
|
|
{
|
|
g_task_report_new_error (source, callback, user_data,
|
|
meta_selection_source_memory_read_async,
|
|
G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
"Mimetype not in selection");
|
|
return;
|
|
}
|
|
|
|
task = g_task_new (source, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, meta_selection_source_memory_read_async);
|
|
|
|
fd = meta_anonymous_file_open_fd (source_mem->content,
|
|
META_ANONYMOUS_FILE_MAPMODE_SHARED);
|
|
|
|
if (fd == -1)
|
|
{
|
|
g_task_report_new_error (source, callback, user_data,
|
|
meta_selection_source_memory_read_async,
|
|
G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
"Failed to open MetaAnonymousFile");
|
|
return;
|
|
}
|
|
|
|
stream = meta_unix_input_stream_new (fd);
|
|
|
|
g_task_return_pointer (task, stream, g_object_unref);
|
|
}
|
|
|
|
static GInputStream *
|
|
meta_selection_source_memory_read_finish (MetaSelectionSource *source,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
g_assert (g_task_get_source_tag (G_TASK (result)) ==
|
|
meta_selection_source_memory_read_async);
|
|
return g_task_propagate_pointer (G_TASK (result), error);
|
|
}
|
|
|
|
static GList *
|
|
meta_selection_source_memory_get_mimetypes (MetaSelectionSource *source)
|
|
{
|
|
MetaSelectionSourceMemory *source_mem = META_SELECTION_SOURCE_MEMORY (source);
|
|
|
|
if (!source_mem->mimetype)
|
|
return NULL;
|
|
|
|
return g_list_prepend (NULL, g_strdup (source_mem->mimetype));
|
|
}
|
|
|
|
static void
|
|
meta_selection_source_memory_finalize (GObject *object)
|
|
{
|
|
MetaSelectionSourceMemory *source_mem = META_SELECTION_SOURCE_MEMORY (object);
|
|
|
|
g_clear_pointer (&source_mem->content, meta_anonymous_file_free);
|
|
g_free (source_mem->mimetype);
|
|
|
|
G_OBJECT_CLASS (meta_selection_source_memory_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
meta_selection_source_memory_class_init (MetaSelectionSourceMemoryClass *klass)
|
|
{
|
|
MetaSelectionSourceClass *source_class = META_SELECTION_SOURCE_CLASS (klass);
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = meta_selection_source_memory_finalize;
|
|
|
|
source_class->read_async = meta_selection_source_memory_read_async;
|
|
source_class->read_finish = meta_selection_source_memory_read_finish;
|
|
source_class->get_mimetypes = meta_selection_source_memory_get_mimetypes;
|
|
}
|
|
|
|
static void
|
|
meta_selection_source_memory_init (MetaSelectionSourceMemory *source)
|
|
{
|
|
}
|
|
|
|
MetaSelectionSource *
|
|
meta_selection_source_memory_new (const char *mimetype,
|
|
GBytes *content,
|
|
GError **error)
|
|
{
|
|
MetaSelectionSourceMemory *source;
|
|
MetaAnonymousFile *anon_file;
|
|
const uint8_t *data;
|
|
size_t size;
|
|
|
|
g_return_val_if_fail (mimetype != NULL, NULL);
|
|
g_return_val_if_fail (content != NULL, NULL);
|
|
|
|
data = g_bytes_get_data (content, &size);
|
|
anon_file = meta_anonymous_file_new (size, data);
|
|
|
|
if (anon_file == NULL)
|
|
{
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
"Failed to create MetaAnonymousFile");
|
|
return NULL;
|
|
}
|
|
|
|
source = g_object_new (META_TYPE_SELECTION_SOURCE_MEMORY, NULL);
|
|
source->mimetype = g_strdup (mimetype);
|
|
source->content = anon_file;
|
|
|
|
return META_SELECTION_SOURCE (source);
|
|
}
|