xref: /dflybsd-src/stand/lib/uuid_from_string.c (revision bb35c4a8079c5ced869d33874def2c51af1e4c29)
1479ab7f0SSascha Wildner /*-
2479ab7f0SSascha Wildner  * Copyright (c) 2015 M. Warner Losh
3479ab7f0SSascha Wildner  * All rights reserved.
4479ab7f0SSascha Wildner  *
5479ab7f0SSascha Wildner  * Redistribution and use in source and binary forms, with or without
6479ab7f0SSascha Wildner  * modification, are permitted provided that the following conditions
7479ab7f0SSascha Wildner  * are met:
8479ab7f0SSascha Wildner  * 1. Redistributions of source code must retain the above copyright
9479ab7f0SSascha Wildner  *    notice, this list of conditions and the following disclaimer.
10479ab7f0SSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
11479ab7f0SSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
12479ab7f0SSascha Wildner  *    documentation and/or other materials provided with the distribution.
13479ab7f0SSascha Wildner  *
14479ab7f0SSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15479ab7f0SSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16479ab7f0SSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17479ab7f0SSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18479ab7f0SSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19479ab7f0SSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20479ab7f0SSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21479ab7f0SSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22479ab7f0SSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23479ab7f0SSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24479ab7f0SSascha Wildner  * SUCH DAMAGE.
25479ab7f0SSascha Wildner  *
26479ab7f0SSascha Wildner  * $FreeBSD: head/lib/libstand/uuid_from_string.c 300264 2016-05-20 05:33:26Z imp $
27479ab7f0SSascha Wildner  */
28479ab7f0SSascha Wildner 
29479ab7f0SSascha Wildner /*
30479ab7f0SSascha Wildner  * Note: some comments taken from lib/libc/uuid/uuid_from_string.c
31479ab7f0SSascha Wildner  * Copyright (c) 2002 Marcel Moolenaar
32479ab7f0SSascha Wildner  * Copyright (c) 2002 Hiten Mahesh Pandya
33479ab7f0SSascha Wildner  */
34479ab7f0SSascha Wildner 
35479ab7f0SSascha Wildner 
36479ab7f0SSascha Wildner #include <uuid.h>
37*bb35c4a8SSascha Wildner #include "stand.h"
38479ab7f0SSascha Wildner 
39479ab7f0SSascha Wildner static int
hex2int(int ch)40479ab7f0SSascha Wildner hex2int(int ch)
41479ab7f0SSascha Wildner {
42479ab7f0SSascha Wildner 	if (ch >= '0' && ch <= '9')
43479ab7f0SSascha Wildner 		return ch - '0';
44479ab7f0SSascha Wildner 	if (ch >= 'a' && ch <= 'f')
45479ab7f0SSascha Wildner 		return 10 + ch - 'a';
46479ab7f0SSascha Wildner 	if (ch >= 'A' && ch <= 'F')
47479ab7f0SSascha Wildner 		return 10 + ch - 'A';
48479ab7f0SSascha Wildner 	return 16;
49479ab7f0SSascha Wildner }
50479ab7f0SSascha Wildner 
51479ab7f0SSascha Wildner static uint32_t
fromhex(const char * s,int len,int * ok)52479ab7f0SSascha Wildner fromhex(const char *s, int len, int *ok)
53479ab7f0SSascha Wildner {
54479ab7f0SSascha Wildner 	uint32_t v;
55479ab7f0SSascha Wildner 	int i, h;
56479ab7f0SSascha Wildner 
57479ab7f0SSascha Wildner 	if (!*ok)
58479ab7f0SSascha Wildner 		return 0;
59479ab7f0SSascha Wildner 	v = 0;
60479ab7f0SSascha Wildner 	for (i = 0; i < len; i++) {
61479ab7f0SSascha Wildner 		h = hex2int(s[i]);
62479ab7f0SSascha Wildner 		if (h == 16) {
63479ab7f0SSascha Wildner 			*ok = 0;
64479ab7f0SSascha Wildner 			return v;
65479ab7f0SSascha Wildner 		}
66479ab7f0SSascha Wildner 		v = (v << 4) | h;
67479ab7f0SSascha Wildner 	}
68479ab7f0SSascha Wildner 	return v;
69479ab7f0SSascha Wildner }
70479ab7f0SSascha Wildner 
71479ab7f0SSascha Wildner /*
72479ab7f0SSascha Wildner  * uuid_from_string() - convert a string representation of an UUID into
73479ab7f0SSascha Wildner  *			a binary representation.
74479ab7f0SSascha Wildner  * See also:
75479ab7f0SSascha Wildner  *	http://www.opengroup.org/onlinepubs/009629399/uuid_from_string.htm
76479ab7f0SSascha Wildner  *
77479ab7f0SSascha Wildner  * NOTE: The sequence field is in big-endian, while the time fields are in
78479ab7f0SSascha Wildner  *	 native byte order.
79479ab7f0SSascha Wildner  *
80479ab7f0SSascha Wildner  * 01234567-89ab-cdef-0123-456789abcdef
81479ab7f0SSascha Wildner  * 000000000011111111112222222222333333
82479ab7f0SSascha Wildner  * 012345678901234567890123456789012345
83479ab7f0SSascha Wildner  *         -    -    -    -
84479ab7f0SSascha Wildner  * hhhhhhhh-hhhh-hhhh-bbbb-bbbbbbbbbbbb
85479ab7f0SSascha Wildner  *
86479ab7f0SSascha Wildner  */
87479ab7f0SSascha Wildner void
uuid_from_string(const char * s,uuid_t * u,uint32_t * status)88479ab7f0SSascha Wildner uuid_from_string(const char *s, uuid_t *u, uint32_t *status)
89479ab7f0SSascha Wildner {
90479ab7f0SSascha Wildner 	int ok = 1;
91479ab7f0SSascha Wildner 	int n;
92479ab7f0SSascha Wildner 
93479ab7f0SSascha Wildner 	if (s == NULL || *s == '\0') {
94479ab7f0SSascha Wildner 		uuid_create_nil(u, status);
95479ab7f0SSascha Wildner 		return;
96479ab7f0SSascha Wildner 	}
97479ab7f0SSascha Wildner 
98479ab7f0SSascha Wildner 	if (status != NULL)
99479ab7f0SSascha Wildner 		*status = uuid_s_invalid_string_uuid;
100479ab7f0SSascha Wildner 	if (strlen(s) != 36)
101479ab7f0SSascha Wildner 		return;
102479ab7f0SSascha Wildner 	/* Only support new format, check for all the right dashes */
103479ab7f0SSascha Wildner 	if (s[8] != '-' || s[13] != '-' || s[18] != '-' || s[23] != '-')
104479ab7f0SSascha Wildner 		return;
105479ab7f0SSascha Wildner 	/* native byte order */
106479ab7f0SSascha Wildner 	u->time_low                  = fromhex(s     , 8, &ok);
107479ab7f0SSascha Wildner 	u->time_mid                  = fromhex(s +  9, 4, &ok);
108479ab7f0SSascha Wildner 	u->time_hi_and_version       = fromhex(s + 14, 4, &ok);
109479ab7f0SSascha Wildner 	/* Big endian, but presented as a whole number so decode as such */
110479ab7f0SSascha Wildner 	u->clock_seq_hi_and_reserved = fromhex(s + 19, 2, &ok);
111479ab7f0SSascha Wildner 	u->clock_seq_low             = fromhex(s + 21, 2, &ok);
112479ab7f0SSascha Wildner 	u->node[0]                   = fromhex(s + 24, 2, &ok);
113479ab7f0SSascha Wildner 	u->node[1]                   = fromhex(s + 26, 2, &ok);
114479ab7f0SSascha Wildner 	u->node[2]                   = fromhex(s + 28, 2, &ok);
115479ab7f0SSascha Wildner 	u->node[3]                   = fromhex(s + 30, 2, &ok);
116479ab7f0SSascha Wildner 	u->node[4]                   = fromhex(s + 32, 2, &ok);
117479ab7f0SSascha Wildner 	u->node[5]                   = fromhex(s + 34, 2, &ok);
118479ab7f0SSascha Wildner 	if (!ok)
119479ab7f0SSascha Wildner 		return;
120479ab7f0SSascha Wildner 
121479ab7f0SSascha Wildner 	/* We have a successful scan. Check semantics... */
122479ab7f0SSascha Wildner 	n = u->clock_seq_hi_and_reserved;
123479ab7f0SSascha Wildner 	if ((n & 0x80) != 0x00 &&			/* variant 0? */
124479ab7f0SSascha Wildner 	    (n & 0xc0) != 0x80 &&			/* variant 1? */
125479ab7f0SSascha Wildner 	    (n & 0xe0) != 0xc0) {			/* variant 2? */
126479ab7f0SSascha Wildner 		if (status != NULL)
127479ab7f0SSascha Wildner 			*status = uuid_s_bad_version;
128479ab7f0SSascha Wildner 	} else {
129479ab7f0SSascha Wildner 		if (status != NULL)
130479ab7f0SSascha Wildner 			*status = uuid_s_ok;
131479ab7f0SSascha Wildner 	}
132479ab7f0SSascha Wildner }
133