/* * Copyright (C) 2005-2008 Martin Willi * Copyright (C) 2005 Jan Hutter * Hochschule fuer Technik Rapperswil * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation; either version 2 of the License, or (at your * option) any later version. See . * * This program 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 General Public License * for more details. */ #include #include #include #include #include #include #include #include "random_rng.h" #ifndef DEV_RANDOM # define DEV_RANDOM "/dev/random" #endif #ifndef DEV_URANDOM # define DEV_URANDOM "/dev/urandom" #endif typedef struct private_random_rng_t private_random_rng_t; /** * Private data of an random_rng_t object. */ struct private_random_rng_t { /** * Public random_rng_t interface. */ random_rng_t public; /** * random device, depends on quality */ int dev; /** * file we read random bytes from */ char *file; }; METHOD(rng_t, get_bytes, void, private_random_rng_t *this, size_t bytes, u_int8_t *buffer) { size_t done; ssize_t got; done = 0; while (done < bytes) { got = read(this->dev, buffer + done, bytes - done); if (got <= 0) { DBG1(DBG_LIB, "reading from \"%s\" failed: %s, retrying...", this->file, strerror(errno)); close(this->dev); sleep(1); this->dev = open(this->file, 0); } done += got; } } METHOD(rng_t, allocate_bytes, void, private_random_rng_t *this, size_t bytes, chunk_t *chunk) { *chunk = chunk_alloc(bytes); get_bytes(this, chunk->len, chunk->ptr); } METHOD(rng_t, destroy, void, private_random_rng_t *this) { close(this->dev); free(this); } /* * Described in header. */ random_rng_t *random_rng_create(rng_quality_t quality) { private_random_rng_t *this; INIT(this, .public = { .rng = { .get_bytes = _get_bytes, .allocate_bytes = _allocate_bytes, .destroy = _destroy, }, }, ); if (quality == RNG_TRUE) { this->file = DEV_RANDOM; } else { this->file = DEV_URANDOM; } this->dev = open(this->file, 0); if (this->dev < 0) { DBG1(DBG_LIB, "opening \"%s\" failed: %s", this->file, strerror(errno)); free(this); return NULL; } return &this->public; }