2012-02-01 12:34:08 +00:00
|
|
|
/* libsecret - GLib wrapper for Secret Service
|
2011-09-25 06:22:36 +00:00
|
|
|
*
|
|
|
|
* Copyright 2011 Collabora Ltd.
|
|
|
|
*
|
|
|
|
* This program 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 licence or (at
|
|
|
|
* your option) any later version.
|
|
|
|
*
|
|
|
|
* See the included COPYING file for more information.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
#include "secret-private.h"
|
|
|
|
#include "secret-value.h"
|
2011-09-25 06:22:36 +00:00
|
|
|
|
|
|
|
#include "egg/egg-secure-memory.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
EGG_SECURE_DECLARE (secret_value);
|
2011-09-25 06:22:36 +00:00
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
struct _SecretValue {
|
2011-09-25 06:22:36 +00:00
|
|
|
gint refs;
|
|
|
|
gpointer secret;
|
|
|
|
gsize length;
|
|
|
|
GDestroyNotify destroy;
|
|
|
|
gchar *content_type;
|
|
|
|
};
|
|
|
|
|
|
|
|
GType
|
2012-02-01 12:34:08 +00:00
|
|
|
secret_value_get_type (void)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
|
|
|
static gsize initialized = 0;
|
|
|
|
static GType type = 0;
|
|
|
|
|
|
|
|
if (g_once_init_enter (&initialized)) {
|
2012-02-01 12:34:08 +00:00
|
|
|
type = g_boxed_type_register_static ("SecretValue",
|
|
|
|
(GBoxedCopyFunc)secret_value_ref,
|
|
|
|
(GBoxedFreeFunc)secret_value_unref);
|
2011-09-25 06:22:36 +00:00
|
|
|
g_once_init_leave (&initialized, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue*
|
|
|
|
secret_value_new (const gchar *secret, gssize length, const gchar *content_type)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
|
|
|
gchar *copy;
|
|
|
|
|
2011-11-12 07:08:12 +00:00
|
|
|
g_return_val_if_fail (secret == NULL || length != 0, NULL);
|
2011-09-25 06:22:36 +00:00
|
|
|
g_return_val_if_fail (content_type, NULL);
|
|
|
|
|
|
|
|
if (length < 0)
|
|
|
|
length = strlen (secret);
|
|
|
|
|
|
|
|
copy = egg_secure_alloc (length + 1);
|
|
|
|
memcpy (copy, secret, length);
|
|
|
|
copy[length] = 0;
|
2012-02-01 12:34:08 +00:00
|
|
|
return secret_value_new_full (copy, length, content_type, egg_secure_free);
|
2011-09-25 06:22:36 +00:00
|
|
|
}
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue*
|
|
|
|
secret_value_new_full (gchar *secret, gssize length,
|
2011-09-25 06:22:36 +00:00
|
|
|
const gchar *content_type, GDestroyNotify destroy)
|
|
|
|
{
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue *value;
|
2011-09-25 06:22:36 +00:00
|
|
|
|
2011-11-12 07:08:12 +00:00
|
|
|
g_return_val_if_fail (secret == NULL || length != 0, NULL);
|
2011-09-25 06:22:36 +00:00
|
|
|
g_return_val_if_fail (content_type, NULL);
|
|
|
|
|
|
|
|
if (length < 0)
|
|
|
|
length = strlen (secret);
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
value = g_slice_new0 (SecretValue);
|
2011-11-12 07:08:12 +00:00
|
|
|
value->refs = 1;
|
|
|
|
value->content_type = g_strdup (content_type);
|
2011-09-25 06:22:36 +00:00
|
|
|
value->destroy = destroy;
|
|
|
|
value->length = length;
|
|
|
|
value->secret = secret;
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
const gchar*
|
2012-02-01 12:34:08 +00:00
|
|
|
secret_value_get (SecretValue *value, gsize *length)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (value, NULL);
|
|
|
|
if (length)
|
|
|
|
*length = value->length;
|
|
|
|
return value->secret;
|
|
|
|
}
|
|
|
|
|
|
|
|
const gchar*
|
2012-02-01 12:34:08 +00:00
|
|
|
secret_value_get_content_type (SecretValue *value)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (value, NULL);
|
|
|
|
return value->content_type;
|
|
|
|
}
|
|
|
|
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue*
|
|
|
|
secret_value_ref (SecretValue *value)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (value, NULL);
|
|
|
|
g_atomic_int_inc (&value->refs);
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-02-01 12:34:08 +00:00
|
|
|
secret_value_unref (gpointer value)
|
2011-09-25 06:22:36 +00:00
|
|
|
{
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue *val = value;
|
2011-09-25 06:22:36 +00:00
|
|
|
|
2012-01-16 14:08:29 +00:00
|
|
|
g_return_if_fail (value != NULL);
|
2011-09-25 06:22:36 +00:00
|
|
|
|
|
|
|
if (g_atomic_int_dec_and_test (&val->refs)) {
|
|
|
|
g_free (val->content_type);
|
|
|
|
if (val->destroy)
|
|
|
|
(val->destroy) (val->secret);
|
2012-02-01 12:34:08 +00:00
|
|
|
g_slice_free (SecretValue, val);
|
2011-09-25 06:22:36 +00:00
|
|
|
}
|
|
|
|
}
|
2012-01-16 14:08:29 +00:00
|
|
|
|
|
|
|
gchar *
|
2012-02-01 12:34:08 +00:00
|
|
|
_secret_value_unref_to_password (SecretValue *value)
|
2012-01-16 14:08:29 +00:00
|
|
|
{
|
2012-02-01 12:34:08 +00:00
|
|
|
SecretValue *val = value;
|
2012-01-16 14:08:29 +00:00
|
|
|
gchar *result;
|
|
|
|
|
|
|
|
g_return_val_if_fail (value != NULL, NULL);
|
|
|
|
|
|
|
|
if (val->content_type && !g_str_equal (val->content_type, "text/plain")) {
|
2012-02-01 12:34:08 +00:00
|
|
|
secret_value_unref (value);
|
2012-01-16 14:08:29 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_atomic_int_dec_and_test (&val->refs)) {
|
|
|
|
if (val->destroy == egg_secure_free) {
|
|
|
|
result = val->secret;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
result = egg_secure_strdup (val->secret);
|
|
|
|
if (val->destroy)
|
|
|
|
(val->destroy) (val->secret);
|
|
|
|
}
|
|
|
|
g_free (val->content_type);
|
2012-02-01 12:34:08 +00:00
|
|
|
g_slice_free (SecretValue, val);
|
2012-01-16 14:08:29 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
result = egg_secure_strdup (val->secret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|