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