xref: /titanic_50/usr/src/lib/libuuid/common/uuid.c (revision 0b5ce10aee80822ecc7df77df92a5e24078ba196)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * CDDL HEADER START
37c478bd9Sstevel@tonic-gate  *
47c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5d62bc4baSyz147064  * Common Development and Distribution License (the "License").
6d62bc4baSyz147064  * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate  *
87c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate  * and limitations under the License.
127c478bd9Sstevel@tonic-gate  *
137c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate  *
197c478bd9Sstevel@tonic-gate  * CDDL HEADER END
207c478bd9Sstevel@tonic-gate  */
217c478bd9Sstevel@tonic-gate /*
22d62bc4baSyz147064  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
237c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
2433f5ff17SMilan Jurik  * Copyright 2012 Milan Jurik. All rights reserved.
2584615402SJerry Jelinek  * Copyright 2013 Joyent, Inc. All rights reserved.
26*0b5ce10aSAndy Stormont  * Copyright 2014 Andrew Stormont.
277c478bd9Sstevel@tonic-gate  */
287c478bd9Sstevel@tonic-gate 
297c478bd9Sstevel@tonic-gate /*
307c478bd9Sstevel@tonic-gate  * The copyright in this file is taken from the original Leach & Salz
317c478bd9Sstevel@tonic-gate  * UUID specification, from which this implementation is derived.
327c478bd9Sstevel@tonic-gate  */
337c478bd9Sstevel@tonic-gate 
347c478bd9Sstevel@tonic-gate /*
357c478bd9Sstevel@tonic-gate  * Copyright (c) 1990- 1993, 1996 Open Software Foundation, Inc.
367c478bd9Sstevel@tonic-gate  * Copyright (c) 1989 by Hewlett-Packard Company, Palo Alto, Ca. &
377c478bd9Sstevel@tonic-gate  * Digital Equipment Corporation, Maynard, Mass.  Copyright (c) 1998
387c478bd9Sstevel@tonic-gate  * Microsoft.  To anyone who acknowledges that this file is provided
397c478bd9Sstevel@tonic-gate  * "AS IS" without any express or implied warranty: permission to use,
407c478bd9Sstevel@tonic-gate  * copy, modify, and distribute this file for any purpose is hereby
417c478bd9Sstevel@tonic-gate  * granted without fee, provided that the above copyright notices and
427c478bd9Sstevel@tonic-gate  * this notice appears in all source code copies, and that none of the
437c478bd9Sstevel@tonic-gate  * names of Open Software Foundation, Inc., Hewlett-Packard Company,
447c478bd9Sstevel@tonic-gate  * or Digital Equipment Corporation be used in advertising or
457c478bd9Sstevel@tonic-gate  * publicity pertaining to distribution of the software without
467c478bd9Sstevel@tonic-gate  * specific, written prior permission.  Neither Open Software
477c478bd9Sstevel@tonic-gate  * Foundation, Inc., Hewlett-Packard Company, Microsoft, nor Digital
487c478bd9Sstevel@tonic-gate  * Equipment Corporation makes any representations about the
497c478bd9Sstevel@tonic-gate  * suitability of this software for any purpose.
507c478bd9Sstevel@tonic-gate  */
517c478bd9Sstevel@tonic-gate 
527c478bd9Sstevel@tonic-gate /*
53cc1a9a89SRafael Vanoni Polanczyk  * This module is the workhorse for generating abstract
547c478bd9Sstevel@tonic-gate  * UUIDs.  It delegates system-specific tasks (such
557c478bd9Sstevel@tonic-gate  * as obtaining the node identifier or system time)
567c478bd9Sstevel@tonic-gate  * to the sysdep module.
577c478bd9Sstevel@tonic-gate  */
587c478bd9Sstevel@tonic-gate 
597c478bd9Sstevel@tonic-gate #include <ctype.h>
607c478bd9Sstevel@tonic-gate #include <sys/param.h>
617c478bd9Sstevel@tonic-gate #include <sys/stat.h>
627c478bd9Sstevel@tonic-gate #include <errno.h>
637c478bd9Sstevel@tonic-gate #include <stdio.h>
647c478bd9Sstevel@tonic-gate #include <stdlib.h>
657c478bd9Sstevel@tonic-gate #include <strings.h>
667c478bd9Sstevel@tonic-gate #include <fcntl.h>
677c478bd9Sstevel@tonic-gate #include <unistd.h>
687c478bd9Sstevel@tonic-gate #include <synch.h>
69cc1a9a89SRafael Vanoni Polanczyk #include <sys/mman.h>
707c478bd9Sstevel@tonic-gate #include "uuid_misc.h"
717c478bd9Sstevel@tonic-gate 
72cc1a9a89SRafael Vanoni Polanczyk shared_buffer_t		*data;
737c478bd9Sstevel@tonic-gate 
74cc1a9a89SRafael Vanoni Polanczyk static	uuid_node_t	node_id_cache;
75cc1a9a89SRafael Vanoni Polanczyk static	int		node_init;
76cc1a9a89SRafael Vanoni Polanczyk static	int		file_type;
77cc1a9a89SRafael Vanoni Polanczyk static	int		fd;
787c478bd9Sstevel@tonic-gate 
79cc1a9a89SRafael Vanoni Polanczyk /*
8084615402SJerry Jelinek  * The urandmtx mutex prevents multiple opens of /dev/urandom and protects the
8184615402SJerry Jelinek  * cache.
8284615402SJerry Jelinek  */
8384615402SJerry Jelinek #define	RCACHE_SIZE	65535
8484615402SJerry Jelinek static	mutex_t		urandmtx;
8584615402SJerry Jelinek static	int		fd_urand = -1;
8684615402SJerry Jelinek static	char		rcache[RCACHE_SIZE];
8784615402SJerry Jelinek static	char		*rcachep = rcache;
8884615402SJerry Jelinek 
8984615402SJerry Jelinek /*
90cc1a9a89SRafael Vanoni Polanczyk  * misc routines
91cc1a9a89SRafael Vanoni Polanczyk  */
92cc1a9a89SRafael Vanoni Polanczyk uint16_t		get_random(void);
93cc1a9a89SRafael Vanoni Polanczyk void			get_current_time(uuid_time_t *);
94cc1a9a89SRafael Vanoni Polanczyk 
957c478bd9Sstevel@tonic-gate void			struct_to_string(uuid_t, struct uuid *);
967c478bd9Sstevel@tonic-gate void			string_to_struct(struct uuid *, uuid_t);
977c478bd9Sstevel@tonic-gate int			get_ethernet_address(uuid_node_t *);
987c478bd9Sstevel@tonic-gate 
997c478bd9Sstevel@tonic-gate /*
1007c478bd9Sstevel@tonic-gate  * local functions
1017c478bd9Sstevel@tonic-gate  */
102cc1a9a89SRafael Vanoni Polanczyk static	int		map_state();
103cc1a9a89SRafael Vanoni Polanczyk static	void 		format_uuid(struct uuid *, uint16_t, uuid_time_t,
1047c478bd9Sstevel@tonic-gate     uuid_node_t);
1057c478bd9Sstevel@tonic-gate static	void		fill_random_bytes(uchar_t *, int);
1067c478bd9Sstevel@tonic-gate static	int		uuid_create(struct uuid *);
1077c478bd9Sstevel@tonic-gate static	void		gen_ethernet_address(uuid_node_t *);
108cc1a9a89SRafael Vanoni Polanczyk static	void		revalidate_data(uuid_node_t *);
109cc1a9a89SRafael Vanoni Polanczyk 
1107c478bd9Sstevel@tonic-gate /*
111cc1a9a89SRafael Vanoni Polanczyk  * Generates a uuid based on version 1 format.
112cc1a9a89SRafael Vanoni Polanczyk  * Returns 0 on success and -1 on failure.
1137c478bd9Sstevel@tonic-gate  */
1147c478bd9Sstevel@tonic-gate static int
1157c478bd9Sstevel@tonic-gate uuid_create(struct uuid *uuid)
1167c478bd9Sstevel@tonic-gate {
117cc1a9a89SRafael Vanoni Polanczyk 	uuid_time_t	timestamp;
1187c478bd9Sstevel@tonic-gate 	uuid_node_t	system_node;
119cc1a9a89SRafael Vanoni Polanczyk 	int		ret, non_unique = 0;
1207c478bd9Sstevel@tonic-gate 
1217c478bd9Sstevel@tonic-gate 	/*
122cc1a9a89SRafael Vanoni Polanczyk 	 * Get the system MAC address and/or cache it
1237c478bd9Sstevel@tonic-gate 	 */
124cc1a9a89SRafael Vanoni Polanczyk 	if (node_init) {
125cc1a9a89SRafael Vanoni Polanczyk 		bcopy(&node_id_cache, &system_node, sizeof (uuid_node_t));
1267c478bd9Sstevel@tonic-gate 	} else {
127cc1a9a89SRafael Vanoni Polanczyk 		gen_ethernet_address(&system_node);
128cc1a9a89SRafael Vanoni Polanczyk 		bcopy(&system_node, &node_id_cache, sizeof (uuid_node_t));
129cc1a9a89SRafael Vanoni Polanczyk 		node_init = 1;
1307c478bd9Sstevel@tonic-gate 	}
1317c478bd9Sstevel@tonic-gate 
132cc1a9a89SRafael Vanoni Polanczyk 	/*
133cc1a9a89SRafael Vanoni Polanczyk 	 * Access the state file, mmap it and initialize the shared lock.
134cc1a9a89SRafael Vanoni Polanczyk 	 * file_type tells us whether we had access to the state file or
135cc1a9a89SRafael Vanoni Polanczyk 	 * created a temporary one.
136cc1a9a89SRafael Vanoni Polanczyk 	 */
137ad2de435SRichard Lowe 	if (map_state() == -1)
138ad2de435SRichard Lowe 		return (-1);
139cc1a9a89SRafael Vanoni Polanczyk 
140cc1a9a89SRafael Vanoni Polanczyk 	/*
141cc1a9a89SRafael Vanoni Polanczyk 	 * Acquire the lock
142cc1a9a89SRafael Vanoni Polanczyk 	 */
143cc1a9a89SRafael Vanoni Polanczyk 	for (;;) {
144cc1a9a89SRafael Vanoni Polanczyk 		if ((ret = mutex_lock(&data->lock)) == 0)
145cc1a9a89SRafael Vanoni Polanczyk 			break;
146cc1a9a89SRafael Vanoni Polanczyk 		else
147cc1a9a89SRafael Vanoni Polanczyk 			switch (ret) {
148cc1a9a89SRafael Vanoni Polanczyk 				case EOWNERDEAD:
149cc1a9a89SRafael Vanoni Polanczyk 					revalidate_data(&system_node);
150cc1a9a89SRafael Vanoni Polanczyk 					(void) mutex_consistent(&data->lock);
151cc1a9a89SRafael Vanoni Polanczyk 					(void) mutex_unlock(&data->lock);
152cc1a9a89SRafael Vanoni Polanczyk 					break;
153cc1a9a89SRafael Vanoni Polanczyk 				case ENOTRECOVERABLE:
154cc1a9a89SRafael Vanoni Polanczyk 					return (ret);
155cc1a9a89SRafael Vanoni Polanczyk 			}
156cc1a9a89SRafael Vanoni Polanczyk 	}
157cc1a9a89SRafael Vanoni Polanczyk 
158cc1a9a89SRafael Vanoni Polanczyk 	/* State file is either new or is temporary, get a random clock seq */
159cc1a9a89SRafael Vanoni Polanczyk 	if (data->state.clock == 0) {
160cc1a9a89SRafael Vanoni Polanczyk 		data->state.clock = get_random();
1617c478bd9Sstevel@tonic-gate 		non_unique++;
1627c478bd9Sstevel@tonic-gate 	}
163cc1a9a89SRafael Vanoni Polanczyk 
164cc1a9a89SRafael Vanoni Polanczyk 	if (memcmp(&system_node, &data->state.node, sizeof (uuid_node_t)) != 0)
165cc1a9a89SRafael Vanoni Polanczyk 		data->state.clock++;
166cc1a9a89SRafael Vanoni Polanczyk 
167cc1a9a89SRafael Vanoni Polanczyk 	get_current_time(&timestamp);
1687c478bd9Sstevel@tonic-gate 
1697c478bd9Sstevel@tonic-gate 	/*
170cc1a9a89SRafael Vanoni Polanczyk 	 * If timestamp is not set or is not in the past, bump
171cc1a9a89SRafael Vanoni Polanczyk 	 * data->state.clock
1727c478bd9Sstevel@tonic-gate 	 */
173cc1a9a89SRafael Vanoni Polanczyk 	if ((data->state.ts == 0) || (data->state.ts >= timestamp)) {
174cc1a9a89SRafael Vanoni Polanczyk 		data->state.clock++;
175cc1a9a89SRafael Vanoni Polanczyk 		data->state.ts = timestamp;
1767c478bd9Sstevel@tonic-gate 	}
1777c478bd9Sstevel@tonic-gate 
1787c478bd9Sstevel@tonic-gate 	if (non_unique)
1797c478bd9Sstevel@tonic-gate 		system_node.nodeID[0] |= 0x80;
180cc1a9a89SRafael Vanoni Polanczyk 
181cc1a9a89SRafael Vanoni Polanczyk 	/* Stuff fields into the UUID struct */
182cc1a9a89SRafael Vanoni Polanczyk 	format_uuid(uuid, data->state.clock, timestamp, system_node);
183cc1a9a89SRafael Vanoni Polanczyk 
184cc1a9a89SRafael Vanoni Polanczyk 	(void) mutex_unlock(&data->lock);
185cc1a9a89SRafael Vanoni Polanczyk 
1867c478bd9Sstevel@tonic-gate 	return (0);
1877c478bd9Sstevel@tonic-gate }
1887c478bd9Sstevel@tonic-gate 
1897c478bd9Sstevel@tonic-gate /*
190cc1a9a89SRafael Vanoni Polanczyk  * Fills system_node with Ethernet address if available,
1917c478bd9Sstevel@tonic-gate  * else fills random numbers
1927c478bd9Sstevel@tonic-gate  */
1937c478bd9Sstevel@tonic-gate static void
1947c478bd9Sstevel@tonic-gate gen_ethernet_address(uuid_node_t *system_node)
1957c478bd9Sstevel@tonic-gate {
1967c478bd9Sstevel@tonic-gate 	uchar_t		node[6];
1977c478bd9Sstevel@tonic-gate 
1987c478bd9Sstevel@tonic-gate 	if (get_ethernet_address(system_node) != 0) {
1997c478bd9Sstevel@tonic-gate 		fill_random_bytes(node, 6);
2007c478bd9Sstevel@tonic-gate 		(void) memcpy(system_node->nodeID, node, 6);
2017c478bd9Sstevel@tonic-gate 		/*
2027c478bd9Sstevel@tonic-gate 		 * use 8:0:20 with the multicast bit set
2037c478bd9Sstevel@tonic-gate 		 * to avoid namespace collisions.
2047c478bd9Sstevel@tonic-gate 		 */
2057c478bd9Sstevel@tonic-gate 		system_node->nodeID[0] = 0x88;
2067c478bd9Sstevel@tonic-gate 		system_node->nodeID[1] = 0x00;
2077c478bd9Sstevel@tonic-gate 		system_node->nodeID[2] = 0x20;
2087c478bd9Sstevel@tonic-gate 	}
2097c478bd9Sstevel@tonic-gate }
2107c478bd9Sstevel@tonic-gate 
2117c478bd9Sstevel@tonic-gate /*
212cc1a9a89SRafael Vanoni Polanczyk  * Formats a UUID, given the clock_seq timestamp, and node address.
213cc1a9a89SRafael Vanoni Polanczyk  * Fills in passed-in pointer with the resulting uuid.
2147c478bd9Sstevel@tonic-gate  */
2157c478bd9Sstevel@tonic-gate static void
216cc1a9a89SRafael Vanoni Polanczyk format_uuid(struct uuid *uuid, uint16_t clock_seq,
2177c478bd9Sstevel@tonic-gate     uuid_time_t timestamp, uuid_node_t node)
2187c478bd9Sstevel@tonic-gate {
2197c478bd9Sstevel@tonic-gate 
2207c478bd9Sstevel@tonic-gate 	/*
2217c478bd9Sstevel@tonic-gate 	 * First set up the first 60 bits from the timestamp
2227c478bd9Sstevel@tonic-gate 	 */
2237c478bd9Sstevel@tonic-gate 	uuid->time_low = (uint32_t)(timestamp & 0xFFFFFFFF);
2247c478bd9Sstevel@tonic-gate 	uuid->time_mid = (uint16_t)((timestamp >> 32) & 0xFFFF);
225cc1a9a89SRafael Vanoni Polanczyk 	uuid->time_hi_and_version = (uint16_t)((timestamp >> 48) & 0x0FFF);
2267c478bd9Sstevel@tonic-gate 
2277c478bd9Sstevel@tonic-gate 	/*
2287c478bd9Sstevel@tonic-gate 	 * This is version 1, so say so in the UUID version field (4 bits)
2297c478bd9Sstevel@tonic-gate 	 */
2307c478bd9Sstevel@tonic-gate 	uuid->time_hi_and_version |= (1 << 12);
2317c478bd9Sstevel@tonic-gate 
2327c478bd9Sstevel@tonic-gate 	/*
2337c478bd9Sstevel@tonic-gate 	 * Now do the clock sequence
2347c478bd9Sstevel@tonic-gate 	 */
2357c478bd9Sstevel@tonic-gate 	uuid->clock_seq_low = clock_seq & 0xFF;
2367c478bd9Sstevel@tonic-gate 
2377c478bd9Sstevel@tonic-gate 	/*
2387c478bd9Sstevel@tonic-gate 	 * We must save the most-significant 2 bits for the reserved field
2397c478bd9Sstevel@tonic-gate 	 */
2407c478bd9Sstevel@tonic-gate 	uuid->clock_seq_hi_and_reserved = (clock_seq & 0x3F00) >> 8;
2417c478bd9Sstevel@tonic-gate 
2427c478bd9Sstevel@tonic-gate 	/*
2437c478bd9Sstevel@tonic-gate 	 * The variant for this format is the 2 high bits set to 10,
2447c478bd9Sstevel@tonic-gate 	 * so here it is
2457c478bd9Sstevel@tonic-gate 	 */
2467c478bd9Sstevel@tonic-gate 	uuid->clock_seq_hi_and_reserved |= 0x80;
2477c478bd9Sstevel@tonic-gate 
2487c478bd9Sstevel@tonic-gate 	/*
2497c478bd9Sstevel@tonic-gate 	 * write result to passed-in pointer
2507c478bd9Sstevel@tonic-gate 	 */
2517c478bd9Sstevel@tonic-gate 	(void) memcpy(&uuid->node_addr, &node, sizeof (uuid->node_addr));
2527c478bd9Sstevel@tonic-gate }
2537c478bd9Sstevel@tonic-gate 
2547c478bd9Sstevel@tonic-gate /*
255cc1a9a89SRafael Vanoni Polanczyk  * Opens/creates the state file, falling back to a tmp
2567c478bd9Sstevel@tonic-gate  */
2577c478bd9Sstevel@tonic-gate static int
258cc1a9a89SRafael Vanoni Polanczyk map_state()
2597c478bd9Sstevel@tonic-gate {
260cc1a9a89SRafael Vanoni Polanczyk 	FILE	*tmp;
2617c478bd9Sstevel@tonic-gate 
262cc1a9a89SRafael Vanoni Polanczyk 	/* If file's mapped, return */
263cc1a9a89SRafael Vanoni Polanczyk 	if (file_type != 0)
264cc1a9a89SRafael Vanoni Polanczyk 		return (1);
265cc1a9a89SRafael Vanoni Polanczyk 
266cc1a9a89SRafael Vanoni Polanczyk 	if ((fd = open(STATE_LOCATION, O_RDWR)) < 0) {
267cc1a9a89SRafael Vanoni Polanczyk 		file_type = TEMP_FILE;
268cc1a9a89SRafael Vanoni Polanczyk 
269cc1a9a89SRafael Vanoni Polanczyk 		if ((tmp = tmpfile()) == NULL)
2707c478bd9Sstevel@tonic-gate 			return (-1);
271cc1a9a89SRafael Vanoni Polanczyk 		else
272cc1a9a89SRafael Vanoni Polanczyk 			fd = fileno(tmp);
273cc1a9a89SRafael Vanoni Polanczyk 	} else {
274cc1a9a89SRafael Vanoni Polanczyk 		file_type = STATE_FILE;
2757c478bd9Sstevel@tonic-gate 	}
2767c478bd9Sstevel@tonic-gate 
277cc1a9a89SRafael Vanoni Polanczyk 	(void) ftruncate(fd, (off_t)sizeof (shared_buffer_t));
2787c478bd9Sstevel@tonic-gate 
279cc1a9a89SRafael Vanoni Polanczyk 	/* LINTED - alignment */
280cc1a9a89SRafael Vanoni Polanczyk 	data = (shared_buffer_t *)mmap(NULL, sizeof (shared_buffer_t),
281cc1a9a89SRafael Vanoni Polanczyk 	    PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
282cc1a9a89SRafael Vanoni Polanczyk 
283cc1a9a89SRafael Vanoni Polanczyk 	if (data == MAP_FAILED)
284cc1a9a89SRafael Vanoni Polanczyk 		return (-1);
285cc1a9a89SRafael Vanoni Polanczyk 
286cc1a9a89SRafael Vanoni Polanczyk 	(void) mutex_init(&data->lock, USYNC_PROCESS|LOCK_ROBUST, 0);
287cc1a9a89SRafael Vanoni Polanczyk 
288cc1a9a89SRafael Vanoni Polanczyk 	(void) close(fd);
289cc1a9a89SRafael Vanoni Polanczyk 
290cc1a9a89SRafael Vanoni Polanczyk 	return (1);
291cc1a9a89SRafael Vanoni Polanczyk }
292cc1a9a89SRafael Vanoni Polanczyk 
293cc1a9a89SRafael Vanoni Polanczyk static void
294cc1a9a89SRafael Vanoni Polanczyk revalidate_data(uuid_node_t *node)
295cc1a9a89SRafael Vanoni Polanczyk {
296cc1a9a89SRafael Vanoni Polanczyk 	int i;
297cc1a9a89SRafael Vanoni Polanczyk 
298cc1a9a89SRafael Vanoni Polanczyk 	data->state.ts = 0;
299cc1a9a89SRafael Vanoni Polanczyk 
300cc1a9a89SRafael Vanoni Polanczyk 	for (i = 0; i < sizeof (data->state.node.nodeID); i++)
301cc1a9a89SRafael Vanoni Polanczyk 		data->state.node.nodeID[i] = 0;
302cc1a9a89SRafael Vanoni Polanczyk 
303cc1a9a89SRafael Vanoni Polanczyk 	data->state.clock = 0;
304cc1a9a89SRafael Vanoni Polanczyk 
305cc1a9a89SRafael Vanoni Polanczyk 	gen_ethernet_address(node);
306cc1a9a89SRafael Vanoni Polanczyk 	bcopy(node, &node_id_cache, sizeof (uuid_node_t));
307cc1a9a89SRafael Vanoni Polanczyk 	node_init = 1;
308cc1a9a89SRafael Vanoni Polanczyk }
3097c478bd9Sstevel@tonic-gate 
3107c478bd9Sstevel@tonic-gate /*
311cc1a9a89SRafael Vanoni Polanczyk  * Prints a nicely-formatted uuid to stdout.
3127c478bd9Sstevel@tonic-gate  */
3137c478bd9Sstevel@tonic-gate void
3147c478bd9Sstevel@tonic-gate uuid_print(struct uuid u)
3157c478bd9Sstevel@tonic-gate {
3167c478bd9Sstevel@tonic-gate 	int i;
3177c478bd9Sstevel@tonic-gate 
3187c478bd9Sstevel@tonic-gate 	(void) printf("%8.8x-%4.4x-%4.4x-%2.2x%2.2x-", u.time_low, u.time_mid,
3197c478bd9Sstevel@tonic-gate 	    u.time_hi_and_version, u.clock_seq_hi_and_reserved,
3207c478bd9Sstevel@tonic-gate 	    u.clock_seq_low);
3217c478bd9Sstevel@tonic-gate 	for (i = 0; i < 6; i++)
3227c478bd9Sstevel@tonic-gate 		(void) printf("%2.2x", u.node_addr[i]);
3237c478bd9Sstevel@tonic-gate 	(void) printf("\n");
3247c478bd9Sstevel@tonic-gate }
3257c478bd9Sstevel@tonic-gate 
3267c478bd9Sstevel@tonic-gate /*
32784615402SJerry Jelinek  * Only called with urandmtx held.
32884615402SJerry Jelinek  * Fills/refills the cache of randomness. We know that our allocations of
32984615402SJerry Jelinek  * randomness are always much less than the total size of the cache.
33084615402SJerry Jelinek  * Tries to use /dev/urandom random number generator - if that fails for some
33184615402SJerry Jelinek  * reason, it retries MAX_RETRY times then sets rcachep to NULL so we no
33284615402SJerry Jelinek  * longer use the cache.
3337c478bd9Sstevel@tonic-gate  */
3347c478bd9Sstevel@tonic-gate static void
33584615402SJerry Jelinek load_cache()
3367c478bd9Sstevel@tonic-gate {
33784615402SJerry Jelinek 	int i, retries = 0;
33884615402SJerry Jelinek 	int nbytes = RCACHE_SIZE;
33984615402SJerry Jelinek 	char *buf = rcache;
3407c478bd9Sstevel@tonic-gate 
3417c478bd9Sstevel@tonic-gate 	while (nbytes > 0) {
34284615402SJerry Jelinek 		i = read(fd_urand, buf, nbytes);
3437c478bd9Sstevel@tonic-gate 		if ((i < 0) && (errno == EINTR)) {
3447c478bd9Sstevel@tonic-gate 			continue;
3457c478bd9Sstevel@tonic-gate 		}
3467c478bd9Sstevel@tonic-gate 		if (i <= 0) {
3477c478bd9Sstevel@tonic-gate 			if (retries++ == MAX_RETRY)
3487c478bd9Sstevel@tonic-gate 				break;
3497c478bd9Sstevel@tonic-gate 			continue;
3507c478bd9Sstevel@tonic-gate 		}
3517c478bd9Sstevel@tonic-gate 		nbytes -= i;
3527c478bd9Sstevel@tonic-gate 		buf += i;
3537c478bd9Sstevel@tonic-gate 		retries = 0;
3547c478bd9Sstevel@tonic-gate 	}
35584615402SJerry Jelinek 	if (nbytes == 0)
35684615402SJerry Jelinek 		rcachep = rcache;
35784615402SJerry Jelinek 	else
35884615402SJerry Jelinek 		rcachep = NULL;
35984615402SJerry Jelinek }
36084615402SJerry Jelinek 
36184615402SJerry Jelinek /*
36284615402SJerry Jelinek  * Fills buf with random numbers - nbytes is the number of bytes
36384615402SJerry Jelinek  * to fill-in. Tries to use cached data from the /dev/urandom random number
36484615402SJerry Jelinek  * generator - if that fails for some reason, it uses srand48(3C)
36584615402SJerry Jelinek  */
36684615402SJerry Jelinek static void
36784615402SJerry Jelinek fill_random_bytes(uchar_t *buf, int nbytes)
36884615402SJerry Jelinek {
36984615402SJerry Jelinek 	int i;
37084615402SJerry Jelinek 
37184615402SJerry Jelinek 	if (fd_urand == -1) {
37284615402SJerry Jelinek 		(void) mutex_lock(&urandmtx);
37384615402SJerry Jelinek 		/* check again now that we have the mutex */
37484615402SJerry Jelinek 		if (fd_urand == -1) {
37584615402SJerry Jelinek 			if ((fd_urand = open(URANDOM_PATH, O_RDONLY)) >= 0)
37684615402SJerry Jelinek 				load_cache();
37784615402SJerry Jelinek 		}
37884615402SJerry Jelinek 		(void) mutex_unlock(&urandmtx);
37984615402SJerry Jelinek 	}
38084615402SJerry Jelinek 	if (fd_urand >= 0 && rcachep != NULL) {
38184615402SJerry Jelinek 		int cnt;
38284615402SJerry Jelinek 
38384615402SJerry Jelinek 		(void) mutex_lock(&urandmtx);
38484615402SJerry Jelinek 		if (rcachep != NULL &&
38584615402SJerry Jelinek 		    (rcachep + nbytes) >= (rcache + RCACHE_SIZE))
38684615402SJerry Jelinek 			load_cache();
38784615402SJerry Jelinek 
38884615402SJerry Jelinek 		if (rcachep != NULL) {
38984615402SJerry Jelinek 			for (cnt = 0; cnt < nbytes; cnt++)
39084615402SJerry Jelinek 				*buf++ = *rcachep++;
39184615402SJerry Jelinek 			(void) mutex_unlock(&urandmtx);
3927c478bd9Sstevel@tonic-gate 			return;
3937c478bd9Sstevel@tonic-gate 		}
39484615402SJerry Jelinek 		(void) mutex_unlock(&urandmtx);
3957c478bd9Sstevel@tonic-gate 	}
3967c478bd9Sstevel@tonic-gate 	for (i = 0; i < nbytes; i++) {
397cc1a9a89SRafael Vanoni Polanczyk 		*buf++ = get_random() & 0xFF;
3987c478bd9Sstevel@tonic-gate 	}
3997c478bd9Sstevel@tonic-gate }
4007c478bd9Sstevel@tonic-gate 
4017c478bd9Sstevel@tonic-gate /*
402cc1a9a89SRafael Vanoni Polanczyk  * Unpacks the structure members in "struct uuid" to a char string "uuid_t".
4037c478bd9Sstevel@tonic-gate  */
4047c478bd9Sstevel@tonic-gate void
4057c478bd9Sstevel@tonic-gate struct_to_string(uuid_t ptr, struct uuid *uu)
4067c478bd9Sstevel@tonic-gate {
4077c478bd9Sstevel@tonic-gate 	uint_t		tmp;
4087c478bd9Sstevel@tonic-gate 	uchar_t		*out = ptr;
4097c478bd9Sstevel@tonic-gate 
4107c478bd9Sstevel@tonic-gate 	tmp = uu->time_low;
4117c478bd9Sstevel@tonic-gate 	out[3] = (uchar_t)tmp;
4127c478bd9Sstevel@tonic-gate 	tmp >>= 8;
4137c478bd9Sstevel@tonic-gate 	out[2] = (uchar_t)tmp;
4147c478bd9Sstevel@tonic-gate 	tmp >>= 8;
4157c478bd9Sstevel@tonic-gate 	out[1] = (uchar_t)tmp;
4167c478bd9Sstevel@tonic-gate 	tmp >>= 8;
4177c478bd9Sstevel@tonic-gate 	out[0] = (uchar_t)tmp;
4187c478bd9Sstevel@tonic-gate 
4197c478bd9Sstevel@tonic-gate 	tmp = uu->time_mid;
4207c478bd9Sstevel@tonic-gate 	out[5] = (uchar_t)tmp;
4217c478bd9Sstevel@tonic-gate 	tmp >>= 8;
4227c478bd9Sstevel@tonic-gate 	out[4] = (uchar_t)tmp;
4237c478bd9Sstevel@tonic-gate 
4247c478bd9Sstevel@tonic-gate 	tmp = uu->time_hi_and_version;
4257c478bd9Sstevel@tonic-gate 	out[7] = (uchar_t)tmp;
4267c478bd9Sstevel@tonic-gate 	tmp >>= 8;
4277c478bd9Sstevel@tonic-gate 	out[6] = (uchar_t)tmp;
4287c478bd9Sstevel@tonic-gate 
4297c478bd9Sstevel@tonic-gate 	tmp = uu->clock_seq_hi_and_reserved;
4307c478bd9Sstevel@tonic-gate 	out[8] = (uchar_t)tmp;
4317c478bd9Sstevel@tonic-gate 	tmp = uu->clock_seq_low;
4327c478bd9Sstevel@tonic-gate 	out[9] = (uchar_t)tmp;
4337c478bd9Sstevel@tonic-gate 
4347c478bd9Sstevel@tonic-gate 	(void) memcpy(out+10, uu->node_addr, 6);
4357c478bd9Sstevel@tonic-gate 
4367c478bd9Sstevel@tonic-gate }
4377c478bd9Sstevel@tonic-gate 
4387c478bd9Sstevel@tonic-gate /*
439cc1a9a89SRafael Vanoni Polanczyk  * Packs the values in the "uuid_t" string into "struct uuid".
4407c478bd9Sstevel@tonic-gate  */
4417c478bd9Sstevel@tonic-gate void
4427c478bd9Sstevel@tonic-gate string_to_struct(struct uuid *uuid, uuid_t in)
4437c478bd9Sstevel@tonic-gate {
4447c478bd9Sstevel@tonic-gate 	uchar_t	*ptr;
4457c478bd9Sstevel@tonic-gate 	uint_t	tmp;
4467c478bd9Sstevel@tonic-gate 
4477c478bd9Sstevel@tonic-gate 	ptr = in;
4487c478bd9Sstevel@tonic-gate 
4497c478bd9Sstevel@tonic-gate 	tmp = *ptr++;
4507c478bd9Sstevel@tonic-gate 	tmp = (tmp << 8) | *ptr++;
4517c478bd9Sstevel@tonic-gate 	tmp = (tmp << 8) | *ptr++;
4527c478bd9Sstevel@tonic-gate 	tmp = (tmp << 8) | *ptr++;
4537c478bd9Sstevel@tonic-gate 	uuid->time_low = tmp;
4547c478bd9Sstevel@tonic-gate 
4557c478bd9Sstevel@tonic-gate 	tmp = *ptr++;
4567c478bd9Sstevel@tonic-gate 	tmp = (tmp << 8) | *ptr++;
4577c478bd9Sstevel@tonic-gate 	uuid->time_mid = tmp;
4587c478bd9Sstevel@tonic-gate 
4597c478bd9Sstevel@tonic-gate 	tmp = *ptr++;
4607c478bd9Sstevel@tonic-gate 	tmp = (tmp << 8) | *ptr++;
4617c478bd9Sstevel@tonic-gate 	uuid->time_hi_and_version = tmp;
4627c478bd9Sstevel@tonic-gate 
4637c478bd9Sstevel@tonic-gate 	tmp = *ptr++;
4647c478bd9Sstevel@tonic-gate 	uuid->clock_seq_hi_and_reserved = tmp;
4657c478bd9Sstevel@tonic-gate 
4667c478bd9Sstevel@tonic-gate 	tmp = *ptr++;
4677c478bd9Sstevel@tonic-gate 	uuid->clock_seq_low = tmp;
4687c478bd9Sstevel@tonic-gate 
4697c478bd9Sstevel@tonic-gate 	(void) memcpy(uuid->node_addr, ptr, 6);
4707c478bd9Sstevel@tonic-gate 
4717c478bd9Sstevel@tonic-gate }
4727c478bd9Sstevel@tonic-gate 
4737c478bd9Sstevel@tonic-gate /*
474cc1a9a89SRafael Vanoni Polanczyk  * Generates UUID based on DCE Version 4
4757c478bd9Sstevel@tonic-gate  */
4767c478bd9Sstevel@tonic-gate void
4777c478bd9Sstevel@tonic-gate uuid_generate_random(uuid_t uu)
4787c478bd9Sstevel@tonic-gate {
4797c478bd9Sstevel@tonic-gate 	struct uuid	uuid;
4807c478bd9Sstevel@tonic-gate 
4817c478bd9Sstevel@tonic-gate 	if (uu == NULL)
4827c478bd9Sstevel@tonic-gate 		return;
4837c478bd9Sstevel@tonic-gate 
4847c478bd9Sstevel@tonic-gate 	(void) memset(uu, 0, sizeof (uuid_t));
4857c478bd9Sstevel@tonic-gate 	(void) memset(&uuid, 0, sizeof (struct uuid));
4867c478bd9Sstevel@tonic-gate 
4877c478bd9Sstevel@tonic-gate 	fill_random_bytes(uu, sizeof (uuid_t));
4887c478bd9Sstevel@tonic-gate 	string_to_struct(&uuid, uu);
4897c478bd9Sstevel@tonic-gate 	/*
4907c478bd9Sstevel@tonic-gate 	 * This is version 4, so say so in the UUID version field (4 bits)
4917c478bd9Sstevel@tonic-gate 	 */
4927c478bd9Sstevel@tonic-gate 	uuid.time_hi_and_version |= (1 << 14);
4937c478bd9Sstevel@tonic-gate 	/*
4947c478bd9Sstevel@tonic-gate 	 * we don't want the bit 1 to be set also which is for version 1
4957c478bd9Sstevel@tonic-gate 	 */
4967c478bd9Sstevel@tonic-gate 	uuid.time_hi_and_version &= VER1_MASK;
4977c478bd9Sstevel@tonic-gate 
4987c478bd9Sstevel@tonic-gate 	/*
4997c478bd9Sstevel@tonic-gate 	 * The variant for this format is the 2 high bits set to 10,
5007c478bd9Sstevel@tonic-gate 	 * so here it is
5017c478bd9Sstevel@tonic-gate 	 */
5027c478bd9Sstevel@tonic-gate 	uuid.clock_seq_hi_and_reserved |= 0x80;
5037c478bd9Sstevel@tonic-gate 
5047c478bd9Sstevel@tonic-gate 	/*
5057c478bd9Sstevel@tonic-gate 	 * Set MSB of Ethernet address to 1 to indicate that it was generated
5067c478bd9Sstevel@tonic-gate 	 * randomly
5077c478bd9Sstevel@tonic-gate 	 */
5087c478bd9Sstevel@tonic-gate 	uuid.node_addr[0] |= 0x80;
5097c478bd9Sstevel@tonic-gate 	struct_to_string(uu, &uuid);
5107c478bd9Sstevel@tonic-gate }
5117c478bd9Sstevel@tonic-gate 
5127c478bd9Sstevel@tonic-gate /*
513cc1a9a89SRafael Vanoni Polanczyk  * Generates UUID based on DCE Version 1.
5147c478bd9Sstevel@tonic-gate  */
5157c478bd9Sstevel@tonic-gate void
5167c478bd9Sstevel@tonic-gate uuid_generate_time(uuid_t uu)
5177c478bd9Sstevel@tonic-gate {
5187c478bd9Sstevel@tonic-gate 	struct 	uuid uuid;
5197c478bd9Sstevel@tonic-gate 
5207c478bd9Sstevel@tonic-gate 	if (uu == NULL)
5217c478bd9Sstevel@tonic-gate 		return;
5227c478bd9Sstevel@tonic-gate 
523cc1a9a89SRafael Vanoni Polanczyk 	if (uuid_create(&uuid) < 0) {
5247c478bd9Sstevel@tonic-gate 		uuid_generate_random(uu);
5257c478bd9Sstevel@tonic-gate 		return;
5267c478bd9Sstevel@tonic-gate 	}
527cc1a9a89SRafael Vanoni Polanczyk 
5287c478bd9Sstevel@tonic-gate 	struct_to_string(uu, &uuid);
5297c478bd9Sstevel@tonic-gate }
5307c478bd9Sstevel@tonic-gate 
5317c478bd9Sstevel@tonic-gate /*
532cc1a9a89SRafael Vanoni Polanczyk  * Creates a new UUID. The uuid will be generated based on high-quality
533cc1a9a89SRafael Vanoni Polanczyk  * randomness from /dev/urandom, if available by calling uuid_generate_random.
534cc1a9a89SRafael Vanoni Polanczyk  * If it failed to generate UUID then uuid_generate will call
535cc1a9a89SRafael Vanoni Polanczyk  * uuid_generate_time.
5367c478bd9Sstevel@tonic-gate  */
5377c478bd9Sstevel@tonic-gate void
5387c478bd9Sstevel@tonic-gate uuid_generate(uuid_t uu)
5397c478bd9Sstevel@tonic-gate {
5407c478bd9Sstevel@tonic-gate 	if (uu == NULL) {
5417c478bd9Sstevel@tonic-gate 		return;
5427c478bd9Sstevel@tonic-gate 	}
54384615402SJerry Jelinek 	if (fd_urand == -1) {
54484615402SJerry Jelinek 		(void) mutex_lock(&urandmtx);
54584615402SJerry Jelinek 		/* check again now that we have the mutex */
54684615402SJerry Jelinek 		if (fd_urand == -1) {
54784615402SJerry Jelinek 			if ((fd_urand = open(URANDOM_PATH, O_RDONLY)) >= 0)
54884615402SJerry Jelinek 				load_cache();
54984615402SJerry Jelinek 		}
55084615402SJerry Jelinek 		(void) mutex_unlock(&urandmtx);
55184615402SJerry Jelinek 	}
55284615402SJerry Jelinek 	if (fd_urand >= 0) {
5537c478bd9Sstevel@tonic-gate 		uuid_generate_random(uu);
5547c478bd9Sstevel@tonic-gate 	} else {
5557c478bd9Sstevel@tonic-gate 		(void) uuid_generate_time(uu);
5567c478bd9Sstevel@tonic-gate 	}
5577c478bd9Sstevel@tonic-gate }
5587c478bd9Sstevel@tonic-gate 
5597c478bd9Sstevel@tonic-gate /*
560cc1a9a89SRafael Vanoni Polanczyk  * Copies the UUID variable src to dst.
5617c478bd9Sstevel@tonic-gate  */
5627c478bd9Sstevel@tonic-gate void
5637c478bd9Sstevel@tonic-gate uuid_copy(uuid_t dst, uuid_t src)
5647c478bd9Sstevel@tonic-gate {
5657c478bd9Sstevel@tonic-gate 	(void) memcpy(dst, src, UUID_LEN);
5667c478bd9Sstevel@tonic-gate }
5677c478bd9Sstevel@tonic-gate 
5687c478bd9Sstevel@tonic-gate /*
569cc1a9a89SRafael Vanoni Polanczyk  * Sets the value of the supplied uuid variable uu, to the NULL value.
5707c478bd9Sstevel@tonic-gate  */
5717c478bd9Sstevel@tonic-gate void
5727c478bd9Sstevel@tonic-gate uuid_clear(uuid_t uu)
5737c478bd9Sstevel@tonic-gate {
5747c478bd9Sstevel@tonic-gate 	(void) memset(uu, 0, UUID_LEN);
5757c478bd9Sstevel@tonic-gate }
5767c478bd9Sstevel@tonic-gate 
5777c478bd9Sstevel@tonic-gate /*
578cc1a9a89SRafael Vanoni Polanczyk  * This function converts the supplied UUID uu from the internal
5797c478bd9Sstevel@tonic-gate  * binary format into a 36-byte string (plus trailing null char)
580cc1a9a89SRafael Vanoni Polanczyk  * and stores this value in the character string pointed to by out.
5817c478bd9Sstevel@tonic-gate  */
582*0b5ce10aSAndy Stormont static void
583*0b5ce10aSAndy Stormont uuid_unparse_common(uuid_t uu, char *out, boolean_t upper)
5847c478bd9Sstevel@tonic-gate {
5857c478bd9Sstevel@tonic-gate 	struct uuid 	uuid;
5867c478bd9Sstevel@tonic-gate 	uint16_t	clock_seq;
5877c478bd9Sstevel@tonic-gate 	char		etheraddr[13];
5887c478bd9Sstevel@tonic-gate 	int		index = 0, i;
5897c478bd9Sstevel@tonic-gate 
5907c478bd9Sstevel@tonic-gate 	/* basic sanity checking */
5917c478bd9Sstevel@tonic-gate 	if (uu == NULL) {
5927c478bd9Sstevel@tonic-gate 		return;
5937c478bd9Sstevel@tonic-gate 	}
5947c478bd9Sstevel@tonic-gate 
5957c478bd9Sstevel@tonic-gate 	string_to_struct(&uuid, uu);
5967c478bd9Sstevel@tonic-gate 	clock_seq = uuid.clock_seq_hi_and_reserved;
5977c478bd9Sstevel@tonic-gate 	clock_seq = (clock_seq  << 8) | uuid.clock_seq_low;
5987c478bd9Sstevel@tonic-gate 	for (i = 0; i < 6; i++) {
599*0b5ce10aSAndy Stormont 		(void) sprintf(&etheraddr[index++], upper ? "%.2X" : "%.2x",
600*0b5ce10aSAndy Stormont 		    uuid.node_addr[i]);
6017c478bd9Sstevel@tonic-gate 		index++;
6027c478bd9Sstevel@tonic-gate 	}
6037c478bd9Sstevel@tonic-gate 	etheraddr[index] = '\0';
6047c478bd9Sstevel@tonic-gate 
605*0b5ce10aSAndy Stormont 	(void) snprintf(out, 25,
606*0b5ce10aSAndy Stormont 	    upper ? "%08X-%04X-%04X-%04X-" : "%08x-%04x-%04x-%04x-",
607cc1a9a89SRafael Vanoni Polanczyk 	    uuid.time_low, uuid.time_mid, uuid.time_hi_and_version, clock_seq);
6086be356c5Sgz161490 	(void) strlcat(out, etheraddr, UUID_PRINTABLE_STRING_LENGTH);
6097c478bd9Sstevel@tonic-gate }
6107c478bd9Sstevel@tonic-gate 
611*0b5ce10aSAndy Stormont void
612*0b5ce10aSAndy Stormont uuid_unparse_upper(uuid_t uu, char *out)
613*0b5ce10aSAndy Stormont {
614*0b5ce10aSAndy Stormont 	uuid_unparse_common(uu, out, B_TRUE);
615*0b5ce10aSAndy Stormont }
616*0b5ce10aSAndy Stormont 
617*0b5ce10aSAndy Stormont void
618*0b5ce10aSAndy Stormont uuid_unparse_lower(uuid_t uu, char *out)
619*0b5ce10aSAndy Stormont {
620*0b5ce10aSAndy Stormont 	uuid_unparse_common(uu, out, B_FALSE);
621*0b5ce10aSAndy Stormont }
622*0b5ce10aSAndy Stormont 
623*0b5ce10aSAndy Stormont void
624*0b5ce10aSAndy Stormont uuid_unparse(uuid_t uu, char *out)
625*0b5ce10aSAndy Stormont {
626*0b5ce10aSAndy Stormont 	/*
627*0b5ce10aSAndy Stormont 	 * Historically uuid_unparse on Solaris returns lower case,
628*0b5ce10aSAndy Stormont 	 * for compatibility we preserve this behaviour.
629*0b5ce10aSAndy Stormont 	 */
630*0b5ce10aSAndy Stormont 	uuid_unparse_common(uu, out, B_FALSE);
631*0b5ce10aSAndy Stormont }
632*0b5ce10aSAndy Stormont 
6337c478bd9Sstevel@tonic-gate /*
634cc1a9a89SRafael Vanoni Polanczyk  * The uuid_is_null function compares the value of the supplied
6357c478bd9Sstevel@tonic-gate  * UUID variable uu to the NULL value. If the value is equal
6367c478bd9Sstevel@tonic-gate  * to the NULL UUID, 1 is returned, otherwise 0 is returned.
6377c478bd9Sstevel@tonic-gate  */
6387c478bd9Sstevel@tonic-gate int
6397c478bd9Sstevel@tonic-gate uuid_is_null(uuid_t uu)
6407c478bd9Sstevel@tonic-gate {
6417c478bd9Sstevel@tonic-gate 	int		i;
6427c478bd9Sstevel@tonic-gate 	uuid_t		null_uu;
6437c478bd9Sstevel@tonic-gate 
6447c478bd9Sstevel@tonic-gate 	(void) memset(null_uu, 0, sizeof (uuid_t));
6457c478bd9Sstevel@tonic-gate 	i = memcmp(uu, null_uu, sizeof (uuid_t));
6467c478bd9Sstevel@tonic-gate 	if (i == 0) {
6477c478bd9Sstevel@tonic-gate 		/* uu is NULL uuid */
6487c478bd9Sstevel@tonic-gate 		return (1);
6497c478bd9Sstevel@tonic-gate 	} else {
6507c478bd9Sstevel@tonic-gate 		return (0);
6517c478bd9Sstevel@tonic-gate 	}
6527c478bd9Sstevel@tonic-gate }
6537c478bd9Sstevel@tonic-gate 
6547c478bd9Sstevel@tonic-gate /*
655cc1a9a89SRafael Vanoni Polanczyk  * uuid_parse converts the UUID string given by 'in' into the
6567c478bd9Sstevel@tonic-gate  * internal uuid_t format. The input UUID is a string of the form
6577c478bd9Sstevel@tonic-gate  * cefa7a9c-1dd2-11b2-8350-880020adbeef in printf(3C) format.
6587c478bd9Sstevel@tonic-gate  * Upon successfully parsing the input string, UUID is stored
6597c478bd9Sstevel@tonic-gate  * in the location pointed to by uu
6607c478bd9Sstevel@tonic-gate  */
6617c478bd9Sstevel@tonic-gate int
6627c478bd9Sstevel@tonic-gate uuid_parse(char *in, uuid_t uu)
6637c478bd9Sstevel@tonic-gate {
6647c478bd9Sstevel@tonic-gate 
6657c478bd9Sstevel@tonic-gate 	char		*ptr, buf[3];
6667c478bd9Sstevel@tonic-gate 	int		i;
6677c478bd9Sstevel@tonic-gate 	struct uuid	uuid;
6687c478bd9Sstevel@tonic-gate 	uint16_t	clock_seq;
6697c478bd9Sstevel@tonic-gate 
6707c478bd9Sstevel@tonic-gate 	/* do some sanity checking */
6717c478bd9Sstevel@tonic-gate 	if ((strlen(in) != 36) || (uu == NULL) || (in[36] != '\0')) {
6727c478bd9Sstevel@tonic-gate 		return (-1);
6737c478bd9Sstevel@tonic-gate 	}
6747c478bd9Sstevel@tonic-gate 
6757c478bd9Sstevel@tonic-gate 	ptr = in;
6767c478bd9Sstevel@tonic-gate 	for (i = 0; i < 36; i++, ptr++) {
6777c478bd9Sstevel@tonic-gate 		if ((i == 8) || (i == 13) || (i == 18) || (i == 23)) {
6787c478bd9Sstevel@tonic-gate 			if (*ptr != '-') {
6797c478bd9Sstevel@tonic-gate 				return (-1);
6807c478bd9Sstevel@tonic-gate 			}
6817c478bd9Sstevel@tonic-gate 		} else {
6827c478bd9Sstevel@tonic-gate 			if (!isxdigit(*ptr)) {
6837c478bd9Sstevel@tonic-gate 				return (-1);
6847c478bd9Sstevel@tonic-gate 			}
6857c478bd9Sstevel@tonic-gate 		}
6867c478bd9Sstevel@tonic-gate 	}
6877c478bd9Sstevel@tonic-gate 
6887c478bd9Sstevel@tonic-gate 	uuid.time_low = strtoul(in, NULL, 16);
6897c478bd9Sstevel@tonic-gate 	uuid.time_mid = strtoul(in+9, NULL, 16);
6907c478bd9Sstevel@tonic-gate 	uuid.time_hi_and_version = strtoul(in+14, NULL, 16);
6917c478bd9Sstevel@tonic-gate 	clock_seq = strtoul(in+19, NULL, 16);
6927c478bd9Sstevel@tonic-gate 	uuid.clock_seq_hi_and_reserved = (clock_seq & 0xFF00) >> 8;
6937c478bd9Sstevel@tonic-gate 	uuid.clock_seq_low = (clock_seq & 0xFF);
6947c478bd9Sstevel@tonic-gate 
6957c478bd9Sstevel@tonic-gate 	ptr = in+24;
6967c478bd9Sstevel@tonic-gate 	buf[2] = '\0';
6977c478bd9Sstevel@tonic-gate 	for (i = 0; i < 6; i++) {
6987c478bd9Sstevel@tonic-gate 		buf[0] = *ptr++;
6997c478bd9Sstevel@tonic-gate 		buf[1] = *ptr++;
7007c478bd9Sstevel@tonic-gate 		uuid.node_addr[i] = strtoul(buf, NULL, 16);
7017c478bd9Sstevel@tonic-gate 	}
7027c478bd9Sstevel@tonic-gate 	struct_to_string(uu, &uuid);
7037c478bd9Sstevel@tonic-gate 	return (0);
7047c478bd9Sstevel@tonic-gate }
7057c478bd9Sstevel@tonic-gate 
7067c478bd9Sstevel@tonic-gate /*
707cc1a9a89SRafael Vanoni Polanczyk  * uuid_time extracts the time at which the supplied UUID uu
7087c478bd9Sstevel@tonic-gate  * was created. This function can only extract the creation
7097c478bd9Sstevel@tonic-gate  * time for UUIDs created with the uuid_generate_time function.
7107c478bd9Sstevel@tonic-gate  * The time at which the UUID was created, in seconds and
7117c478bd9Sstevel@tonic-gate  * microseconds since the epoch is stored in the location
7127c478bd9Sstevel@tonic-gate  * pointed to by ret_tv.
7137c478bd9Sstevel@tonic-gate  */
7147c478bd9Sstevel@tonic-gate time_t
7157c478bd9Sstevel@tonic-gate uuid_time(uuid_t uu, struct timeval *ret_tv)
7167c478bd9Sstevel@tonic-gate {
7177c478bd9Sstevel@tonic-gate 	struct uuid	uuid;
7187c478bd9Sstevel@tonic-gate 	uint_t		high;
7197c478bd9Sstevel@tonic-gate 	struct timeval	tv;
7207c478bd9Sstevel@tonic-gate 	u_longlong_t	clock_reg;
7217c478bd9Sstevel@tonic-gate 	uint_t		tmp;
7227c478bd9Sstevel@tonic-gate 	uint8_t		clk;
7237c478bd9Sstevel@tonic-gate 
7247c478bd9Sstevel@tonic-gate 	string_to_struct(&uuid, uu);
7257c478bd9Sstevel@tonic-gate 	tmp = (uuid.time_hi_and_version & 0xF000) >> 12;
7267c478bd9Sstevel@tonic-gate 	clk = uuid.clock_seq_hi_and_reserved;
7277c478bd9Sstevel@tonic-gate 
7287c478bd9Sstevel@tonic-gate 	/* check if uu is NULL, Version = 1 of DCE and Variant = 0b10x */
7297c478bd9Sstevel@tonic-gate 	if ((uu == NULL) || ((tmp & 0x01) != 0x01) || ((clk & 0x80) != 0x80)) {
7307c478bd9Sstevel@tonic-gate 		return (-1);
7317c478bd9Sstevel@tonic-gate 	}
7327c478bd9Sstevel@tonic-gate 	high = uuid.time_mid | ((uuid.time_hi_and_version & 0xFFF) << 16);
7337c478bd9Sstevel@tonic-gate 	clock_reg = uuid.time_low | ((u_longlong_t)high << 32);
7347c478bd9Sstevel@tonic-gate 
7357c478bd9Sstevel@tonic-gate 	clock_reg -= (((u_longlong_t)0x01B21DD2) << 32) + 0x13814000;
7367c478bd9Sstevel@tonic-gate 	tv.tv_sec = clock_reg / 10000000;
7377c478bd9Sstevel@tonic-gate 	tv.tv_usec = (clock_reg % 10000000) / 10;
7387c478bd9Sstevel@tonic-gate 
7397c478bd9Sstevel@tonic-gate 	if (ret_tv) {
7407c478bd9Sstevel@tonic-gate 		*ret_tv = tv;
7417c478bd9Sstevel@tonic-gate 	}
7427c478bd9Sstevel@tonic-gate 
7437c478bd9Sstevel@tonic-gate 	return (tv.tv_sec);
7447c478bd9Sstevel@tonic-gate }
745