mseaborn@chromium.org | 056521b | 2012-05-30 23:26:13 +0900 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "base/rand_util.h" |
| 6 | |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 7 | #include <errno.h> |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 8 | #include <fcntl.h> |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 9 | #include <unistd.h> |
| 10 | |
phajdan.jr@chromium.org | 2372593 | 2009-04-23 21:38:08 +0900 | [diff] [blame] | 11 | #include "base/file_util.h" |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 12 | #include "base/lazy_instance.h" |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 13 | #include "base/logging.h" |
| 14 | |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 15 | namespace { |
| 16 | |
| 17 | // We keep the file descriptor for /dev/urandom around so we don't need to |
| 18 | // reopen it (which is expensive), and since we may not even be able to reopen |
| 19 | // it if we are later put in a sandbox. This class wraps the file descriptor so |
| 20 | // we can use LazyInstance to handle opening it on the first access. |
| 21 | class URandomFd { |
| 22 | public: |
| 23 | URandomFd() { |
| 24 | fd_ = open("/dev/urandom", O_RDONLY); |
brettw@chromium.org | 5faed3c | 2011-10-27 06:48:00 +0900 | [diff] [blame] | 25 | DCHECK_GE(fd_, 0) << "Cannot open /dev/urandom: " << errno; |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 26 | } |
| 27 | |
| 28 | ~URandomFd() { |
| 29 | close(fd_); |
| 30 | } |
| 31 | |
| 32 | int fd() const { return fd_; } |
| 33 | |
| 34 | private: |
| 35 | int fd_; |
| 36 | }; |
| 37 | |
rvargas@chromium.org | 4abdbe8 | 2012-12-21 11:59:50 +0900 | [diff] [blame^] | 38 | base::LazyInstance<URandomFd>::Leaky g_urandom_fd = LAZY_INSTANCE_INITIALIZER; |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 39 | |
| 40 | } // namespace |
| 41 | |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 42 | namespace base { |
| 43 | |
mniknami@chromium.org | f4331ae | 2012-08-03 05:22:25 +0900 | [diff] [blame] | 44 | // NOTE: This function must be cryptographically secure. http://crbug.com/140076 |
deanm@chromium.org | 48c40ce | 2008-11-15 08:28:29 +0900 | [diff] [blame] | 45 | uint64 RandUint64() { |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 46 | uint64 number; |
| 47 | |
deanm@chromium.org | 2f749ea | 2009-06-26 19:00:02 +0900 | [diff] [blame] | 48 | int urandom_fd = g_urandom_fd.Pointer()->fd(); |
phajdan.jr@chromium.org | 2372593 | 2009-04-23 21:38:08 +0900 | [diff] [blame] | 49 | bool success = file_util::ReadFromFD(urandom_fd, |
| 50 | reinterpret_cast<char*>(&number), |
| 51 | sizeof(number)); |
| 52 | CHECK(success); |
mark@chromium.org | b93c054 | 2008-09-30 07:18:01 +0900 | [diff] [blame] | 53 | |
| 54 | return number; |
| 55 | } |
| 56 | |
agl@chromium.org | ba637ac | 2010-03-05 05:18:55 +0900 | [diff] [blame] | 57 | int GetUrandomFD(void) { |
| 58 | return g_urandom_fd.Pointer()->fd(); |
| 59 | } |
mseaborn@chromium.org | 056521b | 2012-05-30 23:26:13 +0900 | [diff] [blame] | 60 | |
| 61 | } // namespace base |