1e11938eaSJason Hobbs /*
2e11938eaSJason Hobbs * Copyright 2011 Calxeda, Inc.
3e11938eaSJason Hobbs *
41a459660SWolfgang Denk * SPDX-License-Identifier: GPL-2.0+
5e11938eaSJason Hobbs */
6e11938eaSJason Hobbs
789c8230dSPrzemyslaw Marczak #include <common.h>
8e11938eaSJason Hobbs #include <linux/ctype.h>
9a96a0e61SPrzemyslaw Marczak #include <errno.h>
10a96a0e61SPrzemyslaw Marczak #include <common.h>
11d718ded0SPrzemyslaw Marczak #include <asm/io.h>
12d718ded0SPrzemyslaw Marczak #include <part_efi.h>
13d718ded0SPrzemyslaw Marczak #include <malloc.h>
14e11938eaSJason Hobbs
15e11938eaSJason Hobbs /*
16a96a0e61SPrzemyslaw Marczak * UUID - Universally Unique IDentifier - 128 bits unique number.
17a96a0e61SPrzemyslaw Marczak * There are 5 versions and one variant of UUID defined by RFC4122
184e4815feSPrzemyslaw Marczak * specification. A UUID contains a set of fields. The set varies
194e4815feSPrzemyslaw Marczak * depending on the version of the UUID, as shown below:
204e4815feSPrzemyslaw Marczak * - time, MAC address(v1),
214e4815feSPrzemyslaw Marczak * - user ID(v2),
224e4815feSPrzemyslaw Marczak * - MD5 of name or URL(v3),
234e4815feSPrzemyslaw Marczak * - random data(v4),
244e4815feSPrzemyslaw Marczak * - SHA-1 of name or URL(v5),
254e4815feSPrzemyslaw Marczak *
264e4815feSPrzemyslaw Marczak * Layout of UUID:
274e4815feSPrzemyslaw Marczak * timestamp - 60-bit: time_low, time_mid, time_hi_and_version
284e4815feSPrzemyslaw Marczak * version - 4 bit (bit 4 through 7 of the time_hi_and_version)
294e4815feSPrzemyslaw Marczak * clock seq - 14 bit: clock_seq_hi_and_reserved, clock_seq_low
304e4815feSPrzemyslaw Marczak * variant: - bit 6 and 7 of clock_seq_hi_and_reserved
314e4815feSPrzemyslaw Marczak * node - 48 bit
324e4815feSPrzemyslaw Marczak *
334e4815feSPrzemyslaw Marczak * source: https://www.ietf.org/rfc/rfc4122.txt
34e11938eaSJason Hobbs *
35a96a0e61SPrzemyslaw Marczak * UUID binary format (16 bytes):
36a96a0e61SPrzemyslaw Marczak *
37a96a0e61SPrzemyslaw Marczak * 4B-2B-2B-2B-6B (big endian - network byte order)
38a96a0e61SPrzemyslaw Marczak *
39a96a0e61SPrzemyslaw Marczak * UUID string is 36 length of characters (36 bytes):
40e11938eaSJason Hobbs *
41e11938eaSJason Hobbs * 0 9 14 19 24
42e11938eaSJason Hobbs * xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
43a96a0e61SPrzemyslaw Marczak * be be be be be
44a96a0e61SPrzemyslaw Marczak *
45a96a0e61SPrzemyslaw Marczak * where x is a hexadecimal character. Fields are separated by '-'s.
46a96a0e61SPrzemyslaw Marczak * When converting to a binary UUID, le means the field should be converted
47a96a0e61SPrzemyslaw Marczak * to little endian and be means it should be converted to big endian.
48a96a0e61SPrzemyslaw Marczak *
49a96a0e61SPrzemyslaw Marczak * UUID is also used as GUID (Globally Unique Identifier) with the same binary
50a96a0e61SPrzemyslaw Marczak * format but it differs in string format like below.
51a96a0e61SPrzemyslaw Marczak *
52a96a0e61SPrzemyslaw Marczak * GUID:
53a96a0e61SPrzemyslaw Marczak * 0 9 14 19 24
54a96a0e61SPrzemyslaw Marczak * xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
55e11938eaSJason Hobbs * le le le be be
56a96a0e61SPrzemyslaw Marczak *
57a96a0e61SPrzemyslaw Marczak * GUID is used e.g. in GPT (GUID Partition Table) as a partiions unique id.
58e11938eaSJason Hobbs */
uuid_str_valid(const char * uuid)59e11938eaSJason Hobbs int uuid_str_valid(const char *uuid)
60e11938eaSJason Hobbs {
61e11938eaSJason Hobbs int i, valid;
62e11938eaSJason Hobbs
63e11938eaSJason Hobbs if (uuid == NULL)
64e11938eaSJason Hobbs return 0;
65e11938eaSJason Hobbs
66e11938eaSJason Hobbs for (i = 0, valid = 1; uuid[i] && valid; i++) {
67e11938eaSJason Hobbs switch (i) {
68e11938eaSJason Hobbs case 8: case 13: case 18: case 23:
69e11938eaSJason Hobbs valid = (uuid[i] == '-');
70e11938eaSJason Hobbs break;
71e11938eaSJason Hobbs default:
72e11938eaSJason Hobbs valid = isxdigit(uuid[i]);
73e11938eaSJason Hobbs break;
74e11938eaSJason Hobbs }
75e11938eaSJason Hobbs }
76e11938eaSJason Hobbs
77d718ded0SPrzemyslaw Marczak if (i != UUID_STR_LEN || !valid)
78e11938eaSJason Hobbs return 0;
79e11938eaSJason Hobbs
80e11938eaSJason Hobbs return 1;
81e11938eaSJason Hobbs }
82e11938eaSJason Hobbs
83bcb41dcaSPatrick Delaunay #ifdef CONFIG_PARTITION_TYPE_GUID
84bcb41dcaSPatrick Delaunay static const struct {
85bcb41dcaSPatrick Delaunay const char *string;
86bcb41dcaSPatrick Delaunay efi_guid_t guid;
87bcb41dcaSPatrick Delaunay } list_guid[] = {
88bcb41dcaSPatrick Delaunay {"system", PARTITION_SYSTEM_GUID},
89bcb41dcaSPatrick Delaunay {"mbr", LEGACY_MBR_PARTITION_GUID},
90bcb41dcaSPatrick Delaunay {"msft", PARTITION_MSFT_RESERVED_GUID},
91bcb41dcaSPatrick Delaunay {"data", PARTITION_BASIC_DATA_GUID},
92bcb41dcaSPatrick Delaunay {"linux", PARTITION_LINUX_FILE_SYSTEM_DATA_GUID},
93bcb41dcaSPatrick Delaunay {"raid", PARTITION_LINUX_RAID_GUID},
94bcb41dcaSPatrick Delaunay {"swap", PARTITION_LINUX_SWAP_GUID},
95bcb41dcaSPatrick Delaunay {"lvm", PARTITION_LINUX_LVM_GUID}
96bcb41dcaSPatrick Delaunay };
97bcb41dcaSPatrick Delaunay
98bcb41dcaSPatrick Delaunay /*
99bcb41dcaSPatrick Delaunay * uuid_guid_get_bin() - this function get GUID bin for string
100bcb41dcaSPatrick Delaunay *
101bcb41dcaSPatrick Delaunay * @param guid_str - pointer to partition type string
102bcb41dcaSPatrick Delaunay * @param guid_bin - pointer to allocated array for big endian output [16B]
103bcb41dcaSPatrick Delaunay */
uuid_guid_get_bin(const char * guid_str,unsigned char * guid_bin)104bcb41dcaSPatrick Delaunay int uuid_guid_get_bin(const char *guid_str, unsigned char *guid_bin)
105bcb41dcaSPatrick Delaunay {
106bcb41dcaSPatrick Delaunay int i;
107bcb41dcaSPatrick Delaunay
108bcb41dcaSPatrick Delaunay for (i = 0; i < ARRAY_SIZE(list_guid); i++) {
109bcb41dcaSPatrick Delaunay if (!strcmp(list_guid[i].string, guid_str)) {
110bcb41dcaSPatrick Delaunay memcpy(guid_bin, &list_guid[i].guid, 16);
111bcb41dcaSPatrick Delaunay return 0;
112bcb41dcaSPatrick Delaunay }
113bcb41dcaSPatrick Delaunay }
114bcb41dcaSPatrick Delaunay return -ENODEV;
115bcb41dcaSPatrick Delaunay }
116bcb41dcaSPatrick Delaunay
117bcb41dcaSPatrick Delaunay /*
118bcb41dcaSPatrick Delaunay * uuid_guid_get_str() - this function get string for GUID.
119bcb41dcaSPatrick Delaunay *
120bcb41dcaSPatrick Delaunay * @param guid_bin - pointer to string with partition type guid [16B]
121bcb41dcaSPatrick Delaunay * @param guid_str - pointer to allocated partition type string [7B]
122bcb41dcaSPatrick Delaunay */
uuid_guid_get_str(unsigned char * guid_bin,char * guid_str)123bcb41dcaSPatrick Delaunay int uuid_guid_get_str(unsigned char *guid_bin, char *guid_str)
124bcb41dcaSPatrick Delaunay {
125bcb41dcaSPatrick Delaunay int i;
126bcb41dcaSPatrick Delaunay
127bcb41dcaSPatrick Delaunay *guid_str = 0;
128bcb41dcaSPatrick Delaunay for (i = 0; i < ARRAY_SIZE(list_guid); i++) {
129bcb41dcaSPatrick Delaunay if (!memcmp(list_guid[i].guid.b, guid_bin, 16)) {
130bcb41dcaSPatrick Delaunay strcpy(guid_str, list_guid[i].string);
131bcb41dcaSPatrick Delaunay return 0;
132bcb41dcaSPatrick Delaunay }
133bcb41dcaSPatrick Delaunay }
134bcb41dcaSPatrick Delaunay return -ENODEV;
135bcb41dcaSPatrick Delaunay }
136bcb41dcaSPatrick Delaunay #endif
137bcb41dcaSPatrick Delaunay
138d718ded0SPrzemyslaw Marczak /*
139d718ded0SPrzemyslaw Marczak * uuid_str_to_bin() - convert string UUID or GUID to big endian binary data.
140d718ded0SPrzemyslaw Marczak *
141bcb41dcaSPatrick Delaunay * @param uuid_str - pointer to UUID or GUID string [37B] or GUID shorcut
142d718ded0SPrzemyslaw Marczak * @param uuid_bin - pointer to allocated array for big endian output [16B]
143d718ded0SPrzemyslaw Marczak * @str_format - UUID string format: 0 - UUID; 1 - GUID
144d718ded0SPrzemyslaw Marczak */
uuid_str_to_bin(char * uuid_str,unsigned char * uuid_bin,int str_format)145d718ded0SPrzemyslaw Marczak int uuid_str_to_bin(char *uuid_str, unsigned char *uuid_bin, int str_format)
146e11938eaSJason Hobbs {
147e11938eaSJason Hobbs uint16_t tmp16;
148e11938eaSJason Hobbs uint32_t tmp32;
149e11938eaSJason Hobbs uint64_t tmp64;
150e11938eaSJason Hobbs
151bcb41dcaSPatrick Delaunay if (!uuid_str_valid(uuid_str)) {
152bcb41dcaSPatrick Delaunay #ifdef CONFIG_PARTITION_TYPE_GUID
153bcb41dcaSPatrick Delaunay if (!uuid_guid_get_bin(uuid_str, uuid_bin))
154bcb41dcaSPatrick Delaunay return 0;
155bcb41dcaSPatrick Delaunay #endif
156a96a0e61SPrzemyslaw Marczak return -EINVAL;
157bcb41dcaSPatrick Delaunay }
158a96a0e61SPrzemyslaw Marczak
159d718ded0SPrzemyslaw Marczak if (str_format == UUID_STR_FORMAT_STD) {
160d718ded0SPrzemyslaw Marczak tmp32 = cpu_to_be32(simple_strtoul(uuid_str, NULL, 16));
161d718ded0SPrzemyslaw Marczak memcpy(uuid_bin, &tmp32, 4);
162e11938eaSJason Hobbs
163d718ded0SPrzemyslaw Marczak tmp16 = cpu_to_be16(simple_strtoul(uuid_str + 9, NULL, 16));
164d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 4, &tmp16, 2);
165e11938eaSJason Hobbs
166d718ded0SPrzemyslaw Marczak tmp16 = cpu_to_be16(simple_strtoul(uuid_str + 14, NULL, 16));
167d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 6, &tmp16, 2);
168d718ded0SPrzemyslaw Marczak } else {
169d718ded0SPrzemyslaw Marczak tmp32 = cpu_to_le32(simple_strtoul(uuid_str, NULL, 16));
170d718ded0SPrzemyslaw Marczak memcpy(uuid_bin, &tmp32, 4);
171e11938eaSJason Hobbs
172d718ded0SPrzemyslaw Marczak tmp16 = cpu_to_le16(simple_strtoul(uuid_str + 9, NULL, 16));
173d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 4, &tmp16, 2);
174e11938eaSJason Hobbs
175d718ded0SPrzemyslaw Marczak tmp16 = cpu_to_le16(simple_strtoul(uuid_str + 14, NULL, 16));
176d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 6, &tmp16, 2);
177d718ded0SPrzemyslaw Marczak }
178e11938eaSJason Hobbs
179d718ded0SPrzemyslaw Marczak tmp16 = cpu_to_be16(simple_strtoul(uuid_str + 19, NULL, 16));
180d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 8, &tmp16, 2);
181d718ded0SPrzemyslaw Marczak
182d718ded0SPrzemyslaw Marczak tmp64 = cpu_to_be64(simple_strtoull(uuid_str + 24, NULL, 16));
183d718ded0SPrzemyslaw Marczak memcpy(uuid_bin + 10, (char *)&tmp64 + 2, 6);
184a96a0e61SPrzemyslaw Marczak
185a96a0e61SPrzemyslaw Marczak return 0;
186a96a0e61SPrzemyslaw Marczak }
187a96a0e61SPrzemyslaw Marczak
188d718ded0SPrzemyslaw Marczak /*
189d718ded0SPrzemyslaw Marczak * uuid_bin_to_str() - convert big endian binary data to string UUID or GUID.
190d718ded0SPrzemyslaw Marczak *
191d718ded0SPrzemyslaw Marczak * @param uuid_bin - pointer to binary data of UUID (big endian) [16B]
192d718ded0SPrzemyslaw Marczak * @param uuid_str - pointer to allocated array for output string [37B]
193d718ded0SPrzemyslaw Marczak * @str_format - UUID string format: 0 - UUID; 1 - GUID
194d718ded0SPrzemyslaw Marczak */
uuid_bin_to_str(unsigned char * uuid_bin,char * uuid_str,int str_format)195d718ded0SPrzemyslaw Marczak void uuid_bin_to_str(unsigned char *uuid_bin, char *uuid_str, int str_format)
196a96a0e61SPrzemyslaw Marczak {
197d718ded0SPrzemyslaw Marczak const u8 uuid_char_order[UUID_BIN_LEN] = {0, 1, 2, 3, 4, 5, 6, 7, 8,
198d718ded0SPrzemyslaw Marczak 9, 10, 11, 12, 13, 14, 15};
199d718ded0SPrzemyslaw Marczak const u8 guid_char_order[UUID_BIN_LEN] = {3, 2, 1, 0, 5, 4, 7, 6, 8,
200d718ded0SPrzemyslaw Marczak 9, 10, 11, 12, 13, 14, 15};
201d718ded0SPrzemyslaw Marczak const u8 *char_order;
202a96a0e61SPrzemyslaw Marczak int i;
203a96a0e61SPrzemyslaw Marczak
204d718ded0SPrzemyslaw Marczak /*
205d718ded0SPrzemyslaw Marczak * UUID and GUID bin data - always in big endian:
206d718ded0SPrzemyslaw Marczak * 4B-2B-2B-2B-6B
207d718ded0SPrzemyslaw Marczak * be be be be be
208d718ded0SPrzemyslaw Marczak */
209d718ded0SPrzemyslaw Marczak if (str_format == UUID_STR_FORMAT_STD)
210d718ded0SPrzemyslaw Marczak char_order = uuid_char_order;
211d718ded0SPrzemyslaw Marczak else
212d718ded0SPrzemyslaw Marczak char_order = guid_char_order;
213d718ded0SPrzemyslaw Marczak
214a96a0e61SPrzemyslaw Marczak for (i = 0; i < 16; i++) {
215d718ded0SPrzemyslaw Marczak sprintf(uuid_str, "%02x", uuid_bin[char_order[i]]);
216d718ded0SPrzemyslaw Marczak uuid_str += 2;
217a96a0e61SPrzemyslaw Marczak switch (i) {
218a96a0e61SPrzemyslaw Marczak case 3:
219a96a0e61SPrzemyslaw Marczak case 5:
220a96a0e61SPrzemyslaw Marczak case 7:
221a96a0e61SPrzemyslaw Marczak case 9:
222d718ded0SPrzemyslaw Marczak *uuid_str++ = '-';
223a96a0e61SPrzemyslaw Marczak break;
224a96a0e61SPrzemyslaw Marczak }
225a96a0e61SPrzemyslaw Marczak }
226e11938eaSJason Hobbs }
2274e4815feSPrzemyslaw Marczak
2284e4815feSPrzemyslaw Marczak /*
2294e4815feSPrzemyslaw Marczak * gen_rand_uuid() - this function generates a random binary UUID version 4.
2304e4815feSPrzemyslaw Marczak * In this version all fields beside 4 bits of version and
2314e4815feSPrzemyslaw Marczak * 2 bits of variant are randomly generated.
2324e4815feSPrzemyslaw Marczak *
2334e4815feSPrzemyslaw Marczak * @param uuid_bin - pointer to allocated array [16B]. Output is in big endian.
2344e4815feSPrzemyslaw Marczak */
23589c8230dSPrzemyslaw Marczak #if defined(CONFIG_RANDOM_UUID) || defined(CONFIG_CMD_UUID)
gen_rand_uuid(unsigned char * uuid_bin)2364e4815feSPrzemyslaw Marczak void gen_rand_uuid(unsigned char *uuid_bin)
2374e4815feSPrzemyslaw Marczak {
238*4d770529SHeinrich Schuchardt u32 ptr[4];
239*4d770529SHeinrich Schuchardt struct uuid *uuid = (struct uuid *)ptr;
2404e4815feSPrzemyslaw Marczak int i;
2414e4815feSPrzemyslaw Marczak
2424e4815feSPrzemyslaw Marczak /* Set all fields randomly */
243*4d770529SHeinrich Schuchardt for (i = 0; i < 4; i++)
244*4d770529SHeinrich Schuchardt ptr[i] = rand();
2454e4815feSPrzemyslaw Marczak
246*4d770529SHeinrich Schuchardt clrsetbits_be16(&uuid->time_hi_and_version,
2474e4815feSPrzemyslaw Marczak UUID_VERSION_MASK,
2484e4815feSPrzemyslaw Marczak UUID_VERSION << UUID_VERSION_SHIFT);
2494e4815feSPrzemyslaw Marczak
250*4d770529SHeinrich Schuchardt clrsetbits_8(&uuid->clock_seq_hi_and_reserved,
2514e4815feSPrzemyslaw Marczak UUID_VARIANT_MASK,
2524e4815feSPrzemyslaw Marczak UUID_VARIANT << UUID_VARIANT_SHIFT);
2534e4815feSPrzemyslaw Marczak
254*4d770529SHeinrich Schuchardt memcpy(uuid_bin, uuid, 16);
2554e4815feSPrzemyslaw Marczak }
2564e4815feSPrzemyslaw Marczak
2574e4815feSPrzemyslaw Marczak /*
2584e4815feSPrzemyslaw Marczak * gen_rand_uuid_str() - this function generates UUID v4 (random) in two string
2594e4815feSPrzemyslaw Marczak * formats UUID or GUID.
2604e4815feSPrzemyslaw Marczak *
2614e4815feSPrzemyslaw Marczak * @param uuid_str - pointer to allocated array [37B].
2624e4815feSPrzemyslaw Marczak * @param - uuid output type: UUID - 0, GUID - 1
2634e4815feSPrzemyslaw Marczak */
gen_rand_uuid_str(char * uuid_str,int str_format)2644e4815feSPrzemyslaw Marczak void gen_rand_uuid_str(char *uuid_str, int str_format)
2654e4815feSPrzemyslaw Marczak {
2664e4815feSPrzemyslaw Marczak unsigned char uuid_bin[UUID_BIN_LEN];
2674e4815feSPrzemyslaw Marczak
2684e4815feSPrzemyslaw Marczak /* Generate UUID (big endian) */
2694e4815feSPrzemyslaw Marczak gen_rand_uuid(uuid_bin);
2704e4815feSPrzemyslaw Marczak
2714e4815feSPrzemyslaw Marczak /* Convert UUID bin to UUID or GUID formated STRING */
2724e4815feSPrzemyslaw Marczak uuid_bin_to_str(uuid_bin, uuid_str, str_format);
2734e4815feSPrzemyslaw Marczak }
27489c8230dSPrzemyslaw Marczak
27589c8230dSPrzemyslaw Marczak #ifdef CONFIG_CMD_UUID
do_uuid(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])27689c8230dSPrzemyslaw Marczak int do_uuid(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
27789c8230dSPrzemyslaw Marczak {
27889c8230dSPrzemyslaw Marczak char uuid[UUID_STR_LEN + 1];
27989c8230dSPrzemyslaw Marczak int str_format;
28089c8230dSPrzemyslaw Marczak
28189c8230dSPrzemyslaw Marczak if (!strcmp(argv[0], "uuid"))
28289c8230dSPrzemyslaw Marczak str_format = UUID_STR_FORMAT_STD;
28389c8230dSPrzemyslaw Marczak else
28489c8230dSPrzemyslaw Marczak str_format = UUID_STR_FORMAT_GUID;
28589c8230dSPrzemyslaw Marczak
28689c8230dSPrzemyslaw Marczak if (argc > 2)
28789c8230dSPrzemyslaw Marczak return CMD_RET_USAGE;
28889c8230dSPrzemyslaw Marczak
28989c8230dSPrzemyslaw Marczak gen_rand_uuid_str(uuid, str_format);
29089c8230dSPrzemyslaw Marczak
29189c8230dSPrzemyslaw Marczak if (argc == 1)
29289c8230dSPrzemyslaw Marczak printf("%s\n", uuid);
29389c8230dSPrzemyslaw Marczak else
294382bee57SSimon Glass env_set(argv[1], uuid);
29589c8230dSPrzemyslaw Marczak
29689c8230dSPrzemyslaw Marczak return CMD_RET_SUCCESS;
29789c8230dSPrzemyslaw Marczak }
29889c8230dSPrzemyslaw Marczak
29989c8230dSPrzemyslaw Marczak U_BOOT_CMD(uuid, CONFIG_SYS_MAXARGS, 1, do_uuid,
30089c8230dSPrzemyslaw Marczak "UUID - generate random Universally Unique Identifier",
30189c8230dSPrzemyslaw Marczak "[<varname>]\n"
30289c8230dSPrzemyslaw Marczak "Argument:\n"
30389c8230dSPrzemyslaw Marczak "varname: for set result in a environment variable\n"
30489c8230dSPrzemyslaw Marczak "e.g. uuid uuid_env"
30589c8230dSPrzemyslaw Marczak );
30689c8230dSPrzemyslaw Marczak
30789c8230dSPrzemyslaw Marczak U_BOOT_CMD(guid, CONFIG_SYS_MAXARGS, 1, do_uuid,
30889c8230dSPrzemyslaw Marczak "GUID - generate Globally Unique Identifier based on random UUID",
30989c8230dSPrzemyslaw Marczak "[<varname>]\n"
31089c8230dSPrzemyslaw Marczak "Argument:\n"
31189c8230dSPrzemyslaw Marczak "varname: for set result in a environment variable\n"
31289c8230dSPrzemyslaw Marczak "e.g. guid guid_env"
31389c8230dSPrzemyslaw Marczak );
31439206382SPrzemyslaw Marczak #endif /* CONFIG_CMD_UUID */
31539206382SPrzemyslaw Marczak #endif /* CONFIG_RANDOM_UUID || CONFIG_CMD_UUID */
316