Files
infnoise/software/writeentropy.c
2018-04-21 17:22:37 +02:00

87 lines
2.6 KiB
C

// This writes entropy to the Linux /dev/random pool using ioctl, so that entropy increases.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/poll.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <linux/random.h>
#include "libinfnoise.h"
#define SIZE_PROC_FILENAME "/proc/sys/kernel/random/poolsize"
#define FILL_PROC_FILENAME "/proc/sys/kernel/random/write_wakeup_threshold"
static uint32_t inmBufLen;
static bool inmDebug;
static int inmDevRandomFD;
static uint32_t inmFillWatermark;
static struct rand_pool_info *inmPoolInfo;
// Find the entropy pool size.
static uint32_t readNumberFromFile(char *fileName) {
FILE *file = fopen(fileName, "r");
if(file == NULL) {
fprintf(stderr, "Unable to open %s\n", fileName);
exit(1);
}
char buf[42u];
uint32_t i = 0u;
int c = getc(file);
while(c != EOF) {
buf[i++] = c;
c = getc(file);
}
buf[i] = '\0';
uint32_t value = atoi(buf);
fclose(file);
return value;
}
// Open /dev/random
void inmWriteEntropyStart(uint32_t bufLen, bool debug) {
inmBufLen = bufLen;
inmDebug = debug;
//inmDevRandomFD = open("/dev/random", O_WRONLY);
inmDevRandomFD = open("/dev/random", O_RDWR);
if(inmDevRandomFD < 0) {
fprintf(stderr, "Unable to open /dev/random\n");
exit(1);
}
inmPoolInfo = calloc(1, sizeof(struct rand_pool_info) + bufLen);
if(inmPoolInfo == NULL) {
fprintf(stderr, "Unable to allocate memory\n");
exit(1);
}
inmFillWatermark = readNumberFromFile(FILL_PROC_FILENAME);
if(inmDebug) {
printf("Entropy pool size:%u, fill watermark:%u\n", readNumberFromFile(SIZE_PROC_FILENAME), inmFillWatermark);
}
}
// Block until either the entropy pool has room, or 1 minute has passed.
void inmWaitForPoolToHaveRoom() {
int ent_count;
struct pollfd pfd = {
.fd = inmDevRandomFD,
.events = POLLOUT,
};
if (ioctl(inmDevRandomFD, RNDGETENTCNT, &ent_count) == 0 && (uint32_t)ent_count < inmFillWatermark) {
return;
}
poll(&pfd, 1, 1000u * 60u); // One minute
}
// Add the bytes to the entropy pool. This can be unwhitenened, but the estimated bits of
// entropy needs to be accurate or pessimistic. Return false if the Linux entropy pool is
// full after writing.
void inmWriteEntropyToPool(uint8_t *bytes, uint32_t length, uint32_t entropy) {
inmPoolInfo->entropy_count = entropy;
inmPoolInfo->buf_size = length;
memcpy(inmPoolInfo->buf, bytes, length);
//printf("Writing %u bytes with %u bits of entropy to /dev/random\n", length, entropy);
ioctl(inmDevRandomFD, RNDADDENTROPY, inmPoolInfo);
}