| 
									
										
										
										
											2015-10-31 14:39:52 +09:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QEMU Crypto random number provider | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2015-2016 Red Hat, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu/osdep.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "crypto/random.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-26 00:15:01 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2017-05-16 07:24:39 +02:00
										 |  |  | #include <wincrypt.h>
 | 
					
						
							| 
									
										
										
										
											2017-04-26 00:15:01 +02:00
										 |  |  | static HCRYPTPROV hCryptProv; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | static int fd; /* a file handle to either /dev/urandom or /dev/random */ | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-07-21 10:37:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-26 00:15:01 +02:00
										 |  |  | int qcrypto_random_init(Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2016-07-21 10:37:14 +01:00
										 |  |  |     /* TBD perhaps also add support for BSD getentropy / Linux
 | 
					
						
							|  |  |  |      * getrandom syscalls directly */ | 
					
						
							|  |  |  |     fd = open("/dev/urandom", O_RDONLY); | 
					
						
							|  |  |  |     if (fd == -1 && errno == ENOENT) { | 
					
						
							|  |  |  |         fd = open("/dev/random", O_RDONLY); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fd < 0) { | 
					
						
							|  |  |  |         error_setg(errp, "No /dev/urandom or /dev/random found"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-04-26 00:15:01 +02:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     if (!CryptAcquireContext(&hCryptProv, NULL, NULL, PROV_RSA_FULL, | 
					
						
							|  |  |  |                              CRYPT_SILENT | CRYPT_VERIFYCONTEXT)) { | 
					
						
							|  |  |  |         error_setg_win32(errp, GetLastError(), | 
					
						
							|  |  |  |                          "Unable to create cryptographic provider"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qcrypto_random_bytes(uint8_t *buf G_GNUC_UNUSED, | 
					
						
							|  |  |  |                          size_t buflen G_GNUC_UNUSED, | 
					
						
							|  |  |  |                          Error **errp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |     int ret = -1; | 
					
						
							|  |  |  |     int got; | 
					
						
							| 
									
										
										
										
											2016-07-21 10:37:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     while (buflen > 0) { | 
					
						
							|  |  |  |         got = read(fd, buf, buflen); | 
					
						
							|  |  |  |         if (got < 0) { | 
					
						
							|  |  |  |             error_setg_errno(errp, errno, | 
					
						
							|  |  |  |                              "Unable to read random bytes"); | 
					
						
							|  |  |  |             goto cleanup; | 
					
						
							|  |  |  |         } else if (!got) { | 
					
						
							|  |  |  |             error_setg(errp, | 
					
						
							|  |  |  |                        "Unexpected EOF reading random bytes"); | 
					
						
							|  |  |  |             goto cleanup; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         buflen -= got; | 
					
						
							|  |  |  |         buf += got; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = 0; | 
					
						
							|  |  |  |  cleanup: | 
					
						
							|  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2017-04-26 00:15:01 +02:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     if (!CryptGenRandom(hCryptProv, buflen, buf)) { | 
					
						
							|  |  |  |         error_setg_win32(errp, GetLastError(), | 
					
						
							|  |  |  |                          "Unable to read random bytes"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-10-31 14:39:52 +09:00
										 |  |  | } |