gnome-shell/src/shell-secure-text-buffer.c
Florian Müllner 294702d3f1 shell: Use G_DECLARE_*_TYPE macros
Cut down on boilerplate by using the (no longer that) new helper
macros. We don't care about breaking ABI in private libraries, so
use G_DECLARE_FINAL_TYPE even where the class struct used to be
exposed in the header, except for types we inherit from ourselves
(obviously) or where the class exposes any vfuncs (where changes
could affect inheritance in extensions).
2015-10-15 22:58:28 +02:00

192 lines
5.8 KiB
C

/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* shell-secure-text-buffer.c - secure memory clutter text buffer
Copyright (C) 2009 Stefan Walter
Copyright (C) 2012 Red Hat Inc.
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU Library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with the Gnome Library; see the file COPYING.LIB. If not,
see <http://www.gnu.org/licenses/>.
Author: Stef Walter <stefw@gnome.org>
*/
#include "config.h"
#include "shell-secure-text-buffer.h"
#define GCR_API_SUBJECT_TO_CHANGE
#include <gcr/gcr-base.h>
#include <string.h>
struct _ShellSecureTextBuffer {
ClutterTextBuffer parent;
gchar *text;
gsize text_size;
gsize text_bytes;
guint text_chars;
};
/* Initial size of buffer, in bytes */
#define MIN_SIZE 16
G_DEFINE_TYPE (ShellSecureTextBuffer, shell_secure_text_buffer, CLUTTER_TYPE_TEXT_BUFFER);
static const gchar *
shell_secure_text_buffer_real_get_text (ClutterTextBuffer *buffer,
gsize *n_bytes)
{
ShellSecureTextBuffer *self = SHELL_SECURE_TEXT_BUFFER (buffer);
if (n_bytes)
*n_bytes = self->text_bytes;
if (!self->text)
return "";
return self->text;
}
static guint
shell_secure_text_buffer_real_get_length (ClutterTextBuffer *buffer)
{
ShellSecureTextBuffer *self = SHELL_SECURE_TEXT_BUFFER (buffer);
return self->text_chars;
}
static guint
shell_secure_text_buffer_real_insert_text (ClutterTextBuffer *buffer,
guint position,
const gchar *chars,
guint n_chars)
{
ShellSecureTextBuffer *self = SHELL_SECURE_TEXT_BUFFER (buffer);
gsize n_bytes;
gsize at;
n_bytes = g_utf8_offset_to_pointer (chars, n_chars) - chars;
/* Need more memory */
if (n_bytes + self->text_bytes + 1 > self->text_size)
{
/* Calculate our new buffer size */
while (n_bytes + self->text_bytes + 1 > self->text_size)
{
if (self->text_size == 0)
{
self->text_size = MIN_SIZE;
}
else
{
if (2 * self->text_size < CLUTTER_TEXT_BUFFER_MAX_SIZE)
{
self->text_size *= 2;
}
else
{
self->text_size = CLUTTER_TEXT_BUFFER_MAX_SIZE;
if (n_bytes > self->text_size - self->text_bytes - 1)
{
n_bytes = self->text_size - self->text_bytes - 1;
n_bytes = g_utf8_find_prev_char (chars, chars + n_bytes + 1) - chars;
n_chars = g_utf8_strlen (chars, n_bytes);
}
break;
}
}
}
self->text = gcr_secure_memory_realloc (self->text, self->text_size);
}
/* Actual text insertion */
at = g_utf8_offset_to_pointer (self->text, position) - self->text;
g_memmove (self->text + at + n_bytes, self->text + at, self->text_bytes - at);
memcpy (self->text + at, chars, n_bytes);
/* Book keeping */
self->text_bytes += n_bytes;
self->text_chars += n_chars;
self->text[self->text_bytes] = '\0';
clutter_text_buffer_emit_inserted_text (buffer, position, chars, n_chars);
return n_chars;
}
static guint
shell_secure_text_buffer_real_delete_text (ClutterTextBuffer *buffer,
guint position,
guint n_chars)
{
ShellSecureTextBuffer *self = SHELL_SECURE_TEXT_BUFFER (buffer);
gsize start, end;
if (position > self->text_chars)
position = self->text_chars;
if (position + n_chars > self->text_chars)
n_chars = self->text_chars - position;
if (n_chars > 0)
{
start = g_utf8_offset_to_pointer (self->text, position) - self->text;
end = g_utf8_offset_to_pointer (self->text, position + n_chars) - self->text;
g_memmove (self->text + start, self->text + end, self->text_bytes + 1 - end);
self->text_chars -= n_chars;
self->text_bytes -= (end - start);
clutter_text_buffer_emit_deleted_text (buffer, position, n_chars);
}
return n_chars;
}
static void
shell_secure_text_buffer_init (ShellSecureTextBuffer *self)
{
}
static void
shell_secure_text_buffer_finalize (GObject *obj)
{
ShellSecureTextBuffer *self = SHELL_SECURE_TEXT_BUFFER (obj);
if (self->text)
{
gcr_secure_memory_strfree (self->text);
self->text = NULL;
self->text_bytes = self->text_size = 0;
self->text_chars = 0;
}
G_OBJECT_CLASS (shell_secure_text_buffer_parent_class)->finalize (obj);
}
static void
shell_secure_text_buffer_class_init (ShellSecureTextBufferClass *klass)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
ClutterTextBufferClass *buffer_class = CLUTTER_TEXT_BUFFER_CLASS (klass);
gobject_class->finalize = shell_secure_text_buffer_finalize;
buffer_class->get_text = shell_secure_text_buffer_real_get_text;
buffer_class->get_length = shell_secure_text_buffer_real_get_length;
buffer_class->insert_text = shell_secure_text_buffer_real_insert_text;
buffer_class->delete_text = shell_secure_text_buffer_real_delete_text;
}
ClutterTextBuffer *
shell_secure_text_buffer_new (void)
{
return g_object_new (SHELL_TYPE_SECURE_TEXT_BUFFER, NULL);
}