/*- * Copyright (c) 2010, 2014 * Thorsten Glaser * * Provided that these terms and disclaimer and all copyright notices * are retained or reproduced in an accompanying document, permission * is granted to deal in this work without restriction, including un- * limited rights to use, publicly perform, distribute, sell, modify, * merge, give away, or sublicence. * * This work is provided "AS IS" and WITHOUT WARRANTY of any kind, to * the utmost extent permitted by applicable law, neither express nor * implied; without malicious intent or gross negligence. In no event * may a licensor, author or contributor be held liable for indirect, * direct, other damage, loss, or other issues arising in any way out * of dealing in the work, even if advised of the possibility of such * damage or existence of a defect, except proven that it results out * of said person's immediate fault when using the work as intended. *- * Implement a useful arc4random(3) related API. */ #include #include #include #include "arc4random.h" #include "thread_private.h" __RCSID("$MirOS: src/lib/libc/crypt/arc4random_buf.c,v 1.5 2014/03/29 10:35:46 tg Exp $"); void arc4random_buf(void *buf_, size_t len) { uint8_t *buf = (uint8_t *)buf_; pid_t mypid = 0; size_t n; /* reinitialise if necessary */ _ARC4_LOCK(); if (!a4state.a4s_initialised || (a4state.a4s_stir_pid != (mypid = getpid()))) arc4random_stir_locked(mypid); _ARC4_UNLOCK(); while (len) { _ARC4_LOCK(); /* randomly skip 1-4 bytes */ /*XXX this should be constant-time */ n = arcfour_byte(&a4state.cipher) & 3; while (n--) (void)arcfour_byte(&a4state.cipher); /* fill the buffer in small increments */ n = len < 256 ? len : 256; len -= n; a4state.a4s_count -= n; if (a4state.a4s_count < 0) arc4random_stir_locked(mypid); while (n--) *buf++ = arcfour_byte(&a4state.cipher); _ARC4_UNLOCK(); } }