forked from nhorman/rng-tools
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rngd_rtlsdr.c
206 lines (183 loc) · 5.65 KB
/
rngd_rtlsdr.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
/*
* Copyright (c) 2020, Neil Horman
* Author: Neil Horman <[email protected]>
*
* Entropy source to derive random data from atmospheric static obtained by
* randomly varying the frequency and sample rate of rtl software defined radios
* through the rtlsdr library.
*
* Based in part on the work found at:
* http://rtl2832-entropyd.sourceforge.net/
*/
#include <stdlib.h>
#include <string.h>
#include <rtl-sdr.h>
#include <openssl/conf.h>
#include <openssl/evp.h>
#include <openssl/err.h>
#include "rngd.h"
#include "ossl_helpers.h"
#define RAW_BUF_SZ 4096
#define CHUNK_SIZE (AES_BLOCK*8) /* 8 parallel streams */
static rtlsdr_dev_t *radio = NULL;
static unsigned char raw_buffera[RAW_BUF_SZ];
static int freq_min;
static int freq_max;
static int sample_min;
static int sample_max;
static int gain = 200;
/* AES data reduction key */
static unsigned char key[AES_BLOCK];
static unsigned char iv[CHUNK_SIZE];
static struct ossl_aes_ctx *ossl_ctx;
static int get_random_freq()
{
int range = freq_max - freq_min;
long int random_val = random();
random_val = random_val % range;
return freq_min + random_val;
}
static int get_random_sample_rate()
{
int range = sample_min - sample_max;
long int random_val = random();
random_val = random_val % range;
return sample_min + random_val;
}
static int nearest_gain(rtlsdr_dev_t *radio, struct rng *ent_src, int target_gain )
{
int i, err1, err2, count, close_gain;
int *gains;
count = rtlsdr_get_tuner_gains(radio, NULL);
if (count <= 0)
return 0;
gains = malloc(sizeof(int) * count);
count = rtlsdr_get_tuner_gains(radio, gains);
close_gain = gains[0];
message_entsrc(ent_src, LOG_DAEMON, "Capable Gains:\n");
for (i = 0; i < count; i++) {
message_entsrc(ent_src, LOG_DAEMON, "%d\n", gains[i]);
err1 = abs(target_gain - close_gain);
err2 = abs(target_gain - gains[i]);
if (err2 < err1)
close_gain = gains[i];
}
free (gains);
return close_gain;
}
int init_rtlsdr_entropy_source(struct rng *ent_src)
{
int devcount;
int i;
int devid;
char vendor[256];
char product[256];
int rc;
int sample_rate;
int freq;
devcount = rtlsdr_get_device_count();
if (devcount == 0) {
message_entsrc(ent_src,LOG_DAEMON|LOG_DEBUG, "No rtlsdr radio devices found\n");
return 1;
}
message_entsrc(ent_src,LOG_DAEMON, "rtlsdr devices found:\n");
for (i=0; i<devcount; i++) {
memset(vendor, 0, 256);
memset(product, 0, 256);
if (rtlsdr_get_device_usb_strings(i, vendor, product, NULL))
continue;
message_entsrc(ent_src, LOG_DAEMON, "%d: %s %s\n", i, vendor, product);
}
ossl_ctx = ossl_aes_init(key, iv);
if (!ossl_ctx) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to setup openssl\n");
return 1;
}
/*
* Get our default sample rate and freq settings, as well as the devid
* to use
*/
freq_min = ent_src->rng_options[RTLSDR_OPT_FREQ_MIN].int_val;
freq_max = ent_src->rng_options[RTLSDR_OPT_FREQ_MAX].int_val;
sample_min = ent_src->rng_options[RTLSDR_OPT_SRATE_MIN].int_val;
sample_max = ent_src->rng_options[RTLSDR_OPT_SRATE_MAX].int_val;
devid = ent_src->rng_options[RTLSDR_OPT_DEVID].int_val;
message_entsrc(ent_src, LOG_DAEMON, "Using device %d\n", devid);
rc = rtlsdr_open(&radio, devid);
if (rc) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to open radio at index %d: %d\n", i, rc);
return 1;
}
if (rtlsdr_set_tuner_gain_mode(radio, 1)) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to set manual gain mode\n");
return 1;
}
gain = nearest_gain(radio, ent_src, gain);
message_entsrc(ent_src, LOG_DAEMON, "Setting gain to %d\n", gain);
if (rtlsdr_set_tuner_gain(radio, gain)) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to set gain\n");
return 1;
}
sample_rate = get_random_sample_rate();
message_entsrc(ent_src, LOG_DAEMON, "Setting sample rate to %d\n", sample_rate);
if (rtlsdr_set_sample_rate(radio, sample_rate)) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to set sample rate\n");
return 1;
}
freq = get_random_freq();
message_entsrc(ent_src, LOG_DAEMON, "Setting frequency to %d\n", freq);
if (rtlsdr_set_center_freq(radio, freq)) {
message_entsrc(ent_src, LOG_DAEMON, "Failed to set frequency\n");
return 1;
}
rtlsdr_reset_buffer(radio);
return 0;
}
void close_rtlsdr_entropy_source(struct rng *ent_src)
{
if (radio)
rtlsdr_close(radio);
if (ossl_ctx)
ossl_aes_exit(ossl_ctx);
}
static size_t condition_buffer(unsigned char *in, unsigned char *out, size_t insize, size_t outsize)
{
/*
* Setup our key and iv
*/
memcpy(key, in, AES_BLOCK);
memcpy(iv, &in[AES_BLOCK], CHUNK_SIZE);
return ossl_aes_encrypt(ossl_ctx, in, insize, out);
}
int xread_rtlsdr(void *buf, size_t size, struct rng *ent_src)
{
int rc;
int read_len;
size_t gen_len;
char *buf_ptr = buf;
unsigned char outbuf[RAW_BUF_SZ + EVP_MAX_BLOCK_LENGTH];
size_t copy_size;
size_t total_size = 0;
while (total_size < size) {
if (rtlsdr_set_center_freq(radio, get_random_freq())) {
message_entsrc(ent_src, LOG_DAEMON|LOG_DEBUG, "Failed to adjust frequency\n");
return 1;
}
if (rtlsdr_set_sample_rate(radio, get_random_sample_rate())) {
message_entsrc(ent_src, LOG_DAEMON|LOG_DEBUG, "Failed to adjust sample rate\n");
return 1;
}
rtlsdr_reset_buffer(radio);
rc = rtlsdr_read_sync(radio, raw_buffera, RAW_BUF_SZ, &read_len);
if (rc) {
message_entsrc(ent_src, LOG_DAEMON|LOG_DEBUG, "Radio read failed (buffer b): %d\n", rc);
return 1;
}
gen_len = condition_buffer(raw_buffera, outbuf, RAW_BUF_SZ, RAW_BUF_SZ);
copy_size = (size - total_size) < gen_len ? (size - total_size) : gen_len;
memcpy(buf_ptr, outbuf, copy_size);
buf_ptr += copy_size;
total_size += copy_size;
}
return 0;
}