1 /* guuid.c - UUID functions
2 *
3 * Copyright (C) 2013-2015, 2017 Red Hat, Inc.
4 *
5 * This library is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU Lesser General Public License as
7 * published by the Free Software Foundation; either version 2.1 of the
8 * licence, or (at your option) any later version.
9 *
10 * This is distributed in the hope that it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
13 * License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301
18 * USA.
19 *
20 * Authors: Marc-André Lureau <marcandre.lureau@redhat.com>
21 */
22
23 #include "config.h"
24 #include <string.h>
25
26 #include "gi18n.h"
27 #include "gstrfuncs.h"
28 #include "grand.h"
29 #include "gmessages.h"
30 #include "gchecksum.h"
31
32 #include "guuid.h"
33
34 typedef struct {
35 guint8 bytes[16];
36 } GUuid;
37
38 /**
39 * SECTION:uuid
40 * @title: GUuid
41 * @short_description: a universally unique identifier
42 *
43 * A UUID, or Universally unique identifier, is intended to uniquely
44 * identify information in a distributed environment. For the
45 * definition of UUID, see [RFC 4122](https://tools.ietf.org/html/rfc4122.html).
46 *
47 * The creation of UUIDs does not require a centralized authority.
48 *
49 * UUIDs are of relatively small size (128 bits, or 16 bytes). The
50 * common string representation (ex:
51 * 1d6c0810-2bd6-45f3-9890-0268422a6f14) needs 37 bytes.
52 *
53 * The UUID specification defines 5 versions, and calling
54 * g_uuid_string_random() will generate a unique (or rather random)
55 * UUID of the most common version, version 4.
56 *
57 * Since: 2.52
58 */
59
60 /*
61 * g_uuid_to_string:
62 * @uuid: a #GUuid
63 *
64 * Creates a string representation of @uuid, of the form
65 * 06e023d5-86d8-420e-8103-383e4566087a (no braces nor urn:uuid:
66 * prefix).
67 *
68 * Returns: (transfer full): A string that should be freed with g_free().
69 * Since: STATIC
70 */
71 static gchar *
g_uuid_to_string(const GUuid * uuid)72 g_uuid_to_string (const GUuid *uuid)
73 {
74 const guint8 *bytes;
75
76 g_return_val_if_fail (uuid != NULL, NULL);
77
78 bytes = uuid->bytes;
79
80 return g_strdup_printf ("%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x"
81 "-%02x%02x%02x%02x%02x%02x",
82 bytes[0], bytes[1], bytes[2], bytes[3],
83 bytes[4], bytes[5], bytes[6], bytes[7],
84 bytes[8], bytes[9], bytes[10], bytes[11],
85 bytes[12], bytes[13], bytes[14], bytes[15]);
86 }
87
88 static gboolean
uuid_parse_string(const gchar * str,GUuid * uuid)89 uuid_parse_string (const gchar *str,
90 GUuid *uuid)
91 {
92 GUuid tmp;
93 guint8 *bytes = tmp.bytes;
94 gint i, j, hi, lo;
95 guint expected_len = 36;
96
97 if (strlen (str) != expected_len)
98 return FALSE;
99
100 for (i = 0, j = 0; i < 16;)
101 {
102 if (j == 8 || j == 13 || j == 18 || j == 23)
103 {
104 if (str[j++] != '-')
105 return FALSE;
106
107 continue;
108 }
109
110 hi = g_ascii_xdigit_value (str[j++]);
111 lo = g_ascii_xdigit_value (str[j++]);
112
113 if (hi == -1 || lo == -1)
114 return FALSE;
115
116 bytes[i++] = hi << 8 | lo;
117 }
118
119 if (uuid != NULL)
120 *uuid = tmp;
121
122 return TRUE;
123 }
124
125 /**
126 * g_uuid_string_is_valid:
127 * @str: a string representing a UUID
128 *
129 * Parses the string @str and verify if it is a UUID.
130 *
131 * The function accepts the following syntax:
132 *
133 * - simple forms (e.g. `f81d4fae-7dec-11d0-a765-00a0c91e6bf6`)
134 *
135 * Note that hyphens are required within the UUID string itself,
136 * as per the aforementioned RFC.
137 *
138 * Returns: %TRUE if @str is a valid UUID, %FALSE otherwise.
139 * Since: 2.52
140 */
141 gboolean
g_uuid_string_is_valid(const gchar * str)142 g_uuid_string_is_valid (const gchar *str)
143 {
144 g_return_val_if_fail (str != NULL, FALSE);
145
146 return uuid_parse_string (str, NULL);
147 }
148
149 static void
uuid_set_version(GUuid * uuid,guint version)150 uuid_set_version (GUuid *uuid, guint version)
151 {
152 guint8 *bytes = uuid->bytes;
153
154 /*
155 * Set the four most significant bits (bits 12 through 15) of the
156 * time_hi_and_version field to the 4-bit version number from
157 * Section 4.1.3.
158 */
159 bytes[6] &= 0x0f;
160 bytes[6] |= version << 4;
161 /*
162 * Set the two most significant bits (bits 6 and 7) of the
163 * clock_seq_hi_and_reserved to zero and one, respectively.
164 */
165 bytes[8] &= 0x3f;
166 bytes[8] |= 0x80;
167 }
168
169 /*
170 * g_uuid_generate_v4:
171 * @uuid: a #GUuid
172 *
173 * Generates a random UUID (RFC 4122 version 4).
174 * Since: STATIC
175 */
176 static void
g_uuid_generate_v4(GUuid * uuid)177 g_uuid_generate_v4 (GUuid *uuid)
178 {
179 int i;
180 guint8 *bytes;
181 guint32 *ints;
182
183 g_return_if_fail (uuid != NULL);
184
185 bytes = uuid->bytes;
186 ints = (guint32 *) bytes;
187 for (i = 0; i < 4; i++)
188 ints[i] = g_random_int ();
189
190 uuid_set_version (uuid, 4);
191 }
192
193 /**
194 * g_uuid_string_random:
195 *
196 * Generates a random UUID (RFC 4122 version 4) as a string. It has the same
197 * randomness guarantees as #GRand, so must not be used for cryptographic
198 * purposes such as key generation, nonces, salts or one-time pads.
199 *
200 * Returns: (transfer full): A string that should be freed with g_free().
201 * Since: 2.52
202 */
203 gchar *
g_uuid_string_random(void)204 g_uuid_string_random (void)
205 {
206 GUuid uuid;
207
208 g_uuid_generate_v4 (&uuid);
209
210 return g_uuid_to_string (&uuid);
211 }
212